mousiki 0.2.0

Pure Rust Opus codec.
Documentation
use mousiki::{
    Application, Bitrate, Channels, Decoder, Encoder, EncoderBuilderError, OpusDecodeError,
    OpusDecoderInitError, OpusEncodeError,
};
use std::env;
use std::fs::File;
use std::io::{self, Read, Write};
use std::path::Path;

const FRAME_SIZE: usize = 960;
const SAMPLE_RATE: u32 = 48_000;
const CHANNELS: Channels = Channels::Stereo;
const APPLICATION: Application = Application::Audio;
const BITRATE: Bitrate = Bitrate::Bits(64_000);

const MAX_FRAME_SIZE: usize = 6 * 960;
const MAX_PACKET_SIZE: usize = 3 * 1276;

fn main() {
    if let Err(err) = run() {
        report_error(err);
        std::process::exit(1);
    }
}

fn run() -> Result<(), ExampleError> {
    let mut args = env::args_os();
    let _program = args.next();
    let input = args.next().ok_or(ExampleError::Usage)?;
    let output = args.next().ok_or(ExampleError::Usage)?;
    if args.next().is_some() {
        return Err(ExampleError::Usage);
    }

    let input_path = Path::new(&input);
    let output_path = Path::new(&output);

    let mut input_file =
        File::open(input_path).map_err(|err| ExampleError::Io("open input", err.kind()))?;
    let mut output_file =
        File::create(output_path).map_err(|err| ExampleError::Io("create output", err.kind()))?;

    let mut encoder = Encoder::builder(SAMPLE_RATE, CHANNELS, APPLICATION)
        .bitrate(BITRATE)
        .build()
        .map_err(ExampleError::EncoderBuild)?;
    let mut decoder = Decoder::new(SAMPLE_RATE, CHANNELS).map_err(ExampleError::DecoderInit)?;

    let channels = CHANNELS.count();
    let mut input_bytes = vec![0u8; FRAME_SIZE * channels * 2];
    let mut input_pcm = vec![0i16; FRAME_SIZE * channels];
    let mut output_pcm = vec![0i16; MAX_FRAME_SIZE * channels];
    let mut output_bytes = vec![0u8; MAX_FRAME_SIZE * channels * 2];
    let mut packet = vec![0u8; MAX_PACKET_SIZE];

    loop {
        match input_file.read_exact(&mut input_bytes) {
            Ok(()) => {}
            Err(err) if err.kind() == io::ErrorKind::UnexpectedEof => break,
            Err(err) => return Err(ExampleError::Io("read input", err.kind())),
        }

        for (sample, chunk) in input_pcm.iter_mut().zip(input_bytes.chunks_exact(2)) {
            *sample = i16::from_le_bytes([chunk[0], chunk[1]]);
        }

        let packet_len = encoder
            .encode(&input_pcm, &mut packet)
            .map_err(ExampleError::Encode)?;

        let decoded = decoder
            .decode(&packet[..packet_len], &mut output_pcm, false)
            .map_err(ExampleError::Decode)?;

        let total_samples = decoded * channels;
        for (chunk, &sample) in output_bytes
            .chunks_exact_mut(2)
            .take(total_samples)
            .zip(output_pcm.iter().take(total_samples))
        {
            let le = sample.to_le_bytes();
            chunk[0] = le[0];
            chunk[1] = le[1];
        }

        output_file
            .write_all(&output_bytes[..total_samples * 2])
            .map_err(|err| ExampleError::Io("write output", err.kind()))?;
    }

    Ok(())
}

fn report_error(err: ExampleError) {
    match err {
        ExampleError::Usage => {
            eprintln!("usage: trivial_example <input.pcm> <output.pcm>");
            eprintln!("input and output are 16-bit little-endian raw files");
        }
        ExampleError::Io(context, kind) => {
            eprintln!("IO error ({context}): {kind:?}");
        }
        ExampleError::DecoderInit(err) => {
            eprintln!("failed to create decoder: {err:?}");
        }
        ExampleError::EncoderBuild(err) => {
            eprintln!("failed to create encoder: {err:?}");
        }
        ExampleError::Encode(err) => {
            eprintln!("encode failed: {err:?}");
        }
        ExampleError::Decode(err) => {
            eprintln!("decode failed: {err:?}");
        }
    }
}

enum ExampleError {
    Usage,
    Io(&'static str, io::ErrorKind),
    EncoderBuild(EncoderBuilderError),
    DecoderInit(OpusDecoderInitError),
    Encode(OpusEncodeError),
    Decode(OpusDecodeError),
}