use crate::*;
#[derive(Debug, Clone, PartialEq, Eq)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub struct Hdlr {
pub handler: FourCC,
pub name: String,
}
impl Default for Hdlr {
fn default() -> Self {
Hdlr {
handler: FourCC::new(b"none"),
name: String::new(),
}
}
}
impl AtomExt for Hdlr {
type Ext = ();
const KIND_EXT: FourCC = FourCC::new(b"hdlr");
fn decode_body_ext<B: Buf>(buf: &mut B, _ext: ()) -> Result<Self> {
u32::decode(buf)?; let handler = FourCC::decode(buf)?;
<[u8; 12]>::decode(buf)?;
let name = String::decode(buf)?;
if buf.has_remaining() {
tracing::warn!("Skipped {} extra trailing bytes in hdlr", buf.remaining());
buf.advance(buf.remaining());
}
Ok(Hdlr { handler, name })
}
fn encode_body_ext<B: BufMut>(&self, buf: &mut B) -> Result<()> {
0u32.encode(buf)?; self.handler.encode(buf)?;
[0u8; 12].encode(buf)?;
self.name.as_str().encode(buf)?;
Ok(())
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_hdlr() {
let expected = Hdlr {
handler: FourCC::new(b"vide"),
name: String::from("VideoHandler"),
};
let mut buf = Vec::new();
expected.encode(&mut buf).unwrap();
let mut buf = buf.as_ref();
let decoded = Hdlr::decode(&mut buf).unwrap();
assert_eq!(decoded, expected);
}
#[test]
fn test_hdlr_empty() {
let expected = Hdlr {
handler: FourCC::new(b"vide"),
name: String::new(),
};
let mut buf = Vec::new();
expected.encode(&mut buf).unwrap();
let mut buf = buf.as_ref();
let decoded = Hdlr::decode(&mut buf).unwrap();
assert_eq!(decoded, expected);
}
#[test]
fn test_hdlr_with_trailing_bytes() {
let mut buf = Vec::new();
buf.extend_from_slice(&[0x00, 0x00, 0x00, 0x00]); buf.extend_from_slice(b"hdlr");
buf.extend_from_slice(&[0x00, 0x00, 0x00, 0x00]);
buf.extend_from_slice(&[0x00, 0x00, 0x00, 0x00]);
buf.extend_from_slice(b"vide");
buf.extend_from_slice(&[0x00; 12]);
buf.extend_from_slice(b"VideoHandler\0");
buf.extend_from_slice(&[0xAA, 0xBB, 0xCC, 0xDD]);
let size = buf.len() as u32;
buf[0..4].copy_from_slice(&size.to_be_bytes());
let mut cursor = std::io::Cursor::new(&buf);
let decoded = Hdlr::decode(&mut cursor).expect("failed to decode hdlr with trailing bytes");
assert_eq!(decoded.handler, FourCC::new(b"vide"));
assert_eq!(decoded.name, "VideoHandler");
assert_eq!(cursor.position(), buf.len() as u64);
}
#[test]
fn test_hdlr_with_multiple_trailing_nulls() {
let mut buf = Vec::new();
buf.extend_from_slice(&[0x00, 0x00, 0x00, 0x00]); buf.extend_from_slice(b"hdlr");
buf.extend_from_slice(&[0x00, 0x00, 0x00, 0x00]);
buf.extend_from_slice(&[0x00, 0x00, 0x00, 0x00]);
buf.extend_from_slice(b"soun");
buf.extend_from_slice(&[0x00; 12]);
buf.extend_from_slice(b"SoundHandler\0");
buf.extend_from_slice(&[0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00]);
let size = buf.len() as u32;
buf[0..4].copy_from_slice(&size.to_be_bytes());
let mut cursor = std::io::Cursor::new(&buf);
let decoded = Hdlr::decode(&mut cursor).expect("failed to decode hdlr with trailing nulls");
assert_eq!(decoded.handler, FourCC::new(b"soun"));
assert_eq!(decoded.name, "SoundHandler");
assert_eq!(cursor.position(), buf.len() as u64);
}
#[test]
fn test_hdlr_roundtrip_with_trailing_bytes() {
let original = Hdlr {
handler: FourCC::new(b"meta"),
name: "MetaHandler".to_string(),
};
let mut encoded = Vec::new();
original.encode(&mut encoded).unwrap();
let mut cursor = std::io::Cursor::new(&encoded);
let decoded = Hdlr::decode(&mut cursor).expect("failed to decode clean hdlr");
assert_eq!(decoded, original);
let box_end = cursor.position() as usize;
let mut encoded_with_trash = encoded[..box_end].to_vec();
encoded_with_trash.extend_from_slice(&[0xFF, 0xEE, 0xDD]);
let new_size = encoded_with_trash.len() as u32;
encoded_with_trash[0..4].copy_from_slice(&new_size.to_be_bytes());
let mut cursor2 = std::io::Cursor::new(&encoded_with_trash);
let decoded2 = Hdlr::decode(&mut cursor2).expect("failed to decode hdlr with added trash");
assert_eq!(decoded2, original);
assert_eq!(cursor2.position(), encoded_with_trash.len() as u64);
}
}