use bytes::{Buf, BytesMut};
use crate::catalog::hang::CatalogExt;
#[derive(Debug)]
pub(crate) struct Header {
pub len: usize,
pub sample_rate: u32,
pub channel_count: u32,
pub samples: u64,
}
pub(crate) struct Descriptor {
pub track_suffix: &'static str,
pub codec: hang::catalog::AudioCodec,
pub min_header_len: usize,
pub parse: fn(&[u8]) -> anyhow::Result<Header>,
}
pub(crate) struct Config {
pub sample_rate: u32,
pub channel_count: u32,
}
pub(crate) struct Import<E: CatalogExt = ()> {
catalog: crate::catalog::Producer<E>,
track: crate::container::Producer<crate::catalog::hang::Container>,
zero: Option<tokio::time::Instant>,
}
impl<E: CatalogExt> Import<E> {
pub fn new(
descriptor: &'static Descriptor,
mut broadcast: moq_net::BroadcastProducer,
mut catalog: crate::catalog::Producer<E>,
config: Config,
) -> anyhow::Result<Self> {
let track = broadcast.unique_track(descriptor.track_suffix)?;
let mut audio_config =
hang::catalog::AudioConfig::new(descriptor.codec.clone(), config.sample_rate, config.channel_count);
audio_config.container = hang::catalog::Container::Legacy;
tracing::debug!(name = ?track.name, config = ?audio_config, "starting track");
catalog.lock().audio.renditions.insert(track.name.clone(), audio_config);
Ok(Self {
catalog,
track: crate::container::Producer::new(track, crate::catalog::hang::Container::Legacy),
zero: None,
})
}
pub fn finish(&mut self) -> anyhow::Result<()> {
self.track.finish()?;
Ok(())
}
pub fn seek(&mut self, sequence: u64) -> anyhow::Result<()> {
self.track.seek(sequence)?;
Ok(())
}
pub fn decode<T: Buf>(&mut self, buf: &mut T, pts: Option<crate::container::Timestamp>) -> anyhow::Result<()> {
let pts = self.pts(pts)?;
let mut payload = BytesMut::with_capacity(buf.remaining());
while buf.has_remaining() {
let chunk = buf.chunk();
payload.extend_from_slice(chunk);
let len = chunk.len();
buf.advance(len);
}
let frame = crate::container::Frame {
timestamp: pts,
payload: payload.freeze(),
keyframe: true,
};
self.track.write(frame)?;
self.track.finish_group()?;
Ok(())
}
fn pts(&mut self, hint: Option<crate::container::Timestamp>) -> anyhow::Result<crate::container::Timestamp> {
if let Some(pts) = hint {
return Ok(pts);
}
let zero = self.zero.get_or_insert_with(tokio::time::Instant::now);
Ok(crate::container::Timestamp::from_micros(
zero.elapsed().as_micros() as u64
)?)
}
}
impl<E: CatalogExt> Drop for Import<E> {
fn drop(&mut self) {
tracing::debug!(name = ?self.track.name, "ending track");
self.catalog.lock().audio.renditions.remove(&self.track.name);
}
}