pub const PROTOCOL_VERSION: u16 = 5;
pub const TRANSFER_FILE: u8 = 0x00;
pub const TRANSFER_DIR: u8 = 0x01;
pub const FRAME_RAW: u8 = 0x00;
pub const FRAME_ZSTD: u8 = 0x01;
pub const MAX_FILENAME_BYTES: usize = 4096;
pub const MAX_METADATA_ENCRYPTED: usize = 4 + MAX_FILENAME_BYTES + 8 + 1 + 1 + 256 + 12 + 16 + 16;
pub const CHUNK_SIZE: usize = 1024 * 1024;
#[derive(Debug, Clone)]
pub struct Metadata {
pub filename: String,
pub total_size: u64,
pub transfer_type: u8,
pub hash_algo: String,
}
#[cold]
#[inline(never)]
fn invalid_filename_len(len: usize) -> crate::EngineError {
crate::EngineError::InvalidFrame(format!("invalid filename length: {len}"))
}
#[cold]
#[inline(never)]
fn invalid_transfer_type(ty: u8) -> crate::EngineError {
crate::EngineError::InvalidFrame(format!("invalid transfer type: 0x{ty:02x}"))
}
#[cold]
#[inline(never)]
fn invalid_hash_algo_len(len: usize) -> crate::EngineError {
crate::EngineError::InvalidFrame(format!("invalid hash algorithm length: {len}"))
}
impl Metadata {
#[inline]
pub fn validate(&self) -> Result<(), crate::EngineError> {
let name_len = self.filename.len();
if name_len.wrapping_sub(1) >= MAX_FILENAME_BYTES {
return Err(invalid_filename_len(name_len));
}
if self.transfer_type > TRANSFER_DIR {
return Err(invalid_transfer_type(self.transfer_type));
}
let algo_len = self.hash_algo.len();
if algo_len.wrapping_sub(1) >= 255 {
return Err(invalid_hash_algo_len(algo_len));
}
Ok(())
}
pub fn encode(&self) -> Vec<u8> {
let name_bytes = self.filename.as_bytes();
let algo_bytes = self.hash_algo.as_bytes();
let mut buf = Vec::with_capacity(2 + name_bytes.len() + 8 + 1 + 1 + algo_bytes.len());
buf.extend_from_slice(&(name_bytes.len() as u16).to_be_bytes());
buf.extend_from_slice(name_bytes);
buf.extend_from_slice(&self.total_size.to_be_bytes());
buf.push(self.transfer_type);
buf.push(algo_bytes.len() as u8);
buf.extend_from_slice(algo_bytes);
buf
}
pub fn decode(raw: &[u8]) -> Result<Self, crate::EngineError> {
if raw.len() < 12 {
return Err(crate::EngineError::InvalidFrame(
"metadata too short".into(),
));
}
let name_len = u16::from_be_bytes([raw[0], raw[1]]) as usize;
if name_len == 0 || name_len > MAX_FILENAME_BYTES {
return Err(crate::EngineError::InvalidFrame(format!(
"invalid filename length: {name_len}"
)));
}
if raw.len() < 2 + name_len + 8 + 1 + 1 {
return Err(crate::EngineError::InvalidFrame(
"metadata truncated before hash algorithm".into(),
));
}
let filename = std::str::from_utf8(&raw[2..2 + name_len])
.map_err(|_| crate::EngineError::InvalidFrame("filename not UTF-8".into()))?
.to_owned();
let total_size = u64::from_be_bytes(
raw[2 + name_len..2 + name_len + 8]
.try_into()
.map_err(|_| crate::EngineError::InvalidFrame("metadata truncated".into()))?,
);
let transfer_type = raw[2 + name_len + 8];
let algo_len = raw[2 + name_len + 9] as usize;
if raw.len() < 2 + name_len + 10 + algo_len {
return Err(crate::EngineError::InvalidFrame(
"metadata truncated for hash algorithm name".into(),
));
}
let hash_algo = std::str::from_utf8(&raw[2 + name_len + 10..2 + name_len + 10 + algo_len])
.map_err(|_| crate::EngineError::InvalidFrame("hash algorithm not UTF-8".into()))?
.to_owned();
Ok(Self {
filename,
total_size,
transfer_type,
hash_algo,
})
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn decode_rejects_unknown_transfer_type() {
let mut raw = Vec::new();
raw.extend_from_slice(&4u16.to_be_bytes());
raw.extend_from_slice(b"name");
raw.extend_from_slice(&123u64.to_be_bytes());
raw.push(0xff);
let err = Metadata::decode(&raw).unwrap_err();
assert!(matches!(err, crate::EngineError::InvalidFrame(_)));
}
#[test]
fn validate_rejects_unknown_transfer_type() {
let meta = Metadata {
filename: "name".to_owned(),
total_size: 0,
transfer_type: 0xff,
hash_algo: "blake3".to_owned(),
};
let err = meta.validate().unwrap_err();
assert!(matches!(err, crate::EngineError::InvalidFrame(_)));
}
#[test]
fn validate_rejects_empty_filename() {
let meta = Metadata {
filename: String::new(),
total_size: 0,
transfer_type: TRANSFER_FILE,
hash_algo: "blake3".to_owned(),
};
let err = meta.validate().unwrap_err();
assert!(matches!(err, crate::EngineError::InvalidFrame(_)));
}
}