librespot/core/src/connection/handshake.rs

209 lines
6.5 KiB
Rust
Raw Normal View History

2018-02-11 11:37:08 +00:00
use byteorder::{BigEndian, ByteOrder, WriteBytesExt};
2017-01-18 05:57:36 +00:00
use crypto::hmac::Hmac;
2018-02-11 11:37:08 +00:00
use crypto::mac::Mac;
use crypto::sha1::Sha1;
use futures::{Async, Future, Poll};
use protobuf::{self, Message};
2017-01-18 05:57:36 +00:00
use rand::thread_rng;
use std::io::{self, Read};
2017-01-18 05:57:36 +00:00
use std::marker::PhantomData;
2019-03-20 15:46:44 +00:00
use tokio_codec::{Decoder, Framed};
2018-02-11 11:37:08 +00:00
use tokio_io::io::{read_exact, write_all, ReadExact, Window, WriteAll};
use tokio_io::{AsyncRead, AsyncWrite};
2017-01-18 05:57:36 +00:00
2018-02-11 11:37:08 +00:00
use super::codec::APCodec;
2017-01-18 05:57:36 +00:00
use diffie_hellman::DHLocalKeys;
use protocol;
2018-02-11 11:37:08 +00:00
use protocol::keyexchange::{APResponseMessage, ClientHello, ClientResponsePlaintext};
2017-01-18 05:57:36 +00:00
use util;
pub struct Handshake<T> {
keys: DHLocalKeys,
state: HandshakeState<T>,
}
enum HandshakeState<T> {
ClientHello(WriteAll<T, Vec<u8>>),
APResponse(RecvPacket<T, APResponseMessage>),
ClientResponse(Option<APCodec>, WriteAll<T, Vec<u8>>),
}
pub fn handshake<T: AsyncRead + AsyncWrite>(connection: T) -> Handshake<T> {
2017-01-18 05:57:36 +00:00
let local_keys = DHLocalKeys::random(&mut thread_rng());
let client_hello = client_hello(connection, local_keys.public_key());
Handshake {
keys: local_keys,
state: HandshakeState::ClientHello(client_hello),
}
}
2018-02-11 11:37:08 +00:00
impl<T: AsyncRead + AsyncWrite> Future for Handshake<T> {
2017-01-18 05:57:36 +00:00
type Item = Framed<T, APCodec>;
type Error = io::Error;
fn poll(&mut self) -> Poll<Self::Item, io::Error> {
use self::HandshakeState::*;
loop {
self.state = match self.state {
ClientHello(ref mut write) => {
let (connection, accumulator) = try_ready!(write.poll());
2018-02-11 11:37:08 +00:00
2017-01-18 05:57:36 +00:00
let read = recv_packet(connection, accumulator);
APResponse(read)
}
APResponse(ref mut read) => {
let (connection, message, accumulator) = try_ready!(read.poll());
2018-02-11 11:37:08 +00:00
let remote_key = message
.get_challenge()
2017-01-18 05:57:36 +00:00
.get_login_crypto_challenge()
.get_diffie_hellman()
.get_gs()
.to_owned();
let shared_secret = self.keys.shared_secret(&remote_key);
2018-02-11 11:37:08 +00:00
let (challenge, send_key, recv_key) = compute_keys(&shared_secret, &accumulator);
2017-01-18 05:57:36 +00:00
let codec = APCodec::new(&send_key, &recv_key);
let write = client_response(connection, challenge);
ClientResponse(Some(codec), write)
}
ClientResponse(ref mut codec, ref mut write) => {
let (connection, _) = try_ready!(write.poll());
let codec = codec.take().unwrap();
2019-03-20 15:46:44 +00:00
let framed = codec.framed(connection);
2017-01-18 05:57:36 +00:00
return Ok(Async::Ready(framed));
}
}
}
}
}
fn client_hello<T: AsyncWrite>(connection: T, gc: Vec<u8>) -> WriteAll<T, Vec<u8>> {
let mut packet = ClientHello::new();
packet
.mut_build_info()
.set_product(protocol::keyexchange::Product::PRODUCT_PARTNER);
packet
.mut_build_info()
.set_platform(protocol::keyexchange::Platform::PLATFORM_LINUX_X86);
packet.mut_build_info().set_version(0x10800000000);
packet
.mut_cryptosuites_supported()
.push(protocol::keyexchange::Cryptosuite::CRYPTO_SUITE_SHANNON);
packet.mut_login_crypto_hello().mut_diffie_hellman().set_gc(gc);
packet
.mut_login_crypto_hello()
.mut_diffie_hellman()
.set_server_keys_known(1);
packet.set_client_nonce(util::rand_vec(&mut thread_rng(), 0x10));
packet.set_padding(vec![0x1e]);
2017-01-18 05:57:36 +00:00
let mut buffer = vec![0, 4];
let size = 2 + 4 + packet.compute_size();
buffer.write_u32::<BigEndian>(size).unwrap();
packet.write_to_vec(&mut buffer).unwrap();
write_all(connection, buffer)
}
fn client_response<T: AsyncWrite>(connection: T, challenge: Vec<u8>) -> WriteAll<T, Vec<u8>> {
let mut packet = ClientResponsePlaintext::new();
packet
.mut_login_crypto_response()
.mut_diffie_hellman()
.set_hmac(challenge);
packet.mut_pow_response();
packet.mut_crypto_response();
2017-01-18 05:57:36 +00:00
let mut buffer = vec![];
let size = 4 + packet.compute_size();
buffer.write_u32::<BigEndian>(size).unwrap();
packet.write_to_vec(&mut buffer).unwrap();
write_all(connection, buffer)
}
enum RecvPacket<T, M: Message> {
2017-01-18 05:57:36 +00:00
Header(ReadExact<T, Window<Vec<u8>>>, PhantomData<M>),
Body(ReadExact<T, Window<Vec<u8>>>, PhantomData<M>),
}
fn recv_packet<T: AsyncRead, M>(connection: T, acc: Vec<u8>) -> RecvPacket<T, M>
2018-02-11 11:37:08 +00:00
where
T: Read,
M: Message,
2017-01-18 05:57:36 +00:00
{
RecvPacket::Header(read_into_accumulator(connection, 4, acc), PhantomData)
}
2018-02-11 11:37:08 +00:00
impl<T: AsyncRead, M> Future for RecvPacket<T, M>
where
T: Read,
M: Message,
2017-01-18 05:57:36 +00:00
{
type Item = (T, M, Vec<u8>);
type Error = io::Error;
fn poll(&mut self) -> Poll<Self::Item, io::Error> {
use self::RecvPacket::*;
loop {
*self = match *self {
Header(ref mut read, _) => {
let (connection, header) = try_ready!(read.poll());
let size = BigEndian::read_u32(header.as_ref()) as usize;
let acc = header.into_inner();
let read = read_into_accumulator(connection, size - 4, acc);
RecvPacket::Body(read, PhantomData)
}
Body(ref mut read, _) => {
let (connection, data) = try_ready!(read.poll());
let message = protobuf::parse_from_bytes(data.as_ref()).unwrap();
let acc = data.into_inner();
return Ok(Async::Ready((connection, message, acc)));
}
}
}
}
}
2018-02-11 11:37:08 +00:00
fn read_into_accumulator<T: AsyncRead>(
connection: T,
size: usize,
mut acc: Vec<u8>,
) -> ReadExact<T, Window<Vec<u8>>> {
2017-01-18 05:57:36 +00:00
let offset = acc.len();
acc.resize(offset + size, 0);
let mut window = Window::new(acc);
window.set_start(offset);
read_exact(connection, window)
}
fn compute_keys(shared_secret: &[u8], packets: &[u8]) -> (Vec<u8>, Vec<u8>, Vec<u8>) {
let mut data = Vec::with_capacity(0x64);
let mut mac = Hmac::new(Sha1::new(), &shared_secret);
for i in 1..6 {
mac.input(packets);
mac.input(&[i]);
data.extend_from_slice(&mac.result().code());
mac.reset();
}
mac = Hmac::new(Sha1::new(), &data[..0x14]);
mac.input(packets);
2018-02-11 11:37:08 +00:00
(
mac.result().code().to_vec(),
data[0x14..0x34].to_vec(),
data[0x34..0x54].to_vec(),
)
2017-01-18 05:57:36 +00:00
}