str0m 0.18.0

WebRTC library in Sans-IO style
Documentation
use super::{FeedbackMessageType, ReportList, RtcpHeader, RtcpPacket, RtcpType, Ssrc};

/// RTCP packet BY
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct Goodbye {
    /// The SSRC that are no longer in use.
    pub reports: ReportList<Ssrc>,
}

impl RtcpPacket for Goodbye {
    fn header(&self) -> RtcpHeader {
        RtcpHeader {
            rtcp_type: RtcpType::Goodbye,
            feedback_message_type: FeedbackMessageType::SourceCount(self.reports.len() as u8),
            words_less_one: (self.length_words() - 1) as u16,
        }
    }

    fn length_words(&self) -> usize {
        // each ssrc is one word
        1 + self.reports.len()
    }

    fn write_to(&self, buf: &mut [u8]) -> usize {
        self.header().write_to(&mut buf[..4]);
        let buf = &mut buf[4..];
        for (i, s) in self.reports.iter().enumerate() {
            buf[i * 4..(i + 1) * 4].copy_from_slice(&s.to_be_bytes());
        }

        self.length_words() * 4
    }
}

impl<'a> TryFrom<(usize, &'a [u8])> for Goodbye {
    type Error = &'static str;

    fn try_from((count, buf): (usize, &'a [u8])) -> Result<Self, Self::Error> {
        if buf.len() < 4 {
            return Err("Less than 4 bytes for Goodbye");
        }

        let mut reports = ReportList::new();
        let mut buf = buf;

        if buf.len() < count * 4 {
            return Err("Less than count * 4 bytes for Goodbye");
        }

        let max = count.min(31);

        for _ in 0..max {
            let ssrc = u32::from_be_bytes([buf[0], buf[1], buf[2], buf[3]]).into();
            reports.push(ssrc);
            buf = &buf[4..];
        }

        // TODO parse the optional reason string here.

        Ok(Goodbye { reports })
    }
}