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
56
57
58
59
use crate::pack::data::File;
use git_features::progress::Progress;
use git_hash::SIZE_OF_SHA1_DIGEST as SHA1_SIZE;
#[derive(thiserror::Error, Debug)]
#[allow(missing_docs)]
pub enum Error {
#[error("pack checksum mismatch: expected {expected}, got {actual}")]
Mismatch {
expected: git_hash::ObjectId,
actual: git_hash::ObjectId,
},
#[error("could not read pack file")]
Io(#[from] std::io::Error),
}
impl File {
pub fn checksum(&self) -> git_hash::ObjectId {
git_hash::ObjectId::from_20_bytes(&self.data[self.data.len() - SHA1_SIZE..])
}
pub fn verify_checksum(&self, mut progress: impl Progress) -> Result<git_hash::ObjectId, Error> {
let right_before_trailer = self.data.len() - SHA1_SIZE;
let actual = match git_features::hash::bytes_of_file(
&self.path,
right_before_trailer,
git_hash::Kind::Sha1,
&mut progress,
) {
Ok(id) => id,
Err(_io_err) => {
let start = std::time::Instant::now();
let mut hasher = git_features::hash::Sha1::default();
hasher.update(&self.data[..right_before_trailer]);
progress.inc_by(right_before_trailer);
progress.show_throughput(start);
git_hash::ObjectId::new_sha1(hasher.digest())
}
};
let expected = self.checksum();
if actual == expected {
Ok(actual)
} else {
Err(Error::Mismatch { actual, expected })
}
}
}