use std::io::Cursor;
use bytes::BytesMut;
use mp4_atom::{DecodeMaybe, Encode};
#[tokio::test(start_paused = true)]
async fn avc3_source_to_cmaf_export_roundtrip() {
use crate::container::Timestamp;
use hang::catalog::{Container, H264, VideoConfig};
let broadcast = moq_net::Broadcast::new();
let mut producer = broadcast.produce();
let consumer = producer.consume();
let mut catalog = crate::catalog::hang::Producer::new(&mut producer).unwrap();
let track = producer.unique_track(".avc3").unwrap();
let mut config = VideoConfig::new(H264 {
profile: 0x42,
constraints: 0xc0,
level: 0x1f,
inline: true,
});
config.coded_width = Some(320);
config.coded_height = Some(240);
config.container = Container::Legacy;
catalog.lock().video.renditions.insert(track.name.clone(), config);
const SC: &[u8] = &[0, 0, 0, 1];
let sps = &[0x67u8, 0x42, 0xc0, 0x1f, 0xde, 0xad, 0xbe, 0xef][..];
let pps = &[0x68u8, 0xce, 0x3c, 0x80][..];
let idr = &[0x65u8, 0x88, 0x84, 0x21, 0x00, 0x11, 0x22, 0x33][..];
let mut keyframe_payload = BytesMut::new();
for nal in [sps, pps, idr] {
keyframe_payload.extend_from_slice(SC);
keyframe_payload.extend_from_slice(nal);
}
let keyframe_payload = keyframe_payload.freeze();
let mut track_producer = crate::container::Producer::new(track, crate::catalog::hang::Container::Legacy);
track_producer
.write(crate::container::Frame {
timestamp: Timestamp::from_micros(0).unwrap(),
payload: keyframe_payload,
keyframe: true,
})
.unwrap();
track_producer.finish().unwrap();
let mut exporter = crate::container::fmp4::Export::new(consumer).expect("new Fmp4");
let init = tokio::time::timeout(std::time::Duration::from_secs(1), exporter.next())
.await
.expect("exporter timed out")
.expect("exporter result")
.expect("expected init bytes");
drop(track_producer);
drop(catalog);
drop(producer);
let mut cursor = Cursor::new(init.as_ref());
let mut saw_ftyp = false;
let mut moov: Option<mp4_atom::Moov> = None;
while let Some(atom) = mp4_atom::Any::decode_maybe(&mut cursor).expect("decode init") {
match atom {
mp4_atom::Any::Ftyp(_) => saw_ftyp = true,
mp4_atom::Any::Moov(m) => moov = Some(m),
_ => {}
}
}
assert!(saw_ftyp, "init segment missing ftyp");
let moov = moov.expect("init segment missing moov");
assert_eq!(moov.trak.len(), 1, "expected single track in moov");
let trak = &moov.trak[0];
let stsd = &trak.mdia.minf.stbl.stsd;
assert_eq!(stsd.codecs.len(), 1, "expected single sample entry");
let avc1 = match &stsd.codecs[0] {
mp4_atom::Codec::Avc1(avc1) => avc1,
other => panic!("expected Avc1 sample entry, got {:?}", other),
};
assert_eq!(avc1.avcc.avc_profile_indication, sps[1]);
assert_eq!(avc1.avcc.avc_level_indication, sps[3]);
assert_eq!(avc1.avcc.sequence_parameter_sets.len(), 1);
assert_eq!(avc1.avcc.sequence_parameter_sets[0].as_slice(), sps);
assert_eq!(avc1.avcc.picture_parameter_sets[0].as_slice(), pps);
assert_eq!(avc1.visual.width, 320);
assert_eq!(avc1.visual.height, 240);
let mvex = moov.mvex.as_ref().expect("init segment missing mvex");
assert_eq!(mvex.trex.len(), 1);
assert_eq!(mvex.trex[0].track_id, trak.tkhd.track_id);
}
#[tokio::test(start_paused = true)]
async fn legacy_aac_source_to_cmaf_export_synthesizes_esds() {
use crate::container::Timestamp;
use bytes::Bytes;
use hang::catalog::{AAC, AudioConfig, Container};
let broadcast = moq_net::Broadcast::new();
let mut producer = broadcast.produce();
let consumer = producer.consume();
let mut catalog = crate::catalog::hang::Producer::new(&mut producer).unwrap();
let track = producer.unique_track(".aac").unwrap();
let description = crate::codec::aac::Config {
profile: 2,
sample_rate: 44100,
channel_count: 2,
}
.encode();
let mut config = AudioConfig::new(AAC { profile: 2 }, 44100, 2);
config.description = Some(description.clone());
config.container = Container::Legacy;
catalog.lock().audio.renditions.insert(track.name.clone(), config);
let mut track_producer = crate::container::Producer::new(track, crate::catalog::hang::Container::Legacy);
track_producer
.write(crate::container::Frame {
timestamp: Timestamp::from_micros(0).unwrap(),
payload: Bytes::from_static(&[0x01, 0x02, 0x03, 0x04]),
keyframe: true,
})
.unwrap();
track_producer.finish().unwrap();
let mut exporter = crate::container::fmp4::Export::new(consumer).expect("new Fmp4");
let init = tokio::time::timeout(std::time::Duration::from_secs(1), exporter.next())
.await
.expect("exporter timed out")
.expect("exporter result")
.expect("expected init bytes");
drop(track_producer);
drop(catalog);
drop(producer);
let mut cursor = Cursor::new(init.as_ref());
let mut moov: Option<mp4_atom::Moov> = None;
while let Some(atom) = mp4_atom::Any::decode_maybe(&mut cursor).expect("decode init") {
if let mp4_atom::Any::Moov(m) = atom {
moov = Some(m);
}
}
let moov = moov.expect("init segment missing moov");
assert_eq!(moov.trak.len(), 1, "expected single track in moov");
let trak = &moov.trak[0];
let stsd = &trak.mdia.minf.stbl.stsd;
assert_eq!(stsd.codecs.len(), 1, "expected single sample entry");
let mp4a = match &stsd.codecs[0] {
mp4_atom::Codec::Mp4a(mp4a) => mp4a,
other => panic!("expected Mp4a sample entry, got {:?}", other),
};
assert_eq!(mp4a.audio.channel_count, 2);
assert_eq!(mp4a.audio.sample_rate.integer(), 44100);
let dec_config = &mp4a.esds.es_desc.dec_config;
assert_eq!(dec_config.object_type_indication, 0x40, "MPEG-4 AAC");
assert_eq!(dec_config.stream_type, 0x05, "audio stream");
let dec_specific = &dec_config.dec_specific;
assert_eq!(dec_specific.profile, 2, "AAC-LC");
assert_eq!(dec_specific.freq_index, 4, "44100 Hz");
assert_eq!(dec_specific.chan_conf, 2, "stereo");
let mut buf = Vec::new();
moov.encode(&mut buf).expect("encode synthesized moov");
}
#[tokio::test(start_paused = true)]
async fn cmaf_source_to_cmaf_export_passthrough() {
let data = include_bytes!("test_data/bbb.mp4");
let broadcast = moq_net::Broadcast::new();
let mut producer = broadcast.produce();
let consumer = producer.consume();
let catalog = crate::catalog::hang::Producer::new(&mut producer).unwrap();
let mut importer = crate::container::fmp4::Import::new(producer, catalog);
let mut buf = BytesMut::from(data.as_slice());
let _ = importer.decode(&mut buf);
let mut exporter = crate::container::fmp4::Export::new(consumer).expect("new Fmp4");
let init = tokio::time::timeout(std::time::Duration::from_secs(1), exporter.next())
.await
.expect("exporter timed out")
.expect("exporter result")
.expect("expected init bytes");
drop(importer);
let mut cursor = Cursor::new(init.as_ref());
let mut moov: Option<mp4_atom::Moov> = None;
let mut saw_ftyp = false;
while let Some(atom) = mp4_atom::Any::decode_maybe(&mut cursor).expect("decode init") {
match atom {
mp4_atom::Any::Ftyp(_) => saw_ftyp = true,
mp4_atom::Any::Moov(m) => moov = Some(m),
_ => {}
}
}
assert!(saw_ftyp);
let moov = moov.expect("moov");
assert_eq!(moov.trak.len(), 2, "expected two tracks (one video, one audio)");
let mvex = moov.mvex.as_ref().expect("mvex");
assert_eq!(mvex.trex.len(), 2);
let mut buf = Vec::new();
moov.encode(&mut buf).expect("encode merged moov");
}