mod codec; mod handshake; pub use self::codec::APCodec; pub use self::handshake::handshake; use futures::{Future, Sink, Stream}; use hyper::Uri; use protobuf::{self, Message}; use std::io; use std::net::ToSocketAddrs; use std::str::FromStr; use tokio_core::net::TcpStream; use tokio_core::reactor::Handle; use tokio_io::codec::Framed; use authentication::Credentials; use version; use proxytunnel; pub type Transport = Framed; pub fn connect( addr: A, handle: &Handle, proxy: &Option, ) -> Box> { let (addr, connect_url) = match *proxy { Some(ref url) => { let url = Uri::from_str(url).expect("Malformed proxy address"); let host = url.host().expect("Malformed proxy address: no host"); let port = url.port().unwrap_or(3128); ( format!("{}:{}", host, port) .to_socket_addrs() .unwrap() .next() .unwrap(), Some(addr.to_socket_addrs().unwrap().next().unwrap()), ) } None => (addr.to_socket_addrs().unwrap().next().unwrap(), None), }; let socket = TcpStream::connect(&addr, handle); if let Some(connect_url) = connect_url { let connection = socket.and_then(move |socket| proxytunnel::connect(socket, connect_url).and_then(handshake)); Box::new(connection) } else { let connection = socket.and_then(handshake); Box::new(connection) } } pub fn authenticate( transport: Transport, credentials: Credentials, device_id: String, ) -> Box> { use protocol::authentication::{APWelcome, ClientResponseEncrypted, CpuFamily, Os}; use protocol::keyexchange::APLoginFailed; let mut packet = ClientResponseEncrypted::new(); packet.mut_login_credentials().set_username(credentials.username); packet.mut_login_credentials().set_typ(credentials.auth_type); packet .mut_login_credentials() .set_auth_data(credentials.auth_data); packet.mut_system_info().set_cpu_family(CpuFamily::CPU_UNKNOWN); packet.mut_system_info().set_os(Os::OS_UNKNOWN); packet.mut_system_info().set_system_information_string(format!( "librespot_{}_{}", version::short_sha(), version::build_id() )); packet.mut_system_info().set_device_id(device_id); packet.set_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, data)) => { let error_data: APLoginFailed = protobuf::parse_from_bytes(data.as_ref()).unwrap(); panic!( "Authentication failed with reason: {:?}", error_data.get_error_code() ) } Some((cmd, _)) => panic!("Unexpected packet {:?}", cmd), None => panic!("EOF"), }), ) }