more client

This commit is contained in:
nora 2024-08-15 21:37:11 +02:00
parent ff06ea5c72
commit 85a27baaed
4 changed files with 145 additions and 101 deletions

View file

@ -3,7 +3,10 @@ use std::{collections::VecDeque, mem};
use tracing::{debug, info, trace}; use tracing::{debug, info, trace};
use crate::{ use crate::{
crypto::{self, AlgorithmName, AlgorithmNegotiation}, crypto::{
self, AlgorithmName, AlgorithmNegotiation, EncryptionAlgorithm, HostKeySigningAlgorithm,
KeyExchangeSecret, SupportedAlgorithms,
},
numbers, numbers,
packet::{Packet, PacketTransport, ProtocolIdentParser}, packet::{Packet, PacketTransport, ProtocolIdentParser},
parse::{NameList, Parser, Writer}, parse::{NameList, Parser, Writer},
@ -25,6 +28,15 @@ enum ClientState {
}, },
KexInit { KexInit {
client_ident: Vec<u8>, client_ident: Vec<u8>,
server_ident: Vec<u8>,
},
DhKeyInit {
client_ident: Vec<u8>,
server_ident: Vec<u8>,
kex_secret: Option<KeyExchangeSecret>,
server_hostkey_algorithm: HostKeySigningAlgorithm,
encryption_client_to_server: EncryptionAlgorithm,
encryption_server_to_client: EncryptionAlgorithm,
}, },
} }
@ -54,10 +66,10 @@ impl ClientConnection {
} = &mut self.state } = &mut self.state
{ {
ident_parser.recv_bytes(bytes); ident_parser.recv_bytes(bytes);
if ident_parser.get_peer_ident().is_some() { if let Some(server_ident) = ident_parser.get_peer_ident() {
let client_ident = mem::take(client_ident); let client_ident = mem::take(client_ident);
// This moves to the next state. // This moves to the next state.
self.send_kexinit(client_ident); self.send_kexinit(client_ident, server_ident);
return Ok(()); return Ok(());
} }
return Ok(()); return Ok(());
@ -92,7 +104,10 @@ impl ClientConnection {
match &mut self.state { match &mut self.state {
ClientState::ProtoExchange { .. } => unreachable!("handled above"), ClientState::ProtoExchange { .. } => unreachable!("handled above"),
ClientState::KexInit { client_ident } => { ClientState::KexInit {
client_ident,
server_ident,
} => {
let mut kexinit = packet.payload_parser(); let mut kexinit = packet.payload_parser();
let packet_type = kexinit.u8()?; let packet_type = kexinit.u8()?;
if packet_type != numbers::SSH_MSG_KEXINIT { if packet_type != numbers::SSH_MSG_KEXINIT {
@ -101,60 +116,95 @@ impl ClientConnection {
numbers::packet_type_to_string(packet_type) numbers::packet_type_to_string(packet_type)
)); ));
} }
/*
let sup_algs = SupportedAlgorithms::secure();
let cookie = kexinit.array::<16>()?; let cookie = kexinit.array::<16>()?;
let kex_algorithm = kexinit.name_list()?; let kex_algorithm = kexinit.name_list()?;
let kex_algorithms = AlgorithmNegotiation { let kex_algorithm = sup_algs.key_exchange.find(kex_algorithm.0)?;
supported: vec![
crypto::KEX_CURVE_25519_SHA256,
crypto::KEX_ECDH_SHA2_NISTP256,
],
};
let kex_algorithm = kex_algorithms.find(kex_algorithm.0)?;
debug!(name = %kex_algorithm.name(), "Using KEX algorithm"); debug!(name = %kex_algorithm.name(), "Using KEX algorithm");
let server_hostkey_algorithm = kexinit.name_list()?; let server_hostkey_algorithm = kexinit.name_list()?;
let server_hostkey_algorithms = AlgorithmNegotiation {
supported: vec![
crypto::hostkey_ed25519(ED25519_PRIVKEY_BYTES.to_vec()),
crypto::hostkey_ecdsa_sha2_p256(ECDSA_P256_PRIVKEY_BYTES.to_vec()),
],
};
let server_hostkey_algorithm = let server_hostkey_algorithm =
server_hostkey_algorithms.find(server_hostkey_algorithm.0)?; sup_algs.hostkey.find(server_hostkey_algorithm.0)?;
debug!(name = %server_hostkey_algorithm.name(), "Using host key algorithm"); debug!(name = %server_hostkey_algorithm.name(), "Using host key algorithm");
let encryption_algorithms_client_to_server = kexinit.name_list()?; let encryption_algorithms_client_to_server = kexinit.name_list()?;
let encryption_algorithms_client_to_server = select_alg( let encryption_client_to_server = sup_algs
encryption_algorithms_client_to_server, .encryption_to_peer
[ .find(encryption_algorithms_client_to_server.0)?;
crypto::encrypt::CHACHA20POLY1305, debug!(name = %encryption_client_to_server.name(), "Using encryption algorithm C->S");
crypto::encrypt::AES256_GCM,
],
);
let encryption_algorithms_server_to_client = kexinit.name_list()?; let encryption_algorithms_server_to_client = kexinit.name_list()?;
let encryption_algorithms_server_to_client = select_alg( let encryption_server_to_client = sup_algs
encryption_algorithms_server_to_client, .encryption_from_peer
[ .find(encryption_algorithms_server_to_client.0)?;
crypto::encrypt::CHACHA20POLY1305, debug!(name = %encryption_server_to_client.name(), "Using encryption algorithm S->C");
crypto::encrypt::AES256_GCM,
],
);
let mac_algorithms_client_to_server = kexinit.name_list()?; let mac_algorithms_client_to_server = kexinit.name_list()?;
select_alg(mac_algorithms_client_to_server, ["hmac-sha2-256"])?; let _mac_client_to_server = sup_algs
.mac_to_peer
.find(mac_algorithms_client_to_server.0)?;
let mac_algorithms_server_to_client = kexinit.name_list()?; let mac_algorithms_server_to_client = kexinit.name_list()?;
select_alg(mac_algorithms_server_to_client, ["hmac-sha2-256"])?; let _mac_server_to_client = sup_algs
.mac_from_peer
.find(mac_algorithms_server_to_client.0)?;
let compression_algorithms_client_to_server = kexinit.name_list()?; let compression_algorithms_client_to_server = kexinit.name_list()?;
select_alg(compression_algorithms_client_to_server, ["none"])?; let _compression_client_to_server = sup_algs
.compression_to_peer
.find(compression_algorithms_client_to_server.0)?;
let compression_algorithms_server_to_client = kexinit.name_list()?; let compression_algorithms_server_to_client = kexinit.name_list()?;
select_alg(compression_algorithms_server_to_client, ["none"])?; let _compression_server_to_client = sup_algs
.compression_from_peer
.find(compression_algorithms_server_to_client.0)?;
let _languages_client_to_server = kexinit.name_list()?; let _languages_client_to_server = kexinit.name_list()?;
let _languages_server_to_client = kexinit.name_list()?; let _languages_server_to_client = kexinit.name_list()?;
let first_kex_packet_follows = kexinit.bool()?; let first_kex_packet_follows = kexinit.bool()?;
if first_kex_packet_follows { if first_kex_packet_follows {
return Err(peer_error!("does not support guessed kex init packages")); return Err(peer_error!("does not support guessed kex init packages"));
}*/ }
let kex_secret = (kex_algorithm.generate_secret)(&mut *self.rng);
self.packet_transport
.queue_packet(Packet::new_msg_kex_ecdh_init(&kex_secret.pubkey));
self.state = ClientState::DhKeyInit {
client_ident: mem::take(client_ident),
server_ident: mem::take(server_ident),
kex_secret: Some(kex_secret),
server_hostkey_algorithm,
encryption_client_to_server,
encryption_server_to_client,
};
}
ClientState::DhKeyInit {
client_ident,
server_ident,
kex_secret,
server_hostkey_algorithm,
encryption_client_to_server,
encryption_server_to_client,
} => {
let mut dh = packet.payload_parser();
let packet_type = dh.u8()?;
if packet_type != numbers::SSH_MSG_KEX_ECDH_REPLY {
return Err(peer_error!(
"expected SSH_MSG_KEX_ECDH_REPLY, found {}",
numbers::packet_type_to_string(packet_type)
));
}
let sever_host_key = dh.string()?;
let server_ephermal_key = dh.string()?;
let signature = dh.string()?;
let shared_secret =
(mem::take(kex_secret).unwrap().exchange)(server_ephermal_key)?;
} }
} }
} }
@ -165,7 +215,7 @@ impl ClientConnection {
self.packet_transport.next_msg_to_send() self.packet_transport.next_msg_to_send()
} }
fn send_kexinit(&mut self, client_ident: Vec<u8>) { fn send_kexinit(&mut self, client_ident: Vec<u8>, server_ident: Vec<u8>) {
let mut cookie = [0; 16]; let mut cookie = [0; 16];
self.rng.fill_bytes(&mut cookie); self.rng.fill_bytes(&mut cookie);
@ -192,6 +242,9 @@ impl ClientConnection {
self.packet_transport self.packet_transport
.queue_packet(Packet { payload: kexinit }); .queue_packet(Packet { payload: kexinit });
self.state = ClientState::KexInit { client_ident }; self.state = ClientState::KexInit {
client_ident,
server_ident,
};
} }
} }

