# Internal: Parses out and exposes the parts of M3U8 playlist files. # # M3U8 References: # http://developer.apple.com/library/ios/#documentation/networkinginternet/conceptual/streamingmediaguide/HTTPStreamingArchitecture/HTTPStreamingArchitecture.html # # Examples # # p = Playlist.new(File.read("/path/to/playlist.m3u8"), "http://url.tld/where/playlist/was/downloaded/from") # # => # module HLSpider class Playlist # Public: Gets/Sets the raw M3U8 Playlist File. attr_accessor :file # Public: Gets/Sets Optional source of playlist file. Used only for reference. attr_accessor :source # Public: Gets sub-playlists if the playlist has child playlists (Variable Rate Playlist). attr_reader :playlists # Public: Gets segments contained in the playlist. attr_reader :segments # Public: Gets the target duration if available. attr_reader :target_duration # Public: Gets the media sequence of the playlist. attr_reader :media_sequence # Internal: Initialize a Playlist. # # file - A String containing an .m3u8 playlist file. # source - A String source of where the playlist was downloaded from. (optional) def initialize(file, source = nil) @file = file @source = source @valid = false @variable_playlist = false @segment_playlist = false @playlists = [] @segments = [] parse(@file) end # Internal: Set the m3u8 file. # # file - The String of the m3u8 file. # # Examples # # file( File.read('/path/to/playlist.m3u8') ) # # => '#EXTM3U\n#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=713245\n # http://hls.telvue.com/brightstar/2-1/playlist.m3u8?wowzasessionid=268983957' # # Returns the file String. def file=(file) @file = file parse(@file) end # Public: Check whether the playlist is a variable playlist or not. # # # Examples # # variable_playlist? # # => true # # Returns Boolean variable_playlist. def variable_playlist? @variable_playlist end # Public: Check whether the playlist is a segment playlist or not. # # # Examples # # segment_playlist? # # => false # # Returns Boolean segment_playlist. def segment_playlist? @segment_playlist end # Public: Check whether the playlist is valid (either a segment or variable playlist). # # # Examples # # valid? # # => true # # Returns Boolean valid. def valid? @valid end # Public: Sub-Playlists of playlist file. Appends source if # playlists are not absolute urls. # # # # Examples # # playlists # # => ["http://site.tld/playlist_1.m3u8", "http://site.tld/playlist_2.m3u8"] # # Returns Array of Strings. def playlists @playlists.collect do |p| if absolute_url?(p) p elsif @source @source.sub(/[^\/]*.m3u8/, p) end end end # Public: Prints contents of @file. # # # Examples # # to_s # #=> '#EXTM3U\n#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=713245\n # http://hls.telvue.com/brightstar/2-1/playlist.m3u8?wowzasessionid=268983957' # # Returns String file. def inspect @file end alias_method :inspect, :to_s private include PlaylistLine # Internal: Parses @file and sets @variable_playlist, @segment_playlist, and @valid. # # # Examples # # parse(playlist_file) # # Returns nothing. def parse(file) @valid = true if /#EXTM3U/.match(@file) if has_playlist?(@file) && !has_segment?(@file) @variable_playlist = true @file.each_line do |line| @playlists << line[/([^ "]+.m3u8[^ "]*)/].strip if has_playlist?(line) end elsif has_segment?(@file) && !has_playlist?(@file) @segment_playlist = true @file.each_line do |line| if has_segment?(line) @segments << line[/([^ "]+.(ts|aac)[^ "]*)/].strip elsif duration_line?(line) @target_duration = parse_duration(line.strip) elsif media_sequence_line?(line) @media_sequence = parse_sequence(line.strip) end end else @valid = false end end end end