use std::{fmt, mem};
use puffin::codec;
use puffin::codec::{Codec, Reader};
use ring::digest;
use crate::tls::rustls::msgs::handshake::HandshakeMessagePayload;
use crate::tls::rustls::msgs::message::{Message, MessagePayload};
#[derive(Default, Clone)]
pub struct HandshakeHashBuffer {
buffer: Vec<u8>,
client_auth_enabled: bool,
}
impl HandshakeHashBuffer {
pub fn new() -> Self {
Self {
buffer: Vec::new(),
client_auth_enabled: false,
}
}
pub fn set_client_auth_enabled(&mut self) {
self.client_auth_enabled = true;
}
pub fn add_message(&mut self, m: &Message) {
if let MessagePayload::Handshake(hs) = &m.payload {
self.buffer.extend_from_slice(&hs.get_encoding());
}
}
#[cfg(test)]
fn update_raw(&mut self, buf: &[u8]) {
self.buffer.extend_from_slice(buf);
}
pub fn get_hash_given(&self, hash: &'static digest::Algorithm, extra: &[u8]) -> digest::Digest {
let mut ctx = digest::Context::new(hash);
ctx.update(&self.buffer);
ctx.update(extra);
ctx.finish()
}
pub fn start_hash(self, alg: &'static digest::Algorithm) -> HandshakeHash {
let mut ctx = digest::Context::new(alg);
ctx.update(&self.buffer);
HandshakeHash {
ctx,
client_auth: match self.client_auth_enabled {
true => Some(self.buffer),
false => None,
},
override_buffer: None,
}
}
}
#[derive(Clone)]
pub struct HandshakeHash {
ctx: digest::Context,
client_auth: Option<Vec<u8>>,
override_buffer: Option<Vec<u8>>,
}
impl fmt::Debug for HandshakeHash {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
writeln!(f, "HandshakeHash: {:?}", codec::Codec::get_encoding(self))
}
}
impl codec::Codec for HandshakeHash {
fn encode(&self, bytes: &mut Vec<u8>) {
let mut hash = self.get_current_hash_raw();
bytes.append(&mut hash)
}
fn read(r: &mut Reader) -> Option<Self> {
Some(HandshakeHash::new_override(
r.rest().to_vec(),
&ring::digest::SHA256,
))
}
}
impl HandshakeHash {
pub fn new(alg: &'static digest::Algorithm) -> HandshakeHash {
let ctx = digest::Context::new(alg);
HandshakeHash {
ctx,
client_auth: None,
override_buffer: None,
}
}
pub fn new_override(static_buffer: Vec<u8>, alg: &'static digest::Algorithm) -> HandshakeHash {
let ctx = digest::Context::new(alg);
HandshakeHash {
ctx,
client_auth: None,
override_buffer: Some(static_buffer),
}
}
pub fn abandon_client_auth(&mut self) {
self.client_auth = None;
}
pub fn add_message(&mut self, m: &Message) -> &mut Self {
if let MessagePayload::Handshake(hs) = &m.payload {
let buf = hs.get_encoding();
self.update_raw(&buf);
}
self
}
fn update_raw(&mut self, buf: &[u8]) -> &mut Self {
self.ctx.update(buf);
if let Some(buffer) = &mut self.client_auth {
buffer.extend_from_slice(buf);
}
self
}
pub fn get_hash_given(&self, extra: &[u8]) -> digest::Digest {
let mut ctx = self.ctx.clone();
ctx.update(extra);
ctx.finish()
}
pub fn into_hrr_buffer(self) -> HandshakeHashBuffer {
let old_hash = self.ctx.finish();
let old_handshake_hash_msg =
HandshakeMessagePayload::build_handshake_hash(old_hash.as_ref());
HandshakeHashBuffer {
client_auth_enabled: self.client_auth.is_some(),
buffer: old_handshake_hash_msg.get_encoding(),
}
}
pub fn rollup_for_hrr(&mut self) {
let ctx = &mut self.ctx;
let old_ctx = mem::replace(ctx, digest::Context::new(ctx.algorithm()));
let old_hash = old_ctx.finish();
let old_handshake_hash_msg =
HandshakeMessagePayload::build_handshake_hash(old_hash.as_ref());
self.update_raw(&old_handshake_hash_msg.get_encoding());
}
pub fn get_current_hash(&self) -> digest::Digest {
self.ctx.clone().finish()
}
pub fn get_current_hash_raw(&self) -> Vec<u8> {
if let Some(static_buffer) = &self.override_buffer {
(*static_buffer).clone()
} else {
Vec::from(self.get_current_hash().as_ref())
}
}
pub fn take_handshake_buf(&mut self) -> Option<Vec<u8>> {
self.client_auth.take()
}
pub fn algorithm(&self) -> &'static digest::Algorithm {
self.ctx.algorithm()
}
}
#[cfg(test)]
mod tests {
use ring::digest;
use super::HandshakeHashBuffer;
#[test_log::test]
fn hashes_correctly() {
let mut hhb = HandshakeHashBuffer::new();
hhb.update_raw(b"hello");
assert_eq!(hhb.buffer.len(), 5);
let mut hh = hhb.start_hash(&digest::SHA256);
assert!(hh.client_auth.is_none());
hh.update_raw(b"world");
let h = hh.get_current_hash();
let h = h.as_ref();
assert_eq!(h[0], 0x93);
assert_eq!(h[1], 0x6a);
assert_eq!(h[2], 0x18);
assert_eq!(h[3], 0x5c);
}
#[test_log::test]
fn buffers_correctly() {
let mut hhb = HandshakeHashBuffer::new();
hhb.set_client_auth_enabled();
hhb.update_raw(b"hello");
assert_eq!(hhb.buffer.len(), 5);
let mut hh = hhb.start_hash(&digest::SHA256);
assert_eq!(hh.client_auth.as_ref().map(|buf| buf.len()), Some(5));
hh.update_raw(b"world");
assert_eq!(hh.client_auth.as_ref().map(|buf| buf.len()), Some(10));
let h = hh.get_current_hash();
let h = h.as_ref();
assert_eq!(h[0], 0x93);
assert_eq!(h[1], 0x6a);
assert_eq!(h[2], 0x18);
assert_eq!(h[3], 0x5c);
let buf = hh.take_handshake_buf();
assert_eq!(Some(b"helloworld".to_vec()), buf);
}
#[test_log::test]
fn abandon() {
let mut hhb = HandshakeHashBuffer::new();
hhb.set_client_auth_enabled();
hhb.update_raw(b"hello");
assert_eq!(hhb.buffer.len(), 5);
let mut hh = hhb.start_hash(&digest::SHA256);
assert_eq!(hh.client_auth.as_ref().map(|buf| buf.len()), Some(5));
hh.abandon_client_auth();
assert_eq!(hh.client_auth, None);
hh.update_raw(b"world");
assert_eq!(hh.client_auth, None);
let h = hh.get_current_hash();
let h = h.as_ref();
assert_eq!(h[0], 0x93);
assert_eq!(h[1], 0x6a);
assert_eq!(h[2], 0x18);
assert_eq!(h[3], 0x5c);
}
}