View file

@ -23,66 +23,69 @@ impl AlgorithmName for &'static str {
#[derive(Clone, Copy)] #[derive(Clone, Copy)]
pub struct KexAlgorithm { pub struct KexAlgorithm {
name: &'static str, name: &'static str,
pub exchange: fn( /// Generate an ephemeral key for the exchange.
client_public_key: &[u8], pub generate_secret: fn(random: &mut (dyn SshRng + Send + Sync)) -> KeyExchangeSecret,
random: &mut (dyn SshRng + Send + Sync),
) -> Result<KexAlgorithmOutput>,
} }
impl AlgorithmName for KexAlgorithm { impl AlgorithmName for KexAlgorithm {
fn name(&self) -> &'static str { fn name(&self) -> &'static str {
self.name self.name
} }
} }
pub struct KexAlgorithmOutput {
/// K pub struct KeyExchangeSecret {
pub shared_secret: Vec<u8>, /// Q_x
/// Q_S pub pubkey: Vec<u8>,
pub server_public_key: Vec<u8>, /// Does the exchange, returning the shared secret K.
pub exchange: Box<dyn FnOnce(&[u8]) -> Result<Vec<u8>> + Send + Sync>,
} }
/// <https://datatracker.ietf.org/doc/html/rfc8731> /// <https://datatracker.ietf.org/doc/html/rfc8731>
pub const KEX_CURVE_25519_SHA256: KexAlgorithm = KexAlgorithm { pub const KEX_CURVE_25519_SHA256: KexAlgorithm = KexAlgorithm {
name: "curve25519-sha256", name: "curve25519-sha256",
exchange: |client_public_key, rng| { generate_secret: |rng| {
let secret = x25519_dalek::EphemeralSecret::random_from_rng(crate::SshRngRandAdapter(rng)); let secret = x25519_dalek::EphemeralSecret::random_from_rng(crate::SshRngRandAdapter(rng));
let server_public_key = x25519_dalek::PublicKey::from(&secret); // Q_S let my_public_key = x25519_dalek::PublicKey::from(&secret);
let Ok(arr) = <[u8; 32]>::try_from(client_public_key) else { KeyExchangeSecret {
return Err(crate::peer_error!( pubkey: my_public_key.as_bytes().to_vec(),
"invalid x25519 public key length, should be 32, was: {}", exchange: Box::new(move |peer_public_key| {
client_public_key.len() let Ok(peer_public_key) = <[u8; 32]>::try_from(peer_public_key) else {
)); return Err(crate::peer_error!(
}; "invalid x25519 public key length, should be 32, was: {}",
let client_public_key = x25519_dalek::PublicKey::from(arr); peer_public_key.len()
let shared_secret = secret.diffie_hellman(&client_public_key); // K ));
};
let peer_public_key = x25519_dalek::PublicKey::from(peer_public_key);
let shared_secret = secret.diffie_hellman(&peer_public_key); // K
Ok(KexAlgorithmOutput { Ok(shared_secret.as_bytes().to_vec())
server_public_key: server_public_key.as_bytes().to_vec(), }),
shared_secret: shared_secret.as_bytes().to_vec(), }
})
}, },
}; };
/// <https://datatracker.ietf.org/doc/html/rfc5656> /// <https://datatracker.ietf.org/doc/html/rfc5656>
pub const KEX_ECDH_SHA2_NISTP256: KexAlgorithm = KexAlgorithm { pub const KEX_ECDH_SHA2_NISTP256: KexAlgorithm = KexAlgorithm {
name: "ecdh-sha2-nistp256", name: "ecdh-sha2-nistp256",
exchange: |client_public_key, rng| { generate_secret: |rng| {
let secret = p256::ecdh::EphemeralSecret::random(&mut crate::SshRngRandAdapter(rng)); let secret = p256::ecdh::EphemeralSecret::random(&mut crate::SshRngRandAdapter(rng));
let server_public_key = p256::EncodedPoint::from(secret.public_key()); // Q_S let my_public_key = p256::EncodedPoint::from(secret.public_key());
let client_public_key = KeyExchangeSecret {
p256::PublicKey::from_sec1_bytes(client_public_key).map_err(|_| { pubkey: my_public_key.as_bytes().to_vec(),
crate::peer_error!( exchange: Box::new(move |peer_public_key| {
"invalid p256 public key length: {}", let peer_public_key =
client_public_key.len() p256::PublicKey::from_sec1_bytes(peer_public_key).map_err(|_| {
) crate::peer_error!(
})?; // Q_C "invalid p256 public key length: {}",
peer_public_key.len()
)
})?;
let shared_secret = secret.diffie_hellman(&client_public_key); // K let shared_secret = secret.diffie_hellman(&peer_public_key); // K
Ok(KexAlgorithmOutput { Ok(shared_secret.raw_secret_bytes().to_vec())
server_public_key: server_public_key.as_bytes().to_vec(), }),
shared_secret: shared_secret.raw_secret_bytes().to_vec(), }
})
}, },
}; };
@ -191,8 +194,8 @@ pub struct AlgorithmNegotiation<T> {
} }
impl<T: AlgorithmName> AlgorithmNegotiation<T> { impl<T: AlgorithmName> AlgorithmNegotiation<T> {
pub fn find<'a>(mut self, client_supports: &str) -> Result<T> { pub fn find<'a>(mut self, peer_supports: &str) -> Result<T> {
for client_alg in client_supports.split(',') { for client_alg in peer_supports.split(',') {
if let Some(alg) = self if let Some(alg) = self
.supported .supported
.iter() .iter()
@ -203,7 +206,7 @@ impl<T: AlgorithmName> AlgorithmNegotiation<T> {
} }
Err(peer_error!( Err(peer_error!(
"peer does not support any matching algorithm: peer supports: {client_supports:?}" "peer does not support any matching algorithm: peer supports: {peer_supports:?}"
)) ))
} }
} }

View file

@ -3,7 +3,7 @@ mod ctors;
use std::collections::VecDeque; use std::collections::VecDeque;
use std::mem; use std::mem;
use tracing::debug; use tracing::{debug, trace};
use crate::crypto::{EncryptionAlgorithm, Keys, Plaintext, Session}; use crate::crypto::{EncryptionAlgorithm, Keys, Plaintext, Session};
use crate::parse::{NameList, Parser, Writer}; use crate::parse::{NameList, Parser, Writer};
@ -82,6 +82,9 @@ impl PacketTransport {
} }
pub(crate) fn queue_packet(&mut self, packet: Packet) { pub(crate) fn queue_packet(&mut self, packet: Packet) {
let packet_type = packet.packet_type();
let packet_type_string = numbers::packet_type_to_string(packet_type);
trace!(%packet_type, %packet_type_string, packet_len = %packet.payload.len(), "Sending packet");
let seq_nr = self.send_next_seq_nr; let seq_nr = self.send_next_seq_nr;
self.send_next_seq_nr = self.send_next_seq_nr.wrapping_add(1); self.send_next_seq_nr = self.send_next_seq_nr.wrapping_add(1);
let msg = self.keys.encrypt_packet_to_msg(packet, seq_nr); let msg = self.keys.encrypt_packet_to_msg(packet, seq_nr);

View file

@ -1,9 +1,7 @@
use core::str;
use std::{collections::VecDeque, mem::take}; use std::{collections::VecDeque, mem::take};
use crate::crypto::{ use crate::crypto::{
self, AlgorithmName, AlgorithmNegotiation, EncryptionAlgorithm, HostKeySigningAlgorithm, self, AlgorithmName, EncryptionAlgorithm, HostKeySigningAlgorithm, SupportedAlgorithms,
SupportedAlgorithms,
}; };
use crate::packet::{ use crate::packet::{
KeyExchangeEcDhInitPacket, KeyExchangeInitPacket, Packet, PacketTransport, ProtocolIdentParser, KeyExchangeEcDhInitPacket, KeyExchangeInitPacket, Packet, PacketTransport, ProtocolIdentParser,
@ -112,17 +110,6 @@ impl ServerConnection {
} => { } => {
let kex = KeyExchangeInitPacket::parse(&packet.payload)?; let kex = KeyExchangeInitPacket::parse(&packet.payload)?;
let require_algorithm =
|expected: &'static str, list: NameList<'_>| -> Result<&'static str> {
if list.iter().any(|alg| alg == expected) {
Ok(expected)
} else {
Err(peer_error!(
"client does not support algorithm {expected}. supported: {list:?}",
))
}
};
let sup_algs = SupportedAlgorithms::secure(); let sup_algs = SupportedAlgorithms::secure();
let kex_algorithm = sup_algs.key_exchange.find(kex.kex_algorithms.0)?; let kex_algorithm = sup_algs.key_exchange.find(kex.kex_algorithms.0)?;
@ -151,7 +138,6 @@ impl ServerConnection {
let mac_algorithm_server_to_client = sup_algs let mac_algorithm_server_to_client = sup_algs
.mac_to_peer .mac_to_peer
.find(kex.mac_algorithms_server_to_client.0)?; .find(kex.mac_algorithms_server_to_client.0)?;
debug!("x");
let compression_algorithm_client_to_server = sup_algs let compression_algorithm_client_to_server = sup_algs
.compression_from_peer .compression_from_peer
@ -159,7 +145,7 @@ impl ServerConnection {
let compression_algorithm_server_to_client = sup_algs let compression_algorithm_server_to_client = sup_algs
.compression_to_peer .compression_to_peer
.find(kex.compression_algorithms_server_to_client.0)?; .find(kex.compression_algorithms_server_to_client.0)?;
debug!("x");
let _ = kex.languages_client_to_server; let _ = kex.languages_client_to_server;
let _ = kex.languages_server_to_client; let _ = kex.languages_server_to_client;
@ -226,10 +212,9 @@ impl ServerConnection {
let client_public_key = dh.qc; let client_public_key = dh.qc;
let crypto::KexAlgorithmOutput { let server_secret = (kex_algorithm.generate_secret)(&mut *self.rng);
server_public_key, let server_public_key = server_secret.pubkey;
shared_secret, let shared_secret = (server_secret.exchange)(client_public_key)?;
} = (kex_algorithm.exchange)(client_public_key, &mut *self.rng)?;
let pub_hostkey = server_host_key_algorithm.public_key(); let pub_hostkey = server_host_key_algorithm.public_key();