mod codec; mod handshake; pub use self::codec::APCodec; pub use self::handshake::handshake; use futures::{Future, Sink, Stream}; use protobuf::{self, Message}; use std::io; use std::net::ToSocketAddrs; use tokio_core::net::TcpStream; use tokio_core::reactor::Handle; use tokio_io::codec::Framed; use authentication::Credentials; use version; pub type Transport = Framed; pub fn connect( addr: A, handle: &Handle, ) -> Box> { let addr = addr.to_socket_addrs().unwrap().next().unwrap(); let socket = TcpStream::connect(&addr, handle); let connection = socket.and_then(|socket| handshake(socket)); Box::new(connection) } pub fn authenticate( transport: Transport, credentials: Credentials, device_id: String, ) -> Box> { use protocol::authentication::{APWelcome, ClientResponseEncrypted, CpuFamily, Os}; let packet = protobuf_init!(ClientResponseEncrypted::new(), { login_credentials => { username: credentials.username, typ: credentials.auth_type, auth_data: credentials.auth_data, }, system_info => { cpu_family: CpuFamily::CPU_UNKNOWN, os: Os::OS_UNKNOWN, system_information_string: format!("librespot_{}_{}", version::short_sha(), version::build_id()), device_id: device_id, }, version_string: version::version_string(), }); let cmd = 0xab; let data = packet.write_to_bytes().unwrap(); Box::new( transport .send((cmd, data)) .and_then(|transport| transport.into_future().map_err(|(err, _stream)| err)) .and_then(|(packet, transport)| match packet { Some((0xac, data)) => { let welcome_data: APWelcome = protobuf::parse_from_bytes(data.as_ref()).unwrap(); let reusable_credentials = Credentials { username: welcome_data.get_canonical_username().to_owned(), auth_type: welcome_data.get_reusable_auth_credentials_type(), auth_data: welcome_data.get_reusable_auth_credentials().to_owned(), }; Ok((transport, reusable_credentials)) } Some((0xad, _)) => panic!("Authentication failed"), Some((cmd, _)) => panic!("Unexpected packet {:?}", cmd), None => panic!("EOF"), }), ) }