songbird 0.6.0

An async Rust library for the Discord voice API.
use crate::{
    constants::OPUS_PASSTHROUGH_STRIKE_LIMIT,
    driver::tasks::message::*,
    input::{Compose, Input, LiveInput, Metadata, Parsed},
    tracks::{ReadyState, SeekRequest},
};
use flume::Receiver;
use rubato::FftFixedOut;
use std::time::Instant;

pub enum InputState {
    NotReady(Input),
    Preparing(PreparingInfo),
    Ready(Parsed, Option<Box<dyn Compose>>),
}

impl InputState {
    pub fn metadata(&mut self) -> Option<Metadata<'_>> {
        if let Self::Ready(parsed, _) = self {
            Some(parsed.into())
        } else {
            None
        }
    }

    #[must_use]
    pub fn ready_state(&self) -> ReadyState {
        match self {
            Self::NotReady(_) => ReadyState::Uninitialised,
            Self::Preparing(_) => ReadyState::Preparing,
            Self::Ready(_, _) => ReadyState::Playable,
        }
    }
}

impl From<Input> for InputState {
    fn from(val: Input) -> Self {
        match val {
            a @ Input::Lazy(_) => Self::NotReady(a),
            Input::Live(live, rec) => match live {
                LiveInput::Parsed(p) => Self::Ready(p, rec),
                other => Self::NotReady(Input::Live(other, rec)),
            },
        }
    }
}

pub struct PreparingInfo {
    #[allow(dead_code)]
    /// Time this request was fired.
    pub time: Instant,
    /// Used to handle seek requests fired while a track was being created (or a seek was in progress).
    pub queued_seek: Option<SeekRequest>,
    /// Callback from the thread pool to indicate the result of creating/parsing this track.
    pub callback: Receiver<MixerInputResultMessage>,
}

pub struct DecodeState {
    pub inner_pos: usize,
    pub resampler: Option<(usize, FftFixedOut<f32>, Vec<Vec<f32>>)>,
    pub passthrough: Passthrough,
    pub passthrough_violations: u8,
}

impl DecodeState {
    pub fn reset(&mut self) {
        self.inner_pos = 0;
        self.resampler = None;
    }

    pub fn record_and_check_passthrough_strike_final(&mut self, fatal: bool) -> bool {
        self.passthrough_violations = self.passthrough_violations.saturating_add(1);
        let blocked = fatal || self.passthrough_violations > OPUS_PASSTHROUGH_STRIKE_LIMIT;
        if blocked {
            self.passthrough = Passthrough::Block;
        }
        blocked
    }
}

impl Default for DecodeState {
    fn default() -> Self {
        Self {
            inner_pos: 0,
            resampler: None,
            passthrough: Passthrough::Inactive,
            passthrough_violations: 0,
        }
    }
}

/// Simple state to manage decoder resets etc.
///
/// Inactive->Active transitions should trigger a reset.
///
/// Block should be used if a source contains known-bad packets:
/// it's unlikely that packet sizes will vary, but if they do then
/// we can't passthrough (and every attempt will trigger a codec reset,
/// which probably won't sound too smooth).
#[derive(Clone, Copy, Eq, PartialEq)]
pub enum Passthrough {
    Active,
    Inactive,
    Block,
}