#[derive(Debug, Clone, PartialEq, Eq)]
pub struct MediaDescription {
pub media: String,
pub payload_type: u8,
pub encoding: Option<String>,
pub clock_rate: Option<u32>,
pub aac_size_length: Option<u8>,
pub aac_index_length: Option<u8>,
pub control: Option<String>,
}
#[derive(Debug, Clone, Default, PartialEq, Eq)]
pub struct Sdp {
pub media: Vec<MediaDescription>,
}
impl Sdp {
pub fn parse(text: &str) -> Sdp {
let mut media: Vec<MediaDescription> = Vec::new();
for line in text.lines() {
let line = line.trim_end();
if let Some(rest) = line.strip_prefix("m=") {
let mut parts = rest.split(' ');
let kind = parts.next().unwrap_or("").to_string();
let pt = parts.nth(2).and_then(|p| p.parse().ok()).unwrap_or(0);
media.push(MediaDescription {
media: kind,
payload_type: pt,
encoding: None,
clock_rate: None,
aac_size_length: None,
aac_index_length: None,
control: None,
});
} else if let Some(rest) = line.strip_prefix("a=") {
let Some(current) = media.last_mut() else {
continue;
};
if let Some(rtpmap) = rest.strip_prefix("rtpmap:") {
if let Some((_, enc)) = rtpmap.split_once(' ') {
let mut fields = enc.split('/');
current.encoding = fields.next().map(|s| s.to_string());
current.clock_rate = fields.next().and_then(|s| s.parse().ok());
}
} else if let Some(fmtp) = rest.strip_prefix("fmtp:") {
for param in fmtp.split([';', ' ']) {
if let Some(v) = param.trim().strip_prefix("sizelength=") {
current.aac_size_length = v.parse().ok();
} else if let Some(v) = param.trim().strip_prefix("indexlength=") {
current.aac_index_length = v.parse().ok();
}
}
} else if let Some(control) = rest.strip_prefix("control:") {
current.control = Some(control.to_string());
}
}
}
Sdp { media }
}
pub fn first_control(&self, kind: &str, base_url: &str) -> Option<String> {
let media = self.media.iter().find(|m| m.media == kind)?;
let control = media.control.as_deref().unwrap_or("");
if control.is_empty() || control == "*" {
Some(base_url.to_string())
} else if control.starts_with("rtsp://") {
Some(control.to_string())
} else {
Some(format!("{}/{}", base_url.trim_end_matches('/'), control))
}
}
pub fn first_video_control(&self, base_url: &str) -> Option<String> {
self.first_control("video", base_url)
}
pub fn first_audio_control(&self, base_url: &str) -> Option<String> {
self.first_control("audio", base_url)
}
pub fn audio_aac_lengths(&self) -> (u8, u8) {
let audio = self.media.iter().find(|m| m.media == "audio");
let size = audio.and_then(|m| m.aac_size_length).unwrap_or(13);
let index = audio.and_then(|m| m.aac_index_length).unwrap_or(3);
(size, index)
}
pub fn has_aac_audio(&self) -> bool {
self.media.iter().any(|m| {
m.media == "audio"
&& m.encoding
.as_deref()
.is_some_and(|e| e.eq_ignore_ascii_case("MPEG4-GENERIC"))
})
}
}
#[cfg(test)]
mod tests {
use super::*;
const SAMPLE: &str = "v=0\r\n\
o=- 0 0 IN IP4 127.0.0.1\r\n\
s=Session\r\n\
m=video 0 RTP/AVP 96\r\n\
a=rtpmap:96 H264/90000\r\n\
a=control:trackID=1\r\n\
m=audio 0 RTP/AVP 97\r\n\
a=rtpmap:97 MPEG4-GENERIC/48000/2\r\n\
a=fmtp:97 streamtype=5;profile-level-id=1;mode=AAC-hbr;sizelength=13;indexlength=3;indexdeltalength=3\r\n\
a=control:trackID=2\r\n";
#[test]
fn parses_both_media_tracks() {
let sdp = Sdp::parse(SAMPLE);
assert_eq!(sdp.media.len(), 2);
let v = &sdp.media[0];
assert_eq!(v.media, "video");
assert_eq!(v.payload_type, 96);
assert_eq!(v.encoding.as_deref(), Some("H264"));
assert_eq!(v.control.as_deref(), Some("trackID=1"));
assert_eq!(sdp.media[1].encoding.as_deref(), Some("MPEG4-GENERIC"));
}
#[test]
fn parses_aac_fmtp_lengths_and_falls_back_to_defaults() {
let sdp = Sdp::parse(SAMPLE);
assert_eq!(sdp.media[1].aac_size_length, Some(13));
assert_eq!(sdp.media[1].aac_index_length, Some(3));
assert_eq!(sdp.audio_aac_lengths(), (13, 3));
let no_fmtp = Sdp::parse("m=audio 0 RTP/AVP 97\r\na=rtpmap:97 MPEG4-GENERIC/44100\r\n");
assert_eq!(no_fmtp.audio_aac_lengths(), (13, 3));
let custom = Sdp::parse(
"m=audio 0 RTP/AVP 97\r\na=fmtp:97 mode=AAC-hbr;sizelength=6;indexlength=2\r\n",
);
assert_eq!(custom.audio_aac_lengths(), (6, 2));
}
#[test]
fn builds_relative_control_url() {
let sdp = Sdp::parse(SAMPLE);
assert_eq!(
sdp.first_video_control("rtsp://cam/stream"),
Some("rtsp://cam/stream/trackID=1".to_string())
);
}
#[test]
fn absolute_control_url_overrides_base() {
let sdp = Sdp::parse("m=video 0 RTP/AVP 96\r\na=control:rtsp://cam/abs\r\n");
assert_eq!(
sdp.first_video_control("rtsp://cam/stream"),
Some("rtsp://cam/abs".to_string())
);
}
#[test]
fn empty_sdp_yields_no_media() {
assert!(Sdp::parse("v=0\r\n").media.is_empty());
}
#[test]
fn resolves_audio_control_and_detects_aac() {
let sdp = Sdp::parse(SAMPLE);
assert!(sdp.has_aac_audio());
assert_eq!(
sdp.first_audio_control("rtsp://cam/stream"),
Some("rtsp://cam/stream/trackID=2".to_string())
);
}
}