mirror of
https://github.com/rutgersc/m3u8-rs.git
synced 2025-01-03 03:38:40 +00:00
Require each M3U8 playlist to start with the #EXTM3U8 tag
The RFC requires this to be the very first line of every master/media playlist, and without this we would be parsing arbitrary text files as playlist without erroring out. See https://datatracker.ietf.org/doc/html/rfc8216#section-4.3.1.1 Fixes https://github.com/rutgersc/m3u8-rs/issues/27
This commit is contained in:
parent
a44c2a1a72
commit
65c295ee02
3 changed files with 18 additions and 11 deletions
|
@ -1,3 +1,4 @@
|
||||||
|
#EXTM3U
|
||||||
#EXTINF:10,
|
#EXTINF:10,
|
||||||
http://media.example.com/fileSequence7796.ts
|
http://media.example.com/fileSequence7796.ts
|
||||||
#EXTINF:6,
|
#EXTINF:6,
|
||||||
|
|
|
@ -1,5 +1,3 @@
|
||||||
# https://developer.apple.com/library/ios/technotes/tn2288/_index.html
|
|
||||||
#
|
|
||||||
#EXTM3U
|
#EXTM3U
|
||||||
#EXT-X-TARGETDURATION:10
|
#EXT-X-TARGETDURATION:10
|
||||||
#EXT-X-VERSION:3
|
#EXT-X-VERSION:3
|
||||||
|
@ -12,4 +10,4 @@ ad1.ts
|
||||||
#EXTINF:10.0,
|
#EXTINF:10.0,
|
||||||
movieA.ts
|
movieA.ts
|
||||||
#EXTINF:10.0,
|
#EXTINF:10.0,
|
||||||
movieB.ts
|
movieB.ts
|
||||||
|
|
|
@ -157,7 +157,13 @@ pub fn parse_playlist_res(input: &[u8]) -> Result<Playlist, IResult<&[u8], Playl
|
||||||
|
|
||||||
/// Parse input as a master playlist
|
/// Parse input as a master playlist
|
||||||
pub fn parse_master_playlist(input: &[u8]) -> IResult<&[u8], MasterPlaylist> {
|
pub fn parse_master_playlist(input: &[u8]) -> IResult<&[u8], MasterPlaylist> {
|
||||||
map(parse_master_playlist_tags, master_playlist_from_tags)(input)
|
map(
|
||||||
|
pair(
|
||||||
|
complete(pair(m3u_tag, multispace0)),
|
||||||
|
parse_master_playlist_tags,
|
||||||
|
),
|
||||||
|
|(_, tags)| master_playlist_from_tags(tags),
|
||||||
|
)(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Parse input as a master playlist
|
/// Parse input as a master playlist
|
||||||
|
@ -173,7 +179,13 @@ pub fn parse_master_playlist_res(
|
||||||
|
|
||||||
/// Parse input as a media playlist
|
/// Parse input as a media playlist
|
||||||
pub fn parse_media_playlist(input: &[u8]) -> IResult<&[u8], MediaPlaylist> {
|
pub fn parse_media_playlist(input: &[u8]) -> IResult<&[u8], MediaPlaylist> {
|
||||||
map(parse_media_playlist_tags, media_playlist_from_tags)(input)
|
map(
|
||||||
|
pair(
|
||||||
|
complete(pair(m3u_tag, multispace0)),
|
||||||
|
parse_media_playlist_tags,
|
||||||
|
),
|
||||||
|
|(_, tags)| media_playlist_from_tags(tags),
|
||||||
|
)(input)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Parse input as a media playlist
|
/// Parse input as a media playlist
|
||||||
|
@ -271,7 +283,6 @@ pub fn parse_master_playlist_tags(i: &[u8]) -> IResult<&[u8], Vec<MasterPlaylist
|
||||||
/// Contains all the tags required to parse a master playlist.
|
/// Contains all the tags required to parse a master playlist.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum MasterPlaylistTag {
|
pub enum MasterPlaylistTag {
|
||||||
M3U(String),
|
|
||||||
Version(usize),
|
Version(usize),
|
||||||
VariantStream(VariantStream),
|
VariantStream(VariantStream),
|
||||||
AlternativeMedia(AlternativeMedia),
|
AlternativeMedia(AlternativeMedia),
|
||||||
|
@ -289,7 +300,6 @@ pub fn master_playlist_tag(i: &[u8]) -> IResult<&[u8], MasterPlaylistTag> {
|
||||||
peek(take(1usize))(i)?;
|
peek(take(1usize))(i)?;
|
||||||
|
|
||||||
alt((
|
alt((
|
||||||
map(m3u_tag, MasterPlaylistTag::M3U),
|
|
||||||
map(version_tag, MasterPlaylistTag::Version),
|
map(version_tag, MasterPlaylistTag::Version),
|
||||||
map(variant_stream_tag, MasterPlaylistTag::VariantStream),
|
map(variant_stream_tag, MasterPlaylistTag::VariantStream),
|
||||||
map(variant_i_frame_stream_tag, MasterPlaylistTag::VariantStream),
|
map(variant_i_frame_stream_tag, MasterPlaylistTag::VariantStream),
|
||||||
|
@ -404,7 +414,6 @@ pub fn parse_media_playlist_tags(i: &[u8]) -> IResult<&[u8], Vec<MediaPlaylistTa
|
||||||
/// Contains all the tags required to parse a media playlist.
|
/// Contains all the tags required to parse a media playlist.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum MediaPlaylistTag {
|
pub enum MediaPlaylistTag {
|
||||||
M3U(String),
|
|
||||||
Version(usize),
|
Version(usize),
|
||||||
Segment(SegmentTag),
|
Segment(SegmentTag),
|
||||||
TargetDuration(f32),
|
TargetDuration(f32),
|
||||||
|
@ -423,7 +432,6 @@ pub fn media_playlist_tag(i: &[u8]) -> IResult<&[u8], MediaPlaylistTag> {
|
||||||
peek(take(1usize))(i)?;
|
peek(take(1usize))(i)?;
|
||||||
|
|
||||||
alt((
|
alt((
|
||||||
map(m3u_tag, MediaPlaylistTag::M3U),
|
|
||||||
map(version_tag, MediaPlaylistTag::Version),
|
map(version_tag, MediaPlaylistTag::Version),
|
||||||
map(
|
map(
|
||||||
pair(tag("#EXT-X-TARGETDURATION:"), float),
|
pair(tag("#EXT-X-TARGETDURATION:"), float),
|
||||||
|
@ -624,8 +632,8 @@ pub fn extmap(i: &[u8]) -> IResult<&[u8], Map> {
|
||||||
// Basic tags
|
// Basic tags
|
||||||
// -----------------------------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------------------------
|
||||||
|
|
||||||
pub fn m3u_tag(i: &[u8]) -> IResult<&[u8], String> {
|
pub fn m3u_tag(i: &[u8]) -> IResult<&[u8], ()> {
|
||||||
map_res(tag("#EXTM3U"), from_utf8_slice)(i)
|
map(tag("#EXTM3U"), |_| ())(i)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn version_tag(i: &[u8]) -> IResult<&[u8], usize> {
|
pub fn version_tag(i: &[u8]) -> IResult<&[u8], usize> {
|
||||||
|
|
Loading…
Reference in a new issue