use crate::stream::StreamReader;
use crate::{Error, Result};
use tor_cell::relaycell::msg::Resolved;
use tor_cell::relaycell::RelayCmd;
use tor_cell::restricted_msg;
use super::AnyCmdChecker;
pub struct ResolveStream {
s: StreamReader,
}
restricted_msg! {
enum ResolveResponseMsg : RelayMsg {
End,
Resolved,
}
}
impl ResolveStream {
pub(crate) fn new(s: StreamReader) -> Self {
ResolveStream { s }
}
pub async fn read_msg(&mut self) -> Result<Resolved> {
use ResolveResponseMsg::*;
let cell = self.s.recv().await?;
let msg = match cell.decode::<ResolveResponseMsg>() {
Ok(cell) => cell.into_msg(),
Err(e) => {
self.s.protocol_error();
return Err(Error::from_bytes_err(e, "response on a resolve stream"));
}
};
match msg {
End(e) => Err(Error::EndReceived(e.reason())),
Resolved(r) => Ok(r),
}
}
}
#[derive(Debug, Default)]
pub(crate) struct ResolveCmdChecker {}
impl super::CmdChecker for ResolveCmdChecker {
fn check_msg(
&mut self,
msg: &tor_cell::relaycell::UnparsedRelayMsg,
) -> Result<super::StreamStatus> {
use super::StreamStatus::Closed;
match msg.cmd() {
RelayCmd::RESOLVED => Ok(Closed),
RelayCmd::END => Ok(Closed),
_ => Err(Error::StreamProto(format!(
"Unexpected {} on resolve stream",
msg.cmd()
))),
}
}
fn consume_checked_msg(&mut self, msg: tor_cell::relaycell::UnparsedRelayMsg) -> Result<()> {
let _ = msg
.decode::<ResolveResponseMsg>()
.map_err(|err| Error::from_bytes_err(err, "message on resolve stream."))?;
Ok(())
}
}
impl ResolveCmdChecker {
pub(crate) fn new_any() -> AnyCmdChecker {
Box::<Self>::default()
}
}