1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
use super::ReadPDFError;
use crate::PdfParseParams;
use data_encoding::HEXUPPER;
use ring::digest::{Context, SHA256};
use std::fs;
use std::fs::File;
use std::io::{BufReader, Read};
use std::path::Path;

pub fn sha256_hash(file: &str) -> Result<String, ReadPDFError> {
    let input = File::open(file)?;
    let mut reader = BufReader::new(input);

    let mut context = Context::new(&SHA256);
    let mut buffer = Vec::new();
    // read the whole file
    reader.read_to_end(&mut buffer)?;
    context.update(&buffer);

    let digest = context.finish();
    let hash = HEXUPPER.encode(digest.as_ref());
    Ok(hash)
}

/// Store a copy of the pdf file in the path specified in the configuration.
/// Return the base name to be stored in the database.
pub fn store_pdf(
    pdf_file: &str,
    _hash: &str,
    config: &PdfParseParams,
) -> Result<String, ReadPDFError> {
    let path = Path::new(pdf_file);
    let name = path
        .file_name()
        .ok_or(ReadPDFError::NotFound("no valid file name"))?
        .to_string_lossy();
    let new_path = format!("{}/{}", &config.doc_path, name);
    fs::copy(pdf_file, &new_path)?;
    Ok(name.to_string())
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn test_hash() {
        let file = "LICENSE-MIT";
        let hash = sha256_hash(file).unwrap();
        assert_eq!(
            hash,
            "0998C58A8B2993EA0B3AA8EBAF260606A8F84F3C1005F060A72C814199BDD0BA".to_string()
        );
    }
}