@ -5,91 +5,52 @@ import re
from . turner import TurnerBaseIE
from . turner import TurnerBaseIE
from . . utils import (
from . . utils import (
ExtractorError ,
int_or_none ,
int_or_none ,
strip_or_none ,
)
)
class AdultSwimIE ( TurnerBaseIE ) :
class AdultSwimIE ( TurnerBaseIE ) :
_VALID_URL = r ' https?://(?:www \ .)?adultswim \ .com/videos/(?P< is_playlist>playlists/)?(?P<show_path>[^/]+)/(?P<episode_path>[^/?#]+)/ ?'
_VALID_URL = r ' https?://(?:www \ .)?adultswim \ .com/videos/(?P< show_path>[^/?#]+)(?:/(?P<episode_path>[^/?#]+)) ?'
_TESTS = [ {
_TESTS = [ {
' url ' : ' http://adultswim.com/videos/rick-and-morty/pilot ' ,
' url ' : ' http://adultswim.com/videos/rick-and-morty/pilot ' ,
' playlist ' : [
{
' md5 ' : ' 247572debc75c7652f253c8daa51a14d ' ,
' info_dict ' : {
' id ' : ' rQxZvXQ4ROaSOqq-or2Mow-0 ' ,
' ext ' : ' flv ' ,
' title ' : ' Rick and Morty - Pilot Part 1 ' ,
' description ' : " Rick moves in with his daughter ' s family and establishes himself as a bad influence on his grandson, Morty. "
} ,
} ,
{
' md5 ' : ' 77b0e037a4b20ec6b98671c4c379f48d ' ,
' info_dict ' : {
' id ' : ' rQxZvXQ4ROaSOqq-or2Mow-3 ' ,
' ext ' : ' flv ' ,
' title ' : ' Rick and Morty - Pilot Part 4 ' ,
' description ' : " Rick moves in with his daughter ' s family and establishes himself as a bad influence on his grandson, Morty. "
} ,
} ,
] ,
' info_dict ' : {
' info_dict ' : {
' id ' : ' rQxZvXQ4ROaSOqq-or2Mow ' ,
' id ' : ' rQxZvXQ4ROaSOqq-or2Mow ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Rick and Morty - Pilot ' ,
' title ' : ' Rick and Morty - Pilot ' ,
' description ' : " Rick moves in with his daughter ' s family and establishes himself as a bad influence on his grandson, Morty. "
' description ' : ' Rick moves in with his daughter \' s family and establishes himself as a bad influence on his grandson, Morty. ' ,
' timestamp ' : 1493267400 ,
' upload_date ' : ' 20170427 ' ,
} ,
} ,
' skip ' : ' This video is only available for registered users ' ,
' params ' : {
} , {
# m3u8 download
' url ' : ' http://www.adultswim.com/videos/playlists/american-parenting/putting-francine-out-of-business/ ' ,
' skip_download ' : True ,
' playlist ' : [
{
' md5 ' : ' 2eb5c06d0f9a1539da3718d897f13ec5 ' ,
' info_dict ' : {
' id ' : ' -t8CamQlQ2aYZ49ItZCFog-0 ' ,
' ext ' : ' flv ' ,
' title ' : ' American Dad - Putting Francine Out of Business ' ,
' description ' : ' Stan hatches a plan to get Francine out of the real estate business.Watch more American Dad on [adult swim]. '
} ,
}
] ,
' info_dict ' : {
' id ' : ' -t8CamQlQ2aYZ49ItZCFog ' ,
' title ' : ' American Dad - Putting Francine Out of Business ' ,
' description ' : ' Stan hatches a plan to get Francine out of the real estate business.Watch more American Dad on [adult swim]. '
} ,
} ,
' expected_warnings ' : [ ' Unable to download f4m manifest ' ] ,
} , {
} , {
' url ' : ' http://www.adultswim.com/videos/tim-and-eric-awesome-show-great-job/dr-steve-brule-for-your-wine/ ' ,
' url ' : ' http://www.adultswim.com/videos/tim-and-eric-awesome-show-great-job/dr-steve-brule-for-your-wine/ ' ,
' playlist ' : [
{
' md5 ' : ' 3e346a2ab0087d687a05e1e7f3b3e529 ' ,
' info_dict ' : {
' id ' : ' sY3cMUR_TbuE4YmdjzbIcQ-0 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Tim and Eric Awesome Show Great Job! - Dr. Steve Brule, For Your Wine ' ,
' description ' : ' Dr. Brule reports live from Wine Country with a special report on wines. \r \n Watch Tim and Eric Awesome Show Great Job! episode #20, " Embarrassed " on Adult Swim. \r \n \r \n ' ,
} ,
}
] ,
' info_dict ' : {
' info_dict ' : {
' id ' : ' sY3cMUR_TbuE4YmdjzbIcQ ' ,
' id ' : ' sY3cMUR_TbuE4YmdjzbIcQ ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Tim and Eric Awesome Show Great Job! - Dr. Steve Brule, For Your Wine ' ,
' title ' : ' Tim and Eric Awesome Show Great Job! - Dr. Steve Brule, For Your Wine ' ,
' description ' : ' Dr. Brule reports live from Wine Country with a special report on wines. \r \n Watch Tim and Eric Awesome Show Great Job! episode #20, " Embarrassed " on Adult Swim. \r \n \r \n ' ,
' description ' : ' Dr. Brule reports live from Wine Country with a special report on wines. \n Watch Tim and Eric Awesome Show Great Job! episode #20, " Embarrassed " on Adult Swim. ' ,
' upload_date ' : ' 20080124 ' ,
' timestamp ' : 1201150800 ,
} ,
} ,
' params ' : {
' params ' : {
# m3u8 download
# m3u8 download
' skip_download ' : True ,
' skip_download ' : True ,
}
} ,
} , {
} , {
# heroMetadata.trailer
' url ' : ' http://www.adultswim.com/videos/decker/inside-decker-a-new-hero/ ' ,
' url ' : ' http://www.adultswim.com/videos/decker/inside-decker-a-new-hero/ ' ,
' info_dict ' : {
' info_dict ' : {
' id ' : ' I0LQFQkaSUaFp8PnAWHhoQ ' ,
' id ' : ' I0LQFQkaSUaFp8PnAWHhoQ ' ,
' ext ' : ' mp4 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Decker - Inside Decker: A New Hero ' ,
' title ' : ' Decker - Inside Decker: A New Hero ' ,
' description ' : ' md5:c916df071d425d62d70c86d4399d3ee0 ' ,
' description ' : ' The guys recap the conclusion of the season. They announce a new hero, take a peek into the Victorville Film Archive and welcome back the talented James Dean. ' ,
' duration ' : 249.008 ,
' timestamp ' : 1469480460 ,
' upload_date ' : ' 20160725 ' ,
} ,
} ,
' params ' : {
' params ' : {
# m3u8 download
# m3u8 download
@ -97,136 +58,102 @@ class AdultSwimIE(TurnerBaseIE):
} ,
} ,
' expected_warnings ' : [ ' Unable to download f4m manifest ' ] ,
' expected_warnings ' : [ ' Unable to download f4m manifest ' ] ,
} , {
} , {
' url ' : ' http://www.adultswim.com/videos/toonami/friday-october-14th-2016/ ' ,
' url ' : ' http://www.adultswim.com/videos/attack-on-titan ' ,
' info_dict ' : {
' id ' : ' b7A69dzfRzuaXIECdxW8XQ ' ,
' title ' : ' Attack on Titan ' ,
' description ' : ' md5:6c8e003ea0777b47013e894767f5e114 ' ,
} ,
' playlist_mincount ' : 12 ,
} , {
' url ' : ' http://www.adultswim.com/videos/streams/williams-stream ' ,
' info_dict ' : {
' info_dict ' : {
' id ' : ' eYiLsKVgQ6qTC6agD67Sig ' ,
' id ' : ' d8DEBj7QRfetLsRgFnGEyg ' ,
' title ' : ' Toonami - Friday, October 14th, 2016 ' ,
' ext ' : ' mp4 ' ,
' description ' : ' md5:99892c96ffc85e159a428de85c30acde ' ,
' title ' : r ' re:^Williams Stream \ d {4} - \ d {2} - \ d {2} \ d {2} : \ d {2} $ ' ,
' description ' : ' original programming ' ,
} ,
} ,
' playlist ' : [ {
' md5 ' : ' ' ,
' info_dict ' : {
' id ' : ' eYiLsKVgQ6qTC6agD67Sig ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Toonami - Friday, October 14th, 2016 ' ,
' description ' : ' md5:99892c96ffc85e159a428de85c30acde ' ,
} ,
} ] ,
' params ' : {
' params ' : {
# m3u8 download
# m3u8 download
' skip_download ' : True ,
' skip_download ' : True ,
} ,
} ,
' expected_warnings ' : [ ' Unable to download f4m manifest ' ] ,
} ]
} ]
@staticmethod
def find_video_info ( collection , slug ) :
for video in collection . get ( ' videos ' ) :
if video . get ( ' slug ' ) == slug :
return video
@staticmethod
def find_collection_by_linkURL ( collections , linkURL ) :
for collection in collections :
if collection . get ( ' linkURL ' ) == linkURL :
return collection
@staticmethod
def find_collection_containing_video ( collections , slug ) :
for collection in collections :
for video in collection . get ( ' videos ' ) :
if video . get ( ' slug ' ) == slug :
return collection , video
return None , None
def _real_extract ( self , url ) :
def _real_extract ( self , url ) :
mobj = re . match ( self . _VALID_URL , url )
show_path , episode_path = re . match ( self . _VALID_URL , url ) . groups ( )
show_path = mobj . group ( ' show_path ' )
display_id = episode_path or show_path
episode_path = mobj . group ( ' episode_path ' )
webpage = self . _download_webpage ( url , display_id )
is_playlist = True if mobj . group ( ' is_playlist ' ) else False
initial_data = self . _parse_json ( self . _search_regex (
r ' AS_INITIAL_DATA(?:__)? \ s*= \ s*( { .+?}); ' ,
webpage = self . _download_webpage ( url , episode_path )
webpage , ' initial data ' ) , display_id )
# Extract the value of `bootstrappedData` from the Javascript in the page.
is_stream = show_path == ' streams '
bootstrapped_data = self . _parse_json ( self . _search_regex (
if is_stream :
r ' var bootstrappedData = ( { .*}); ' , webpage , ' bootstraped data ' ) , episode_path )
if not episode_path :
episode_path = ' live-stream '
# Downloading videos from a /videos/playlist/ URL needs to be handled differently.
# NOTE: We are only downloading one video (the current one) not the playlist
video_data = next ( stream for stream_path , stream in initial_data [ ' streams ' ] . items ( ) if stream_path == episode_path )
if is_playlist :
video_id = video_data . get ( ' stream ' )
collections = bootstrapped_data [ ' playlists ' ] [ ' collections ' ]
collection = self . find_collection_by_linkURL ( collections , show_path )
if not video_id :
video_info = self . find_video_info ( collection , episode_path )
entries = [ ]
for episode in video_data . get ( ' archiveEpisodes ' , [ ] ) :
show_title = video_info [ ' showTitle ' ]
episode_url = episode . get ( ' url ' )
segment_ids = [ video_info [ ' videoPlaybackID ' ] ]
if not episode_url :
continue
entries . append ( self . url_result (
episode_url , ' AdultSwim ' , episode . get ( ' id ' ) ) )
return self . playlist_result (
entries , video_data . get ( ' id ' ) , video_data . get ( ' title ' ) ,
strip_or_none ( video_data . get ( ' description ' ) ) )
else :
else :
collections = bootstrapped_data [ ' show ' ] [ ' collections ' ]
show_data = initial_data [ ' show ' ]
collection , video_info = self . find_collection_containing_video ( collections , episode_path )
# Video wasn't found in the collections, let's try `slugged_video`.
if not episode_path :
if video_info is None :
entries = [ ]
if bootstrapped_data . get ( ' slugged_video ' , { } ) . get ( ' slug ' ) == episode_path :
for video in show_data . get ( ' videos ' , [ ] ) :
video_info = bootstrapped_data [ ' slugged_video ' ]
slug = video . get ( ' slug ' )
if not video_info :
if not slug :
video_info = bootstrapped_data . get (
continue
' heroMetadata ' , { } ) . get ( ' trailer ' , { } ) . get ( ' video ' )
entries . append ( self . url_result (
if not video_info :
' http://adultswim.com/videos/ %s / %s ' % ( show_path , slug ) ,
video_info = bootstrapped_data . get ( ' onlineOriginals ' , [ None ] ) [ 0 ]
' AdultSwim ' , video . get ( ' id ' ) ) )
if not video_info :
return self . playlist_result (
raise ExtractorError ( ' Unable to find video info ' )
entries , show_data . get ( ' id ' ) , show_data . get ( ' title ' ) ,
strip_or_none ( show_data . get ( ' metadata ' , { } ) . get ( ' description ' ) ) )
show = bootstrapped_data [ ' show ' ]
show_title = show [ ' title ' ]
video_data = show_data [ ' sluggedVideo ' ]
stream = video_info . get ( ' stream ' )
video_id = video_data [ ' id ' ]
if stream and stream . get ( ' videoPlaybackID ' ) :
segment_ids = [ stream [ ' videoPlaybackID ' ] ]
info = self . _extract_cvp_info (
elif video_info . get ( ' clips ' ) :
' http://www.adultswim.com/videos/api/v0/assets?id= ' + video_id ,
segment_ids = [ clip [ ' videoPlaybackID ' ] for clip in video_info [ ' clips ' ] ]
video_id , {
elif video_info . get ( ' videoPlaybackID ' ) :
' secure ' : {
segment_ids = [ video_info [ ' videoPlaybackID ' ] ]
' media_src ' : ' http://androidhls-secure.cdn.turner.com/adultswim/big ' ,
elif video_info . get ( ' id ' ) :
' tokenizer_src ' : ' http://www.adultswim.com/astv/mvpd/processors/services/token_ipadAdobe.do ' ,
segment_ids = [ video_info [ ' id ' ] ]
} ,
else :
} , {
if video_info . get ( ' auth ' ) is True :
' url ' : url ,
raise ExtractorError (
' site_name ' : ' AdultSwim ' ,
' This video is only available via cable service provider subscription that '
' auth_required ' : video_data . get ( ' auth ' ) ,
' is not currently supported. You may want to use --cookies. ' , expected = True )
} )
else :
raise ExtractorError ( ' Unable to find stream or clips ' )
episode_id = video_info [ ' id ' ]
episode_title = video_info [ ' title ' ]
episode_description = video_info . get ( ' description ' )
episode_duration = int_or_none ( video_info . get ( ' duration ' ) )
view_count = int_or_none ( video_info . get ( ' views ' ) )
entries = [ ]
info . update ( {
for part_num , segment_id in enumerate ( segment_ids ) :
' id ' : video_id ,
segement_info = self . _extract_cvp_info (
' display_id ' : display_id ,
' http://www.adultswim.com/videos/api/v0/assets?id= %s &platform=desktop ' % segment_id ,
' description ' : info . get ( ' description ' ) or strip_or_none ( video_data . get ( ' description ' ) ) ,
segment_id , {
} )
' secure ' : {
if not is_stream :
' media_src ' : ' http://androidhls-secure.cdn.turner.com/adultswim/big ' ,
info . update ( {
' tokenizer_src ' : ' http://www.adultswim.com/astv/mvpd/processors/services/token_ipadAdobe.do ' ,
' duration ' : info . get ( ' duration ' ) or int_or_none ( video_data . get ( ' duration ' ) ) ,
} ,
' timestamp ' : info . get ( ' timestamp ' ) or int_or_none ( video_data . get ( ' launch_date ' ) ) ,
} )
' season_number ' : info . get ( ' season_number ' ) or int_or_none ( video_data . get ( ' season_number ' ) ) ,
segment_title = ' %s - %s ' % ( show_title , episode_title )
' episode ' : info [ ' title ' ] ,
if len ( segment_ids ) > 1 :
' episode_number ' : info . get ( ' episode_number ' ) or int_or_none ( video_data . get ( ' episode_number ' ) ) ,
segment_title + = ' Part %d ' % ( part_num + 1 )
segement_info . update ( {
' id ' : segment_id ,
' title ' : segment_title ,
' description ' : episode_description ,
} )
} )
entries . append ( segement_info )
return {
info [ ' series ' ] = video_data . get ( ' collection_title ' ) or info . get ( ' series ' )
' _type ' : ' playlist ' ,
if info [ ' series ' ] and info [ ' series ' ] != info [ ' title ' ] :
' id ' : episode_id ,
info [ ' title ' ] = ' %s - %s ' % ( info [ ' series ' ] , info [ ' title ' ] )
' display_id ' : episode_path ,
' entries ' : entries ,
return info
' title ' : ' %s - %s ' % ( show_title , episode_title ) ,
' description ' : episode_description ,
' duration ' : episode_duration ,
' view_count ' : view_count ,
}