src/backup/index.rs: switch to async

Signed-off-by: Wolfgang Bumiller <w.bumiller@proxmox.com>
This commit is contained in:
Wolfgang Bumiller 2019-08-23 14:20:18 +02:00
parent 32bef1e2d1
commit 745e652a7f
1 changed files with 34 additions and 26 deletions

View File

@ -1,4 +1,6 @@
use std::collections::HashMap; use std::collections::HashMap;
use std::pin::Pin;
use std::task::{Context, Poll};
use bytes::{Bytes, BytesMut}; use bytes::{Bytes, BytesMut};
use failure::*; use failure::*;
@ -93,55 +95,61 @@ impl std::io::Read for DigestListEncoder {
/// ///
/// The reader simply returns a birary stream of 32 byte digest values. /// The reader simply returns a birary stream of 32 byte digest values.
pub struct DigestListDecoder<S> { pub struct DigestListDecoder<S: Unpin> {
input: S, input: S,
buffer: BytesMut, buffer: BytesMut,
} }
impl <S> DigestListDecoder<S> { impl<S: Unpin> DigestListDecoder<S> {
pub fn new(input: S) -> Self { pub fn new(input: S) -> Self {
Self { input, buffer: BytesMut::new() } Self { input, buffer: BytesMut::new() }
} }
} }
impl <S> Stream for DigestListDecoder<S> impl<S: Unpin> Unpin for DigestListDecoder<S> {}
where S: Stream<Item=Bytes>,
S::Error: Into<Error>, impl<S: Unpin, E> Stream for DigestListDecoder<S>
where
S: Stream<Item=Result<Bytes, E>>,
E: Into<Error>,
{ {
type Item = [u8; 32]; type Item = Result<[u8; 32], Error>;
type Error = Error;
fn poll_next(self: Pin<&mut Self>, cx: &mut Context) -> Poll<Option<Self::Item>> {
let this = self.get_mut();
fn poll(&mut self) -> Result<Async<Option<Self::Item>>, Self::Error> {
loop { loop {
if this.buffer.len() >= 32 {
if self.buffer.len() >= 32 { let left = this.buffer.split_to(32);
let left = self.buffer.split_to(32);
let mut digest = std::mem::MaybeUninit::<[u8; 32]>::uninit(); let mut digest = std::mem::MaybeUninit::<[u8; 32]>::uninit();
unsafe { unsafe {
(*digest.as_mut_ptr()).copy_from_slice(&left[..]); (*digest.as_mut_ptr()).copy_from_slice(&left[..]);
return Ok(Async::Ready(Some(digest.assume_init()))); return Poll::Ready(Some(Ok(digest.assume_init())));
} }
} }
match self.input.poll() { match Pin::new(&mut this.input).poll_next(cx) {
Err(err) => { Poll::Pending => {
return Err(err.into()); return Poll::Pending;
} }
Ok(Async::NotReady) => { Poll::Ready(Some(Err(err))) => {
return Ok(Async::NotReady); return Poll::Ready(Some(Err(err.into())));
} }
Ok(Async::Ready(None)) => { Poll::Ready(Some(Ok(data))) => {
let rest = self.buffer.len(); this.buffer.extend_from_slice(&data);
if rest == 0 { return Ok(Async::Ready(None)); }
return Err(format_err!("got small digest ({} != 32).", rest));
}
Ok(Async::Ready(Some(data))) => {
self.buffer.extend_from_slice(&data);
// continue // continue
} }
Poll::Ready(None) => {
let rest = this.buffer.len();
if rest == 0 {
return Poll::Ready(None);
}
return Poll::Ready(Some(Err(format_err!(
"got small digest ({} != 32).",
rest,
))));
}
} }
} }
} }