ff-encode 0.14.1

Video and audio encoding - the Rust way
Documentation
//! Container format definitions.

/// Output container format for encoding.
///
/// The container format is usually auto-detected from the file extension,
/// but can be explicitly specified if needed.
///
/// Named `OutputContainer` (rather than `Container`) to avoid confusion with
/// `ff_format::ContainerInfo`, which describes a container *read* from a probed file.
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
#[non_exhaustive]
pub enum OutputContainer {
    /// MP4 / `QuickTime`
    Mp4,

    /// Fragmented MP4 — CMAF-compatible streaming container.
    ///
    /// Uses the same `mp4` `FFmpeg` muxer as [`OutputContainer::Mp4`] but with
    /// `movflags=+frag_keyframe+empty_moov+default_base_moof` applied before
    /// writing the header. Required for HTTP Live Streaming fMP4 segments
    /// (CMAF) and MPEG-DASH.
    FMp4,

    /// `WebM`
    WebM,

    /// Matroska
    Mkv,

    /// AVI
    Avi,

    /// MOV
    Mov,

    /// FLAC (lossless audio container)
    Flac,

    /// OGG (audio container for Vorbis/Opus)
    Ogg,
}

impl OutputContainer {
    /// Get `FFmpeg` format name.
    #[must_use]
    pub const fn as_str(self) -> &'static str {
        match self {
            Self::Mp4 | Self::FMp4 => "mp4",
            Self::WebM => "webm",
            Self::Mkv => "matroska",
            Self::Avi => "avi",
            Self::Mov => "mov",
            Self::Flac => "flac",
            Self::Ogg => "ogg",
        }
    }

    /// Get default file extension.
    #[must_use]
    pub const fn default_extension(self) -> &'static str {
        match self {
            Self::Mp4 | Self::FMp4 => "mp4",
            Self::WebM => "webm",
            Self::Mkv => "mkv",
            Self::Avi => "avi",
            Self::Mov => "mov",
            Self::Flac => "flac",
            Self::Ogg => "ogg",
        }
    }

    /// Returns `true` if this container is fragmented MP4.
    ///
    /// When `true`, the encoder applies
    /// `movflags=+frag_keyframe+empty_moov+default_base_moof` before writing
    /// the file header, enabling CMAF-compatible streaming output.
    #[must_use]
    pub const fn is_fragmented(self) -> bool {
        matches!(self, Self::FMp4)
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn test_container_as_str() {
        assert_eq!(OutputContainer::Mp4.as_str(), "mp4");
        assert_eq!(OutputContainer::WebM.as_str(), "webm");
        assert_eq!(OutputContainer::Mkv.as_str(), "matroska");
    }

    #[test]
    fn test_container_extension() {
        assert_eq!(OutputContainer::Mp4.default_extension(), "mp4");
        assert_eq!(OutputContainer::WebM.default_extension(), "webm");
        assert_eq!(OutputContainer::Mkv.default_extension(), "mkv");
        assert_eq!(OutputContainer::Flac.default_extension(), "flac");
        assert_eq!(OutputContainer::Ogg.default_extension(), "ogg");
    }

    #[test]
    fn flac_as_str_should_return_flac() {
        assert_eq!(OutputContainer::Flac.as_str(), "flac");
    }

    #[test]
    fn ogg_as_str_should_return_ogg() {
        assert_eq!(OutputContainer::Ogg.as_str(), "ogg");
    }

    #[test]
    fn fmp4_as_str_should_return_mp4() {
        assert_eq!(OutputContainer::FMp4.as_str(), "mp4");
    }

    #[test]
    fn fmp4_extension_should_return_mp4() {
        assert_eq!(OutputContainer::FMp4.default_extension(), "mp4");
    }

    #[test]
    fn fmp4_is_fragmented_should_return_true() {
        assert!(OutputContainer::FMp4.is_fragmented());
        assert!(!OutputContainer::Mp4.is_fragmented());
        assert!(!OutputContainer::Mkv.is_fragmented());
    }
}