xmpp-rs/src/lib.rs

130 lines
4 KiB
Rust
Raw Normal View History

2017-06-02 23:58:31 +00:00
#[macro_use]
2017-06-01 22:42:57 +00:00
extern crate futures;
extern crate tokio_core;
extern crate xml;
use std::net::SocketAddr;
use std::net::ToSocketAddrs;
use std::sync::Arc;
use std::io::ErrorKind;
2017-06-02 23:58:31 +00:00
use futures::{Future, BoxFuture, Sink, Poll, Async};
2017-06-01 22:42:57 +00:00
use futures::stream::{Stream, iter};
use futures::future::result;
use tokio_core::reactor::Handle;
use tokio_core::io::Io;
2017-06-02 23:58:31 +00:00
use tokio_core::net::{TcpStream, TcpStreamNew};
2017-06-01 22:42:57 +00:00
mod xmpp_codec;
use xmpp_codec::*;
// type FullClient = sasl::Client<StartTLS<TCPConnection>>
2017-06-02 23:58:31 +00:00
#[derive(Debug)]
pub struct TcpClient {
state: TcpClientState,
}
2017-06-01 22:42:57 +00:00
2017-06-02 23:58:31 +00:00
enum TcpClientState {
Connecting(TcpStreamNew),
SendStart(futures::sink::Send<XMPPStream<TcpStream>>),
RecvStart(Option<XMPPStream<TcpStream>>),
Established,
Invalid,
2017-06-01 22:42:57 +00:00
}
2017-06-02 23:58:31 +00:00
impl std::fmt::Debug for TcpClientState {
fn fmt(&self, fmt: &mut std::fmt::Formatter) -> Result<(), std::fmt::Error> {
let s = match *self {
TcpClientState::Connecting(_) => "Connecting",
TcpClientState::SendStart(_) => "SendStart",
TcpClientState::RecvStart(_) => "RecvStart",
TcpClientState::Established => "Established",
TcpClientState::Invalid => "Invalid",
};
write!(fmt, "{}", s)
2017-06-01 22:42:57 +00:00
}
2017-06-02 23:58:31 +00:00
}
2017-06-01 22:42:57 +00:00
2017-06-02 23:58:31 +00:00
impl TcpClient {
pub fn connect(addr: &SocketAddr, handle: &Handle) -> Self {
let tcp_stream_new = TcpStream::connect(addr, handle);
TcpClient {
state: TcpClientState::Connecting(tcp_stream_new),
}
2017-06-01 22:42:57 +00:00
}
}
2017-06-02 23:58:31 +00:00
impl Future for TcpClient {
type Item = XMPPStream<TcpStream>;
type Error = std::io::Error;
2017-06-01 22:42:57 +00:00
2017-06-02 23:58:31 +00:00
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
let (new_state, result) = match self.state {
TcpClientState::Connecting(ref mut tcp_stream_new) => {
let tcp_stream = try_ready!(tcp_stream_new.poll());
let xmpp_stream = tcp_stream.framed(XMPPCodec::new());
let send = xmpp_stream.send(Packet::StreamStart);
let new_state = TcpClientState::SendStart(send);
(new_state, Ok(Async::NotReady))
},
TcpClientState::SendStart(ref mut send) => {
let xmpp_stream = try_ready!(send.poll());
let new_state = TcpClientState::RecvStart(Some(xmpp_stream));
(new_state, Ok(Async::NotReady))
},
TcpClientState::RecvStart(ref mut opt_xmpp_stream) => {
let mut xmpp_stream = opt_xmpp_stream.take().unwrap();
match xmpp_stream.poll() {
Ok(Async::Ready(Some(events))) => println!("Recv start: {:?}", events),
Ok(Async::Ready(_)) => return Err(std::io::Error::from(ErrorKind::InvalidData)),
Ok(Async::NotReady) => {
*opt_xmpp_stream = Some(xmpp_stream);
return Ok(Async::NotReady);
},
Err(e) => return Err(e)
};
let new_state = TcpClientState::Established;
(new_state, Ok(Async::Ready(xmpp_stream)))
},
TcpClientState::Established | TcpClientState::Invalid =>
unreachable!(),
};
println!("Next state: {:?}", new_state);
self.state = new_state;
match result {
// by polling again, we register new future
Ok(Async::NotReady) => self.poll(),
result => result
}
2017-06-01 22:42:57 +00:00
}
}
#[cfg(test)]
mod tests {
use tokio_core::reactor::Core;
2017-06-02 23:58:31 +00:00
use futures::{Future, Stream};
2017-06-01 22:42:57 +00:00
#[test]
fn it_works() {
2017-06-02 23:58:31 +00:00
use std::net::ToSocketAddrs;
let addr = "[2a01:4f8:a0:33d0::5]:5222"
.to_socket_addrs().unwrap()
.next().unwrap();
2017-06-01 22:42:57 +00:00
let mut core = Core::new().unwrap();
2017-06-02 23:58:31 +00:00
let client = super::TcpClient::connect(
&addr,
2017-06-01 22:42:57 +00:00
&core.handle()
).and_then(|stream| {
2017-06-02 23:58:31 +00:00
stream.for_each(|item| {
2017-06-01 22:42:57 +00:00
Ok(println!("stream item: {:?}", item))
})
2017-06-02 23:58:31 +00:00
});
2017-06-01 22:42:57 +00:00
core.run(client).unwrap();
}
// TODO: test truncated utf8
}