xmpp-rs/src/stream_start.rs

178 lines
6.7 KiB
Rust
Raw Normal View History

use std::mem::replace;
2018-09-06 15:46:06 +00:00
// use std::error::Error as StdError;
// use std::{fmt, io};
2017-07-18 20:54:10 +00:00
use futures::{Future, Async, Poll, Stream, sink, Sink};
use tokio_io::{AsyncRead, AsyncWrite};
2018-07-31 21:34:04 +00:00
use tokio_codec::Framed;
2017-06-13 23:55:56 +00:00
use jid::Jid;
2017-07-22 00:59:51 +00:00
use minidom::Element;
2018-09-06 15:46:06 +00:00
use xmpp_codec::{XMPPCodec, Packet};
2017-07-18 20:54:10 +00:00
use xmpp_stream::XMPPStream;
2018-09-06 15:46:06 +00:00
use {Error, ProtocolError};
const NS_XMPP_STREAM: &str = "http://etherx.jabber.org/streams";
pub struct StreamStart<S: AsyncWrite> {
state: StreamStartState<S>,
2017-06-13 23:55:56 +00:00
jid: Jid,
2017-07-18 23:02:45 +00:00
ns: String,
}
enum StreamStartState<S: AsyncWrite> {
SendStart(sink::Send<Framed<S, XMPPCodec>>),
RecvStart(Framed<S, XMPPCodec>),
2017-07-18 23:02:45 +00:00
RecvFeatures(Framed<S, XMPPCodec>, String),
Invalid,
}
impl<S: AsyncWrite> StreamStart<S> {
2017-07-18 23:02:45 +00:00
pub fn from_stream(stream: Framed<S, XMPPCodec>, jid: Jid, ns: String) -> Self {
2017-06-13 23:55:56 +00:00
let attrs = [("to".to_owned(), jid.domain.clone()),
("version".to_owned(), "1.0".to_owned()),
2017-07-18 23:02:45 +00:00
("xmlns".to_owned(), ns.clone()),
("xmlns:stream".to_owned(), NS_XMPP_STREAM.to_owned()),
].iter().cloned().collect();
let send = stream.send(Packet::StreamStart(attrs));
StreamStart {
state: StreamStartState::SendStart(send),
2017-06-13 23:55:56 +00:00
jid,
2017-07-18 23:02:45 +00:00
ns,
}
}
}
impl<S: AsyncRead + AsyncWrite> Future for StreamStart<S> {
type Item = XMPPStream<S>;
2018-09-06 15:46:06 +00:00
type Error = Error;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
let old_state = replace(&mut self.state, StreamStartState::Invalid);
let mut retry = false;
let (new_state, result) = match old_state {
StreamStartState::SendStart(mut send) =>
match send.poll() {
Ok(Async::Ready(stream)) => {
retry = true;
(StreamStartState::RecvStart(stream), Ok(Async::NotReady))
},
Ok(Async::NotReady) =>
(StreamStartState::SendStart(send), Ok(Async::NotReady)),
Err(e) =>
(StreamStartState::Invalid, Err(e.into())),
},
StreamStartState::RecvStart(mut stream) =>
match stream.poll() {
Ok(Async::Ready(Some(Packet::StreamStart(stream_attrs)))) => {
2017-07-18 23:02:45 +00:00
let stream_ns = match stream_attrs.get("xmlns") {
Some(ns) => ns.clone(),
None =>
2018-09-06 15:46:06 +00:00
return Err(ProtocolError::NoStreamNamespace.into()),
2017-07-18 23:02:45 +00:00
};
2017-07-22 00:59:51 +00:00
if self.ns == "jabber:client" {
retry = true;
// TODO: skip RecvFeatures for version < 1.0
(StreamStartState::RecvFeatures(stream, stream_ns), Ok(Async::NotReady))
} else {
let id = match stream_attrs.get("id") {
Some(id) => id.clone(),
None =>
2018-09-06 15:46:06 +00:00
return Err(ProtocolError::NoStreamId.into()),
2017-07-22 00:59:51 +00:00
};
// FIXME: huge hack, shouldnt be an element!
let stream = XMPPStream::new(self.jid.clone(), stream, self.ns.clone(), Element::builder(id).build());
(StreamStartState::Invalid, Ok(Async::Ready(stream)))
}
},
Ok(Async::Ready(_)) =>
2018-09-06 15:46:06 +00:00
return Err(ProtocolError::InvalidToken.into()),
Ok(Async::NotReady) =>
(StreamStartState::RecvStart(stream), Ok(Async::NotReady)),
Err(e) =>
2018-09-06 15:46:06 +00:00
return Err(ProtocolError::from(e).into()),
},
2017-07-18 23:02:45 +00:00
StreamStartState::RecvFeatures(mut stream, stream_ns) =>
match stream.poll() {
Ok(Async::Ready(Some(Packet::Stanza(stanza)))) =>
2017-08-14 01:56:08 +00:00
if stanza.is("features", NS_XMPP_STREAM) {
2017-07-18 23:02:45 +00:00
let stream = XMPPStream::new(self.jid.clone(), stream, self.ns.clone(), stanza);
2017-06-13 23:55:56 +00:00
(StreamStartState::Invalid, Ok(Async::Ready(stream)))
} else {
2017-07-18 23:02:45 +00:00
(StreamStartState::RecvFeatures(stream, stream_ns), Ok(Async::NotReady))
},
2017-07-20 22:19:08 +00:00
Ok(Async::Ready(_)) | Ok(Async::NotReady) =>
2017-07-18 23:02:45 +00:00
(StreamStartState::RecvFeatures(stream, stream_ns), Ok(Async::NotReady)),
Err(e) =>
2018-09-06 15:46:06 +00:00
return Err(ProtocolError::from(e).into()),
},
StreamStartState::Invalid =>
unreachable!(),
};
self.state = new_state;
if retry {
self.poll()
} else {
result
}
}
}
2018-09-06 15:46:06 +00:00
// #[derive(Debug)]
// pub enum StreamStartError {
// MissingStreamNs,
// MissingStreamId,
// Unexpected,
// Parser(ParserError),
// IO(io::Error),
// }
// impl From<io::Error> for StreamStartError {
// fn from(e: io::Error) -> Self {
// StreamStartError::IO(e)
// }
// }
// impl From<ParserError> for StreamStartError {
// fn from(e: ParserError) -> Self {
// match e {
// ParserError::IO(e) => StreamStartError::IO(e),
// _ => StreamStartError::Parser(e)
// }
// }
// }
// impl StdError for StreamStartError {
// fn description(&self) -> &str {
// match *self {
// StreamStartError::MissingStreamNs => "Missing stream namespace",
// StreamStartError::MissingStreamId => "Missing stream id",
// StreamStartError::Unexpected => "Unexpected",
// StreamStartError::Parser(ref pe) => pe.description(),
// StreamStartError::IO(ref ie) => ie.description(),
// }
// }
// fn cause(&self) -> Option<&StdError> {
// match *self {
// StreamStartError::Parser(ref pe) => pe.cause(),
// StreamStartError::IO(ref ie) => ie.cause(),
// _ => None,
// }
// }
// }
// impl fmt::Display for StreamStartError {
// fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
// match *self {
// StreamStartError::MissingStreamNs => write!(f, "Missing stream namespace"),
// StreamStartError::MissingStreamId => write!(f, "Missing stream id"),
// StreamStartError::Unexpected => write!(f, "Received unexpected data"),
// StreamStartError::Parser(ref pe) => write!(f, "{}", pe),
// StreamStartError::IO(ref ie) => write!(f, "{}", ie),
// }
// }
// }