2021-12-17 02:46:46 +00:00
|
|
|
use sha2::{digest::FixedOutputReset, Digest};
|
2021-10-18 23:44:05 +00:00
|
|
|
use std::{
|
2022-09-24 22:18:53 +00:00
|
|
|
cell::RefCell,
|
2021-10-18 23:44:05 +00:00
|
|
|
pin::Pin,
|
2022-09-24 22:18:53 +00:00
|
|
|
rc::Rc,
|
2021-10-18 23:44:05 +00:00
|
|
|
task::{Context, Poll},
|
|
|
|
};
|
|
|
|
use tokio::io::{AsyncRead, ReadBuf};
|
|
|
|
|
|
|
|
pin_project_lite::pin_project! {
|
|
|
|
pub(crate) struct Hasher<I, D> {
|
|
|
|
#[pin]
|
|
|
|
inner: I,
|
|
|
|
|
2022-09-24 22:18:53 +00:00
|
|
|
hasher: Rc<RefCell<D>>,
|
2021-10-18 23:44:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<I, D> Hasher<I, D>
|
|
|
|
where
|
2021-12-17 02:46:46 +00:00
|
|
|
D: Digest + FixedOutputReset + Send + 'static,
|
2021-10-18 23:44:05 +00:00
|
|
|
{
|
|
|
|
pub(super) fn new(reader: I, digest: D) -> Self {
|
|
|
|
Hasher {
|
|
|
|
inner: reader,
|
2022-09-24 22:18:53 +00:00
|
|
|
hasher: Rc::new(RefCell::new(digest)),
|
2021-10-18 23:44:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-24 22:18:53 +00:00
|
|
|
pub(super) fn hasher(&self) -> Rc<RefCell<D>> {
|
|
|
|
Rc::clone(&self.hasher)
|
2021-10-18 23:44:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<I, D> AsyncRead for Hasher<I, D>
|
|
|
|
where
|
|
|
|
I: AsyncRead,
|
|
|
|
D: Digest,
|
|
|
|
{
|
|
|
|
fn poll_read(
|
|
|
|
mut self: Pin<&mut Self>,
|
|
|
|
cx: &mut Context<'_>,
|
|
|
|
buf: &mut ReadBuf<'_>,
|
|
|
|
) -> Poll<std::io::Result<()>> {
|
|
|
|
let this = self.as_mut().project();
|
|
|
|
|
|
|
|
let reader = this.inner;
|
|
|
|
let hasher = this.hasher;
|
|
|
|
|
|
|
|
let before_len = buf.filled().len();
|
|
|
|
let poll_res = reader.poll_read(cx, buf);
|
|
|
|
let after_len = buf.filled().len();
|
|
|
|
if after_len > before_len {
|
2022-09-24 22:18:53 +00:00
|
|
|
hasher
|
|
|
|
.borrow_mut()
|
|
|
|
.update(&buf.filled()[before_len..after_len]);
|
2021-10-18 23:44:05 +00:00
|
|
|
}
|
|
|
|
poll_res
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use super::Hasher;
|
|
|
|
use sha2::{Digest, Sha256};
|
|
|
|
use std::io::Read;
|
|
|
|
|
|
|
|
macro_rules! test_on_arbiter {
|
|
|
|
($fut:expr) => {
|
|
|
|
actix_rt::System::new().block_on(async move {
|
|
|
|
let arbiter = actix_rt::Arbiter::new();
|
|
|
|
|
2022-04-07 17:56:40 +00:00
|
|
|
let (tx, rx) = tracing::trace_span!(parent: None, "Create channel")
|
|
|
|
.in_scope(|| tokio::sync::oneshot::channel());
|
2021-10-18 23:44:05 +00:00
|
|
|
|
|
|
|
arbiter.spawn(async move {
|
|
|
|
let handle = actix_rt::spawn($fut);
|
|
|
|
|
|
|
|
let _ = tx.send(handle.await.unwrap());
|
|
|
|
});
|
|
|
|
|
|
|
|
rx.await.unwrap()
|
|
|
|
})
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn hasher_works() {
|
|
|
|
let hash = test_on_arbiter!(async move {
|
|
|
|
let file1 = tokio::fs::File::open("./client-examples/earth.gif").await?;
|
|
|
|
|
2022-09-25 14:09:05 +00:00
|
|
|
let mut reader = Hasher::new(file1, Sha256::new());
|
2021-10-18 23:44:05 +00:00
|
|
|
|
2022-09-25 14:09:05 +00:00
|
|
|
tokio::io::copy(&mut reader, &mut tokio::io::sink()).await?;
|
2021-10-18 23:44:05 +00:00
|
|
|
|
2022-09-25 14:09:05 +00:00
|
|
|
Ok(reader.hasher().borrow_mut().finalize_reset().to_vec()) as std::io::Result<_>
|
2021-10-18 23:44:05 +00:00
|
|
|
})
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let mut file = std::fs::File::open("./client-examples/earth.gif").unwrap();
|
|
|
|
let mut vec = Vec::new();
|
|
|
|
file.read_to_end(&mut vec).unwrap();
|
|
|
|
let mut hasher = Sha256::new();
|
|
|
|
hasher.update(vec);
|
|
|
|
let correct_hash = hasher.finalize_reset().to_vec();
|
|
|
|
|
2022-04-02 21:44:03 +00:00
|
|
|
assert_eq!(hash, correct_hash);
|
2021-10-18 23:44:05 +00:00
|
|
|
}
|
|
|
|
}
|