use buf_list::BufList;
use bytes::Buf;
const MAX_GROUP_DURATION: hang::container::Timestamp = hang::container::Timestamp::from_millis_unchecked(100);
pub struct OpusConfig {
pub sample_rate: u32,
pub channel_count: u32,
}
impl OpusConfig {
pub fn parse<T: Buf>(buf: &mut T) -> anyhow::Result<Self> {
anyhow::ensure!(buf.remaining() >= 19, "OpusHead must be at least 19 bytes");
const OPUS_HEAD: u64 = u64::from_be_bytes(*b"OpusHead");
let signature = buf.get_u64();
anyhow::ensure!(signature == OPUS_HEAD, "invalid OpusHead signature");
buf.advance(1); let channel_count = buf.get_u8() as u32;
buf.advance(2); let sample_rate = buf.get_u32_le();
if buf.remaining() > 0 {
buf.advance(buf.remaining());
}
Ok(Self {
sample_rate,
channel_count,
})
}
}
pub struct Opus {
catalog: crate::CatalogProducer,
track: hang::container::OrderedProducer,
zero: Option<tokio::time::Instant>,
}
impl Opus {
pub fn new(
mut broadcast: moq_lite::BroadcastProducer,
mut catalog: crate::CatalogProducer,
config: OpusConfig,
) -> anyhow::Result<Self> {
let track = broadcast.unique_track(".opus")?;
let audio_config = hang::catalog::AudioConfig {
codec: hang::catalog::AudioCodec::Opus,
sample_rate: config.sample_rate,
channel_count: config.channel_count,
bitrate: None,
description: None,
container: hang::catalog::Container::Legacy,
jitter: Some(moq_lite::Time::new(20)),
};
catalog.lock().audio.insert(&track.info.name, audio_config.clone())?;
tracing::debug!(name = ?track.info.name, config = ?audio_config, "starting track");
Ok(Self {
catalog,
track: hang::container::OrderedProducer::new(track).with_max_group_duration(MAX_GROUP_DURATION),
zero: None,
})
}
pub fn track(&self) -> &moq_lite::TrackProducer {
&self.track
}
pub fn finish(&mut self) -> anyhow::Result<()> {
self.track.finish()?;
Ok(())
}
pub fn decode<T: Buf>(&mut self, buf: &mut T, pts: Option<hang::container::Timestamp>) -> anyhow::Result<()> {
let pts = self.pts(pts)?;
let mut payload = BufList::new();
while !buf.chunk().is_empty() {
payload.push_chunk(buf.copy_to_bytes(buf.chunk().len()));
}
let frame = hang::container::Frame {
timestamp: pts,
payload,
};
self.track.write(frame)?;
Ok(())
}
fn pts(&mut self, hint: Option<hang::container::Timestamp>) -> anyhow::Result<hang::container::Timestamp> {
if let Some(pts) = hint {
return Ok(pts);
}
let zero = self.zero.get_or_insert_with(tokio::time::Instant::now);
Ok(hang::container::Timestamp::from_micros(
zero.elapsed().as_micros() as u64
)?)
}
}
impl Drop for Opus {
fn drop(&mut self) {
tracing::debug!(name = ?self.track.info.name, "ending track");
self.catalog.lock().audio.remove(&self.track.info.name);
}
}