use Error;
use protocol;
use super::Io;
use mio::tcp::{TcpStream, TcpListener};
use mio;
pub struct Connection
{
pub pi: Interpreter,
pub dtp: DataTransfer,
}
pub struct Interpreter
{
pub stream: TcpStream,
pub token: mio::Token,
}
pub enum DataTransfer
{
None,
Listening {
listener: TcpListener,
token: mio::Token,
},
Connecting {
stream: TcpStream,
token: mio::Token,
},
Connected {
stream: TcpStream,
token: mio::Token,
},
}
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
pub enum DataTransferMode
{
Active,
Passive { port: u16 },
}
impl Connection
{
pub fn send_command<C>(&mut self, command: &C) -> Result<(), Error>
where C: protocol::Command {
command.write(&mut self.pi.stream)?;
Ok(())
}
pub fn send_reply<R>(&mut self, reply: R) -> Result<(), Error>
where R: Into<protocol::Reply> {
let reply = reply.into();
reply.write(&mut self.pi.stream)?;
Ok(())
}
pub fn uses_token(&self, the_token: mio::Token) -> bool {
if self.pi.token == the_token { return true };
match self.dtp {
DataTransfer::None => false,
DataTransfer::Listening { ref token, .. } => *token == the_token,
DataTransfer::Connecting { ref token, .. } => *token == the_token,
DataTransfer::Connected { ref token, .. } => *token == the_token,
}
}
}
impl DataTransfer
{
pub fn bind(port: u16, io: &mut Io) -> Result<Self, Error> {
use std::net::ToSocketAddrs;
let addr = ("127.0.0.1", port).to_socket_addrs()?.next().unwrap();
let listener = TcpListener::bind(&addr)?;
let token = io.allocate_token();
io.poll.register(&listener, token, mio::Ready::readable(),
mio::PollOpt::edge())?;
Ok(DataTransfer::Listening {
listener: listener,
token: token,
})
}
pub fn connect() -> Result<Self, Error> {
unimplemented!();
}
}
impl Default for DataTransferMode
{
fn default() -> Self { DataTransferMode::Active }
}