66 lines
1.8 KiB
Rust
66 lines
1.8 KiB
Rust
|
use std::io;
|
||
|
use futures::{Future, Poll, Async};
|
||
|
use tokio_core::io::{read_exact, ReadExact};
|
||
|
use crypto::checksum;
|
||
|
use message::{Error, MessageHeader, MessageResult, Magic, Command};
|
||
|
use bytes::Bytes;
|
||
|
use io::{read_header, ReadHeader};
|
||
|
|
||
|
pub fn read_any_message<A>(a: A, magic: Magic) -> ReadAnyMessage<A> where A: io::Read {
|
||
|
ReadAnyMessage {
|
||
|
state: ReadAnyMessageState::ReadHeader(read_header(a, magic)),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub enum ReadAnyMessageState<A> {
|
||
|
ReadHeader(ReadHeader<A>),
|
||
|
ReadPayload {
|
||
|
header: MessageHeader,
|
||
|
future: ReadExact<A, Bytes>
|
||
|
},
|
||
|
Finished,
|
||
|
}
|
||
|
|
||
|
pub struct ReadAnyMessage<A> {
|
||
|
state: ReadAnyMessageState<A>,
|
||
|
}
|
||
|
|
||
|
impl<A> Future for ReadAnyMessage<A> where A: io::Read {
|
||
|
type Item = MessageResult<(Command, Bytes)>;
|
||
|
type Error = io::Error;
|
||
|
|
||
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
||
|
let (next, result) = match self.state {
|
||
|
ReadAnyMessageState::ReadHeader(ref mut header) => {
|
||
|
let (stream, header) = try_ready!(header.poll());
|
||
|
let header = match header {
|
||
|
Ok(header) => header,
|
||
|
Err(err) => return Ok(Err(err).into()),
|
||
|
};
|
||
|
let future = read_exact(stream, Bytes::new_with_len(header.len as usize));
|
||
|
let next = ReadAnyMessageState::ReadPayload {
|
||
|
header: header,
|
||
|
future: future,
|
||
|
};
|
||
|
(next, Async::NotReady)
|
||
|
},
|
||
|
ReadAnyMessageState::ReadPayload { ref mut header, ref mut future } => {
|
||
|
let (_stream, bytes) = try_ready!(future.poll());
|
||
|
if checksum(&bytes) != header.checksum {
|
||
|
return Ok(Err(Error::InvalidChecksum).into());
|
||
|
}
|
||
|
let next = ReadAnyMessageState::Finished;
|
||
|
(next, Ok((header.command.clone(), bytes)).into())
|
||
|
},
|
||
|
ReadAnyMessageState::Finished => panic!("poll ReadAnyMessage after it's done"),
|
||
|
};
|
||
|
|
||
|
self.state = next;
|
||
|
match result {
|
||
|
// by polling again, we register new future
|
||
|
Async::NotReady => self.poll(),
|
||
|
result => Ok(result)
|
||
|
}
|
||
|
}
|
||
|
}
|