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
60
61
62
63
64
65
66
67
68
69
70
71
use sha2::{digest::Output, Digest, Sha256};
use std::io::{self, Read, Write};
pub struct HashReader<R: Read> {
reader: R,
hasher: Sha256,
}
impl<R: Read> HashReader<R> {
pub fn new(reader: R) -> Self {
HashReader {
reader,
hasher: Sha256::new(),
}
}
pub fn into_base_reader(self) -> R {
self.reader
}
pub fn into_hash(self) -> Output<Sha256> {
Sha256::digest(&self.hasher.finalize())
}
}
impl<R: Read> Read for HashReader<R> {
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
let bytes = self.reader.read(buf)?;
if bytes > 0 {
self.hasher.update(&buf[0..bytes]);
}
Ok(bytes)
}
}
pub struct HashWriter {
hasher: Sha256,
}
impl Default for HashWriter {
fn default() -> Self {
HashWriter {
hasher: Sha256::new(),
}
}
}
impl HashWriter {
pub fn into_hash(self) -> Output<Sha256> {
Sha256::digest(&self.hasher.finalize())
}
}
impl Write for HashWriter {
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
self.hasher.update(&buf);
Ok(buf.len())
}
fn flush(&mut self) -> io::Result<()> {
Ok(())
}
}