2013-07-10 15:49:11 +00:00
# encoding: utf-8
2014-01-06 00:47:52 +00:00
from __future__ import unicode_literals
2013-06-23 18:31:45 +00:00
import os
import re
from . common import InfoExtractor
2014-01-06 00:42:58 +00:00
from . youtube import YoutubeIE
2013-06-23 18:31:45 +00:00
from . . utils import (
compat_urllib_parse ,
2013-08-28 10:47:27 +00:00
compat_urlparse ,
2014-02-21 15:59:10 +00:00
compat_xml_parse_error ,
2013-06-23 18:31:45 +00:00
ExtractorError ,
2014-08-24 00:02:17 +00:00
float_or_none ,
2013-12-20 16:05:28 +00:00
HEADRequest ,
2014-08-22 16:19:56 +00:00
orderedSet ,
2014-03-10 16:31:32 +00:00
parse_xml ,
2013-10-15 10:05:13 +00:00
smuggle_url ,
unescapeHTML ,
2013-12-17 11:33:55 +00:00
unified_strdate ,
2014-08-24 02:47:18 +00:00
unsmuggle_url ,
2013-12-17 11:33:55 +00:00
url_basename ,
2013-06-23 18:31:45 +00:00
)
2013-07-10 15:49:11 +00:00
from . brightcove import BrightcoveIE
2013-12-19 19:28:52 +00:00
from . ooyala import OoyalaIE
2014-03-16 19:00:31 +00:00
from . rutv import RUTVIE
2014-03-28 12:58:49 +00:00
from . smotri import SmotriIE
2013-06-23 18:31:45 +00:00
2013-08-24 20:49:52 +00:00
2013-06-23 18:31:45 +00:00
class GenericIE ( InfoExtractor ) :
2014-01-06 00:47:52 +00:00
IE_DESC = ' Generic downloader that works on some sites '
2013-06-23 18:31:45 +00:00
_VALID_URL = r ' .* '
2014-01-06 00:47:52 +00:00
IE_NAME = ' generic '
2013-07-10 15:49:11 +00:00
_TESTS = [
{
2014-01-06 00:47:52 +00:00
' url ' : ' http://www.hodiho.fr/2013/02/regis-plante-sa-jeep.html ' ,
2014-04-14 11:51:46 +00:00
' md5 ' : ' 85b90ccc9d73b4acd9138d3af4c27f89 ' ,
2014-01-06 00:47:52 +00:00
' info_dict ' : {
2014-04-14 11:51:46 +00:00
' id ' : ' 13601338388002 ' ,
' ext ' : ' mp4 ' ,
2014-01-06 00:47:52 +00:00
' uploader ' : ' www.hodiho.fr ' ,
' title ' : ' R \u00e9 gis plante sa Jeep ' ,
2013-07-10 15:49:11 +00:00
}
} ,
2013-10-27 13:40:25 +00:00
# bandcamp page with custom domain
{
2014-01-06 00:47:52 +00:00
' add_ie ' : [ ' Bandcamp ' ] ,
' url ' : ' http://bronyrock.com/track/the-pony-mash ' ,
' info_dict ' : {
2014-04-14 11:56:29 +00:00
' id ' : ' 3235767654 ' ,
' ext ' : ' mp3 ' ,
2014-01-06 00:47:52 +00:00
' title ' : ' The Pony Mash ' ,
' uploader ' : ' M_Pallante ' ,
2013-10-27 13:40:25 +00:00
} ,
2014-01-06 00:47:52 +00:00
' skip ' : ' There is a limit of 200 free downloads / month for the test song ' ,
2013-10-27 13:40:25 +00:00
} ,
2013-11-06 15:40:24 +00:00
# embedded brightcove video
2013-11-07 20:06:48 +00:00
# it also tests brightcove videos that need to set the 'Referer' in the
# http requests
2013-11-06 15:40:24 +00:00
{
2014-01-06 00:47:52 +00:00
' add_ie ' : [ ' Brightcove ' ] ,
' url ' : ' http://www.bfmtv.com/video/bfmbusiness/cours-bourse/cours-bourse-l-analyse-technique-154522/ ' ,
' info_dict ' : {
' id ' : ' 2765128793001 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Le cours de bourse : l’ analyse technique ' ,
' description ' : ' md5:7e9ad046e968cb2d1114004aba466fd9 ' ,
' uploader ' : ' BFM BUSINESS ' ,
2013-11-06 15:40:24 +00:00
} ,
2014-01-06 00:47:52 +00:00
' params ' : {
' skip_download ' : True ,
2013-11-06 15:40:24 +00:00
} ,
} ,
2014-01-28 02:35:32 +00:00
{
# https://github.com/rg3/youtube-dl/issues/2253
' url ' : ' http://bcove.me/i6nfkrc3 ' ,
' md5 ' : ' 0ba9446db037002366bab3b3eb30c88c ' ,
' info_dict ' : {
2014-04-14 11:56:29 +00:00
' id ' : ' 3101154703001 ' ,
' ext ' : ' mp4 ' ,
2014-01-28 02:35:32 +00:00
' title ' : ' Still no power ' ,
' uploader ' : ' thestar.com ' ,
' description ' : ' Mississauga resident David Farmer is still out of power as a result of the ice storm a month ago. To keep the house warm, Farmer cuts wood from his property for a wood burning stove downstairs. ' ,
} ,
' add_ie ' : [ ' Brightcove ' ] ,
} ,
2014-04-01 13:17:35 +00:00
{
' url ' : ' http://www.championat.com/video/football/v/87/87499.html ' ,
' md5 ' : ' fb973ecf6e4a78a67453647444222983 ' ,
' info_dict ' : {
' id ' : ' 3414141473001 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Видео. Удаление Дзагоева (ЦСКА) ' ,
' description ' : ' Онлайн-трансляция матча ЦСКА - " Волга " ' ,
' uploader ' : ' Championat ' ,
} ,
} ,
2013-12-17 11:33:55 +00:00
# Direct link to a video
{
2014-01-06 00:47:52 +00:00
' url ' : ' http://media.w3.org/2010/05/sintel/trailer.mp4 ' ,
' md5 ' : ' 67d406c2bcb6af27fa886f31aa934bbe ' ,
' info_dict ' : {
' id ' : ' trailer ' ,
2014-02-27 06:21:59 +00:00
' ext ' : ' mp4 ' ,
2014-01-06 00:47:52 +00:00
' title ' : ' trailer ' ,
' upload_date ' : ' 20100513 ' ,
2013-12-17 11:33:55 +00:00
}
2013-12-19 19:28:52 +00:00
} ,
# ooyala video
{
2014-01-06 00:47:52 +00:00
' url ' : ' http://www.rollingstone.com/music/videos/norwegian-dj-cashmere-cat-goes-spartan-on-with-me-premiere-20131219 ' ,
' md5 ' : ' 5644c6ca5d5782c1d0d350dad9bd840c ' ,
' info_dict ' : {
' id ' : ' BwY2RxaTrTkslxOfcan0UCf0YqyvWysJ ' ,
' ext ' : ' mp4 ' ,
2014-01-21 00:40:34 +00:00
' title ' : ' 2cc213299525360.mov ' , # that's what we get
2013-12-19 19:28:52 +00:00
} ,
} ,
2014-02-27 06:21:59 +00:00
# google redirect
{
' url ' : ' http://www.google.com/url?sa=t&rct=j&q=&esrc=s&source=web&cd=1&cad=rja&ved=0CCUQtwIwAA&url=http % 3A %2F %2F www.youtube.com %2F watch %3F v % 3DcmQHVoWB5FY&ei=F-sNU-LLCaXk4QT52ICQBQ&usg=AFQjCNEw4hL29zgOohLXvpJ-Bdh2bils1Q&bvm=bv.61965928,d.bGE ' ,
' info_dict ' : {
' id ' : ' cmQHVoWB5FY ' ,
' ext ' : ' mp4 ' ,
' upload_date ' : ' 20130224 ' ,
' uploader_id ' : ' TheVerge ' ,
' description ' : ' Chris Ziegler takes a look at the Alcatel OneTouch Fire and the ZTE Open; two of the first Firefox OS handsets to be officially announced. ' ,
' uploader ' : ' The Verge ' ,
' title ' : ' First Firefox OS phones side-by-side ' ,
} ,
' params ' : {
' skip_download ' : False ,
}
2014-03-05 13:01:53 +00:00
} ,
2014-02-24 00:15:51 +00:00
# embed.ly video
{
' url ' : ' http://www.tested.com/science/weird/460206-tested-grinding-coffee-2000-frames-second/ ' ,
' info_dict ' : {
' id ' : ' 9ODmcdjQcHQ ' ,
' ext ' : ' mp4 ' ,
2014-03-05 13:05:44 +00:00
' title ' : ' Tested: Grinding Coffee at 2000 Frames Per Second ' ,
' upload_date ' : ' 20140225 ' ,
' description ' : ' md5:06a40fbf30b220468f1e0957c0f558ff ' ,
' uploader ' : ' Tested ' ,
' uploader_id ' : ' testedcom ' ,
2014-02-24 00:15:51 +00:00
} ,
# No need to test YoutubeIE here
' params ' : {
' skip_download ' : True ,
} ,
} ,
2014-03-11 15:51:36 +00:00
# funnyordie embed
{
' url ' : ' http://www.theguardian.com/world/2014/mar/11/obama-zach-galifianakis-between-two-ferns ' ,
' md5 ' : ' 7cf780be104d40fea7bae52eed4a470e ' ,
' info_dict ' : {
' id ' : ' 18e820ec3f ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Between Two Ferns with Zach Galifianakis: President Barack Obama ' ,
' description ' : ' Episode 18: President Barack Obama sits down with Zach Galifianakis for his most memorable interview yet. ' ,
2014-03-16 19:00:31 +00:00
} ,
2014-03-11 15:51:36 +00:00
} ,
2014-03-16 19:00:31 +00:00
# RUTV embed
{
' url ' : ' http://www.rg.ru/2014/03/15/reg-dfo/anklav-anons.html ' ,
' info_dict ' : {
' id ' : ' 776940 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Охотское море стало целиком российским ' ,
' description ' : ' md5:5ed62483b14663e2a95ebbe115eb8f43 ' ,
} ,
' params ' : {
# m3u8 download
' skip_download ' : True ,
} ,
2014-03-20 15:33:23 +00:00
} ,
# Embedded TED video
{
' url ' : ' http://en.support.wordpress.com/videos/ted-talks/ ' ,
' md5 ' : ' deeeabcc1085eb2ba205474e7235a3d5 ' ,
' info_dict ' : {
' id ' : ' 981 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' My web playroom ' ,
' uploader ' : ' Ze Frank ' ,
' description ' : ' md5:ddb2a40ecd6b6a147e400e535874947b ' ,
}
2014-03-11 15:51:36 +00:00
} ,
2014-04-04 14:23:09 +00:00
# Embeded Ustream video
{
' url ' : ' http://www.american.edu/spa/pti/nsa-privacy-janus-2014.cfm ' ,
' md5 ' : ' 27b99cdb639c9b12a79bca876a073417 ' ,
' info_dict ' : {
2014-04-04 16:56:29 +00:00
' id ' : ' 45734260 ' ,
' ext ' : ' flv ' ,
' uploader ' : ' AU SPA: The NSA and Privacy ' ,
2014-04-04 14:23:09 +00:00
' title ' : ' NSA and Privacy Forum Debate featuring General Hayden and Barton Gellman '
}
} ,
2014-03-14 21:39:53 +00:00
# nowvideo embed hidden behind percent encoding
{
' url ' : ' http://www.waoanime.tv/the-super-dimension-fortress-macross-episode-1/ ' ,
' md5 ' : ' 2baf4ddd70f697d94b1c18cf796d5107 ' ,
' info_dict ' : {
' id ' : ' 06e53103ca9aa ' ,
' ext ' : ' flv ' ,
' title ' : ' Macross Episode 001 Watch Macross Episode 001 onl ' ,
' description ' : ' No description ' ,
} ,
2014-03-21 21:14:24 +00:00
} ,
2014-03-24 21:01:47 +00:00
# arte embed
{
' url ' : ' http://www.tv-replay.fr/redirection/20-03-14/x-enius-arte-10753389.html ' ,
' md5 ' : ' 7653032cbb25bf6c80d80f217055fa43 ' ,
' info_dict ' : {
' id ' : ' 048195-004_PLUS7-F ' ,
' ext ' : ' flv ' ,
' title ' : ' X:enius ' ,
' description ' : ' md5:d5fdf32ef6613cdbfd516ae658abf168 ' ,
' upload_date ' : ' 20140320 ' ,
} ,
' params ' : {
' skip_download ' : ' Requires rtmpdump '
}
} ,
2014-03-28 12:58:49 +00:00
# smotri embed
{
' url ' : ' http://rbctv.rbc.ru/archive/news/562949990879132.shtml ' ,
' md5 ' : ' ec40048448e9284c9a1de77bb188108b ' ,
' info_dict ' : {
' id ' : ' v27008541fad ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Крым и Севастополь вошли в состав России ' ,
' description ' : ' md5:fae01b61f68984c7bd2fa741e11c3175 ' ,
' duration ' : 900 ,
' upload_date ' : ' 20140318 ' ,
' uploader ' : ' rbctv_2012_4 ' ,
' uploader_id ' : ' rbctv_2012_4 ' ,
} ,
} ,
2014-04-21 03:47:52 +00:00
# Condé Nast embed
{
' url ' : ' http://www.wired.com/2014/04/honda-asimo/ ' ,
' md5 ' : ' ba0dfe966fa007657bd1443ee672db0f ' ,
' info_dict ' : {
' id ' : ' 53501be369702d3275860000 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Honda’ s New Asimo Robot Is More Human Than Ever ' ,
}
2014-04-29 23:46:06 +00:00
} ,
# Dailymotion embed
{
' url ' : ' http://www.spi0n.com/zap-spi0n-com-n216/ ' ,
' md5 ' : ' 441aeeb82eb72c422c7f14ec533999cd ' ,
' info_dict ' : {
' id ' : ' k2mm4bCdJ6CQ2i7c8o2 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Le Zap de Spi0n n°216 - Zapping du Web ' ,
' uploader ' : ' Spi0n ' ,
} ,
' add_ie ' : [ ' Dailymotion ' ] ,
2014-06-09 20:06:45 +00:00
} ,
# YouTube embed
{
' url ' : ' http://www.badzine.de/ansicht/datum/2014/06/09/so-funktioniert-die-neue-englische-badminton-liga.html ' ,
' info_dict ' : {
' id ' : ' FXRb4ykk4S0 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' The NBL Auction 2014 ' ,
' uploader ' : ' BADMINTON England ' ,
' uploader_id ' : ' BADMINTONEvents ' ,
' upload_date ' : ' 20140603 ' ,
' description ' : ' md5:9ef128a69f1e262a700ed83edb163a73 ' ,
} ,
' add_ie ' : [ ' Youtube ' ] ,
' params ' : {
' skip_download ' : True ,
}
} ,
2014-06-22 19:38:04 +00:00
# MTVSercices embed
{
' url ' : ' http://www.gametrailers.com/news-post/76093/north-america-europe-is-getting-that-mario-kart-8-mercedes-dlc-too ' ,
' md5 ' : ' 35727f82f58c76d996fc188f9755b0d5 ' ,
' info_dict ' : {
' id ' : ' 0306a69b-8adf-4fb5-aace-75f8e8cbfca9 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Review ' ,
' description ' : ' Mario \' s life in the fast lane has never looked so good. ' ,
} ,
} ,
2014-05-21 09:55:37 +00:00
# YouTube embed via <data-embed-url="">
{
' url ' : ' https://play.google.com/store/apps/details?id=com.gameloft.android.ANMP.GloftA8HM ' ,
' info_dict ' : {
2014-08-22 16:19:56 +00:00
' id ' : ' jpSGZsgga_I ' ,
2014-05-21 09:55:37 +00:00
' ext ' : ' mp4 ' ,
2014-08-22 16:19:56 +00:00
' title ' : ' Asphalt 8: Airborne - Launch Trailer ' ,
' uploader ' : ' Gameloft ' ,
' uploader_id ' : ' gameloft ' ,
' upload_date ' : ' 20130821 ' ,
' description ' : ' md5:87bd95f13d8be3e7da87a5f2c443106a ' ,
} ,
' params ' : {
' skip_download ' : True ,
2014-05-21 09:55:37 +00:00
}
2014-08-24 00:02:17 +00:00
} ,
# Camtasia studio
{
' url ' : ' http://www.ll.mit.edu/workshops/education/videocourses/antennas/lecture1/video/ ' ,
' playlist ' : [ {
' md5 ' : ' 0c5e352edabf715d762b0ad4e6d9ee67 ' ,
' info_dict ' : {
' id ' : ' Fenn-AA_PA_Radar_Course_Lecture_1c_Final ' ,
' title ' : ' Fenn-AA_PA_Radar_Course_Lecture_1c_Final - video1 ' ,
' ext ' : ' flv ' ,
' duration ' : 2235.90 ,
}
} , {
' md5 ' : ' 10e4bb3aaca9fd630e273ff92d9f3c63 ' ,
' info_dict ' : {
' id ' : ' Fenn-AA_PA_Radar_Course_Lecture_1c_Final_PIP ' ,
' title ' : ' Fenn-AA_PA_Radar_Course_Lecture_1c_Final - pip ' ,
' ext ' : ' flv ' ,
' duration ' : 2235.93 ,
}
} ] ,
' info_dict ' : {
' title ' : ' Fenn-AA_PA_Radar_Course_Lecture_1c_Final ' ,
}
2014-08-24 03:31:32 +00:00
} ,
# Flowplayer
{
' url ' : ' http://www.handjobhub.com/video/busty-blonde-siri-tit-fuck-while-wank-6313.html ' ,
' md5 ' : ' 9d65602bf31c6e20014319c7d07fba27 ' ,
' info_dict ' : {
' id ' : ' 5123ea6d5e5a7 ' ,
' ext ' : ' mp4 ' ,
' age_limit ' : 18 ,
' uploader ' : ' www.handjobhub.com ' ,
' title ' : ' Busty Blonde Siri Tit Fuck While Wank at Handjob Hub ' ,
}
2014-08-25 16:03:01 +00:00
} ,
# RSS feed
{
' url ' : ' http://phihag.de/2014/youtube-dl/rss2.xml ' ,
' info_dict ' : {
' id ' : ' http://phihag.de/2014/youtube-dl/rss2.xml ' ,
' title ' : ' Zero Punctuation ' ,
' description ' : ' re: '
} ,
' playlist_mincount ' : 11 ,
2014-04-21 03:47:52 +00:00
}
2013-07-10 15:49:11 +00:00
]
2013-06-23 18:31:45 +00:00
def report_download_webpage ( self , video_id ) :
""" Report webpage download. """
if not self . _downloader . params . get ( ' test ' , False ) :
2014-01-06 00:47:52 +00:00
self . _downloader . report_warning ( ' Falling back on generic information extractor. ' )
2013-06-23 18:31:45 +00:00
super ( GenericIE , self ) . report_download_webpage ( video_id )
def report_following_redirect ( self , new_url ) :
""" Report information extraction. """
2014-01-06 00:47:52 +00:00
self . _downloader . to_screen ( ' [redirect] Following redirect to %s ' % new_url )
2013-06-23 18:31:45 +00:00
2014-02-20 12:14:05 +00:00
def _extract_rss ( self , url , video_id , doc ) :
playlist_title = doc . find ( ' ./channel/title ' ) . text
playlist_desc_el = doc . find ( ' ./channel/description ' )
playlist_desc = None if playlist_desc_el is None else playlist_desc_el . text
entries = [ {
' _type ' : ' url ' ,
' url ' : e . find ( ' link ' ) . text ,
' title ' : e . find ( ' title ' ) . text ,
} for e in doc . findall ( ' ./channel/item ' ) ]
return {
' _type ' : ' playlist ' ,
' id ' : url ,
' title ' : playlist_title ,
' description ' : playlist_desc ,
' entries ' : entries ,
}
2014-08-24 00:02:17 +00:00
def _extract_camtasia ( self , url , video_id , webpage ) :
""" Returns None if no camtasia video can be found. """
camtasia_cfg = self . _search_regex (
r ' fo \ .addVariable \ ( \ s* " csConfigFile " , \ s* " ([^ " ]+) " \ s* \ ); ' ,
webpage , ' camtasia configuration file ' , default = None )
if camtasia_cfg is None :
return None
title = self . _html_search_meta ( ' DC.title ' , webpage , fatal = True )
camtasia_url = compat_urlparse . urljoin ( url , camtasia_cfg )
camtasia_cfg = self . _download_xml (
camtasia_url , video_id ,
note = ' Downloading camtasia configuration ' ,
errnote = ' Failed to download camtasia configuration ' )
fileset_node = camtasia_cfg . find ( ' ./playlist/array/fileset ' )
entries = [ ]
for n in fileset_node . getchildren ( ) :
url_n = n . find ( ' ./uri ' )
if url_n is None :
continue
entries . append ( {
' id ' : os . path . splitext ( url_n . text . rpartition ( ' / ' ) [ 2 ] ) [ 0 ] ,
' title ' : ' %s - %s ' % ( title , n . tag ) ,
' url ' : compat_urlparse . urljoin ( url , url_n . text ) ,
' duration ' : float_or_none ( n . find ( ' ./duration ' ) . text ) ,
} )
return {
' _type ' : ' playlist ' ,
' entries ' : entries ,
' title ' : title ,
}
2013-06-23 18:31:45 +00:00
def _real_extract ( self , url ) :
2014-04-29 23:46:06 +00:00
if url . startswith ( ' // ' ) :
return {
' _type ' : ' url ' ,
2014-05-05 01:12:41 +00:00
' url ' : self . http_scheme ( ) + url ,
2014-04-29 23:46:06 +00:00
}
2013-09-06 16:39:35 +00:00
parsed_url = compat_urlparse . urlparse ( url )
if not parsed_url . scheme :
2014-01-22 13:16:43 +00:00
default_search = self . _downloader . params . get ( ' default_search ' )
if default_search is None :
2014-07-29 15:17:43 +00:00
default_search = ' fixup_error '
2014-01-22 13:16:43 +00:00
2014-07-29 15:17:43 +00:00
if default_search in ( ' auto ' , ' auto_warning ' , ' fixup_error ' ) :
2014-01-22 13:16:43 +00:00
if ' / ' in url :
self . _downloader . report_warning ( ' The url doesn \' t specify the protocol, trying with http ' )
return self . url_result ( ' http:// ' + url )
2014-07-29 15:17:43 +00:00
elif default_search != ' fixup_error ' :
2014-03-30 13:57:31 +00:00
if default_search == ' auto_warning ' :
2014-05-19 15:10:11 +00:00
if re . match ( r ' ^(?:url|URL)$ ' , url ) :
raise ExtractorError (
' Invalid URL: %r . Call youtube-dl like this: youtube-dl -v " https://www.youtube.com/watch?v=BaW_jenozKc " ' % url ,
expected = True )
else :
self . _downloader . report_warning (
2014-07-06 09:22:44 +00:00
' Falling back to youtube search for %s . Set --default-search " auto " to suppress this warning. ' % url )
2014-01-22 13:16:43 +00:00
return self . url_result ( ' ytsearch: ' + url )
2014-07-29 15:17:43 +00:00
if default_search in ( ' error ' , ' fixup_error ' ) :
2014-07-06 09:22:44 +00:00
raise ExtractorError (
( ' %r is not a valid URL. '
2014-07-21 00:37:44 +00:00
' Set --default-search " ytsearch " (or run youtube-dl " ytsearch: %s " ) to search YouTube '
2014-07-06 09:22:44 +00:00
) % ( url , url ) , expected = True )
2014-01-22 13:16:43 +00:00
else :
assert ' : ' in default_search
return self . url_result ( default_search + url )
2014-08-24 02:47:18 +00:00
url , smuggled_data = unsmuggle_url ( url )
force_videoid = None
if smuggled_data and ' force_videoid ' in smuggled_data :
force_videoid = smuggled_data [ ' force_videoid ' ]
video_id = force_videoid
else :
video_id = os . path . splitext ( url . rstrip ( ' / ' ) . split ( ' / ' ) [ - 1 ] ) [ 0 ]
2013-09-06 16:39:35 +00:00
2014-01-06 00:47:52 +00:00
self . to_screen ( ' %s : Requesting header ' % video_id )
2013-12-27 07:38:42 +00:00
2014-08-24 04:58:11 +00:00
head_req = HEADRequest ( url )
response = self . _request_webpage (
head_req , video_id ,
note = False , errnote = ' Could not send HEAD request to %s ' % url ,
fatal = False )
2013-12-17 11:33:55 +00:00
2014-08-24 04:58:11 +00:00
if response is not False :
2013-12-17 11:33:55 +00:00
# Check for redirect
new_url = response . geturl ( )
if url != new_url :
self . report_following_redirect ( new_url )
2014-08-24 02:47:18 +00:00
if force_videoid :
new_url = smuggle_url (
new_url , { ' force_videoid ' : force_videoid } )
2013-12-17 11:04:33 +00:00
return self . url_result ( new_url )
2013-12-17 11:33:55 +00:00
# Check for direct link to a video
content_type = response . headers . get ( ' Content-Type ' , ' ' )
2013-12-17 15:26:32 +00:00
m = re . match ( r ' ^(?P<type>audio|video|application(?=/ogg$))/(?P<format_id>.+)$ ' , content_type )
2013-12-17 11:33:55 +00:00
if m :
upload_date = response . headers . get ( ' Last-Modified ' )
if upload_date :
upload_date = unified_strdate ( upload_date )
return {
' id ' : video_id ,
' title ' : os . path . splitext ( url_basename ( url ) ) [ 0 ] ,
' formats ' : [ {
' format_id ' : m . group ( ' format_id ' ) ,
' url ' : url ,
2014-01-06 00:47:52 +00:00
' vcodec ' : ' none ' if m . group ( ' type ' ) == ' audio ' else None
2013-12-17 11:33:55 +00:00
} ] ,
' upload_date ' : upload_date ,
}
2013-06-23 18:31:45 +00:00
try :
webpage = self . _download_webpage ( url , video_id )
except ValueError :
# since this is the last-resort InfoExtractor, if
# this error is thrown, it'll be thrown here
2014-01-06 00:47:52 +00:00
raise ExtractorError ( ' Failed to download URL: %s ' % url )
2013-06-23 18:31:45 +00:00
self . report_extraction ( video_id )
2013-11-18 12:28:26 +00:00
2014-02-20 12:14:05 +00:00
# Is it an RSS feed?
try :
2014-03-10 16:31:32 +00:00
doc = parse_xml ( webpage )
2014-02-20 12:14:05 +00:00
if doc . tag == ' rss ' :
return self . _extract_rss ( url , video_id , doc )
2014-02-21 15:59:10 +00:00
except compat_xml_parse_error :
2014-02-20 12:14:05 +00:00
pass
2014-08-24 00:02:17 +00:00
# Is it a Camtasia project?
camtasia_res = self . _extract_camtasia ( url , video_id , webpage )
if camtasia_res is not None :
return camtasia_res
2014-03-14 21:38:49 +00:00
# Sometimes embedded video player is hidden behind percent encoding
# (e.g. https://github.com/rg3/youtube-dl/issues/2448)
# Unescaping the whole page allows to handle those cases in a generic way
2014-02-24 16:44:31 +00:00
webpage = compat_urllib_parse . unquote ( webpage )
2013-11-18 12:28:26 +00:00
# it's tempting to parse this further, but you would
# have to take into account all the variations like
# Video Title - Site Name
# Site Name | Video Title
# Video Title - Tagline | Site Name
# and so on and so forth; it's just not practical
2013-12-06 08:15:04 +00:00
video_title = self . _html_search_regex (
2014-01-06 00:47:52 +00:00
r ' (?s)<title>(.*?)</title> ' , webpage , ' video title ' ,
default = ' video ' )
2013-12-06 08:15:04 +00:00
2014-08-24 03:31:32 +00:00
# Try to detect age limit automatically
age_limit = self . _rta_search ( webpage )
# And then there are the jokers who advertise that they use RTA,
# but actually don't.
AGE_LIMIT_MARKERS = [
r ' Proudly Labeled <a href= " http://www.rtalabel.org/ " title= " Restricted to Adults " >RTA</a> ' ,
]
if any ( re . search ( marker , webpage ) for marker in AGE_LIMIT_MARKERS ) :
age_limit = 18
2013-12-06 08:15:04 +00:00
# video uploader is domain name
video_uploader = self . _search_regex (
2014-01-06 00:47:52 +00:00
r ' ^(?:https?://)?([^/]*)/.* ' , url , ' video uploader ' )
2013-11-18 12:28:26 +00:00
2014-08-22 16:19:56 +00:00
# Helper method
def _playlist_from_matches ( matches , getter , ie = None ) :
urlrs = orderedSet ( self . url_result ( getter ( m ) , ie ) for m in matches )
return self . playlist_result (
urlrs , playlist_id = video_id , playlist_title = video_title )
2013-08-26 19:29:31 +00:00
# Look for BrightCove:
2014-02-03 14:19:40 +00:00
bc_urls = BrightcoveIE . _extract_brightcove_urls ( webpage )
if bc_urls :
2014-01-06 00:47:52 +00:00
self . to_screen ( ' Brightcove video detected. ' )
2014-02-03 14:19:40 +00:00
entries = [ {
' _type ' : ' url ' ,
' url ' : smuggle_url ( bc_url , { ' Referer ' : url } ) ,
' ie_key ' : ' Brightcove '
} for bc_url in bc_urls ]
return {
' _type ' : ' playlist ' ,
' title ' : video_title ,
' id ' : video_id ,
' entries ' : entries ,
}
2013-07-10 15:49:11 +00:00
2013-12-22 02:34:13 +00:00
# Look for embedded (iframe) Vimeo player
2013-10-15 10:05:13 +00:00
mobj = re . search (
2014-03-15 17:47:04 +00:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>(?:https?:)?//player \ .vimeo \ .com/video/.+?) \ 1 ' , webpage )
2013-10-15 10:05:13 +00:00
if mobj :
2014-03-15 17:47:04 +00:00
player_url = unescapeHTML ( mobj . group ( ' url ' ) )
2013-10-15 10:05:13 +00:00
surl = smuggle_url ( player_url , { ' Referer ' : url } )
return self . url_result ( surl , ' Vimeo ' )
2013-12-22 02:34:13 +00:00
# Look for embedded (swf embed) Vimeo player
mobj = re . search (
2014-01-29 21:26:46 +00:00
r ' <embed[^>]+?src= " (https?://(?:www \ .)?vimeo \ .com/moogaloop \ .swf.+?) " ' , webpage )
2013-12-22 02:34:13 +00:00
if mobj :
return self . url_result ( mobj . group ( 1 ) , ' Vimeo ' )
2013-10-18 09:44:57 +00:00
# Look for embedded YouTube player
2013-12-19 19:44:30 +00:00
matches = re . findall ( r ''' (?x)
2014-06-09 20:06:45 +00:00
( ? :
< iframe [ ^ > ] + ? src = |
2014-08-22 15:40:36 +00:00
data - video - url = |
2014-06-09 20:06:45 +00:00
< embed [ ^ > ] + ? src = |
embedSWF \( ? : \s *
)
( [ " \' ])
( ? P < url > ( ? : https ? : ) ? / / ( ? : www \. ) ? youtube \. com /
2013-12-19 19:44:30 +00:00
( ? : embed | v ) / . + ? )
\1 ''' , webpage)
2013-11-18 12:28:26 +00:00
if matches :
2014-08-22 16:19:56 +00:00
return _playlist_from_matches (
matches , lambda m : unescapeHTML ( m [ 1 ] ) , ie = ' Youtube ' )
2013-10-18 09:44:57 +00:00
2013-12-01 00:21:33 +00:00
# Look for embedded Dailymotion player
matches = re . findall (
2013-12-06 08:15:04 +00:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>(?:https?:)?//(?:www \ .)?dailymotion \ .com/embed/video/.+?) \ 1 ' , webpage )
2013-12-01 00:21:33 +00:00
if matches :
2014-08-22 16:19:56 +00:00
return _playlist_from_matches (
matches , lambda m : unescapeHTML ( m [ 1 ] ) )
2013-12-01 00:21:33 +00:00
2013-12-06 08:15:04 +00:00
# Look for embedded Wistia player
match = re . search (
r ' <iframe[^>]+?src=([ " \' ])(?P<url>(?:https?:)?//(?:fast \ .)?wistia \ .net/embed/iframe/.+?) \ 1 ' , webpage )
if match :
return {
' _type ' : ' url_transparent ' ,
' url ' : unescapeHTML ( match . group ( ' url ' ) ) ,
' ie_key ' : ' Wistia ' ,
' uploader ' : video_uploader ,
' title ' : video_title ,
' id ' : video_id ,
}
2013-12-16 19:08:23 +00:00
# Look for embedded blip.tv player
2013-12-30 05:15:02 +00:00
mobj = re . search ( r ' <meta \ s[^>]*https?://api \ .blip \ .tv/ \ w+/redirect/ \ w+/( \ d+) ' , webpage )
2013-12-16 19:08:23 +00:00
if mobj :
2013-12-30 05:15:02 +00:00
return self . url_result ( ' http://blip.tv/a/a- ' + mobj . group ( 1 ) , ' BlipTV ' )
2014-08-24 01:20:31 +00:00
mobj = re . search ( r ' <(?:iframe|embed|object) \ s[^>]*(https?://(?: \ w+ \ .)?blip \ .tv/(?:play/|api \ .swf#)[a-zA-Z0-9_]+) ' , webpage )
2013-12-16 19:08:23 +00:00
if mobj :
2013-12-30 05:15:02 +00:00
return self . url_result ( mobj . group ( 1 ) , ' BlipTV ' )
2013-12-16 19:08:23 +00:00
2014-04-21 03:47:52 +00:00
# Look for embedded condenast player
matches = re . findall (
r ' <iframe \ s+(?:[a-zA-Z-]+= " [^ " ]+ " \ s+)*?src= " (https?://player \ .cnevids \ .com/embed/[^ " ]+ " ) ' ,
webpage )
if matches :
return {
' _type ' : ' playlist ' ,
' entries ' : [ {
' _type ' : ' url ' ,
' ie_key ' : ' CondeNast ' ,
' url ' : ma ,
} for ma in matches ] ,
' title ' : video_title ,
' id ' : video_id ,
}
2013-10-27 13:40:25 +00:00
# Look for Bandcamp pages with custom domain
mobj = re . search ( r ' <meta property= " og:url " [^>]*?content= " (.*?bandcamp \ .com.*?) " ' , webpage )
if mobj is not None :
burl = unescapeHTML ( mobj . group ( 1 ) )
2013-11-22 15:05:14 +00:00
# Don't set the extractor because it can be a track url or an album
return self . url_result ( burl )
2013-10-27 13:40:25 +00:00
2013-12-16 20:45:21 +00:00
# Look for embedded Vevo player
mobj = re . search (
r ' <iframe[^>]+?src=([ " \' ])(?P<url>(?:https?:)?//(?:cache \ .)?vevo \ .com/.+?) \ 1 ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) )
2013-12-19 19:28:52 +00:00
# Look for Ooyala videos
2014-03-21 20:51:33 +00:00
mobj = ( re . search ( r ' player.ooyala.com/[^ " ?]+ \ ?[^ " ]*?(?:embedCode|ec)=(?P<ec>[^ " &]+) ' , webpage ) or
re . search ( r ' OO.Player.create \ ([ \' " ].*?[ \' " ], \ s*[ \' " ](?P<ec>. {32} )[ \' " ] ' , webpage ) )
2013-12-19 19:28:52 +00:00
if mobj is not None :
2014-03-21 20:51:33 +00:00
return OoyalaIE . _build_url_result ( mobj . group ( ' ec ' ) )
2013-12-19 19:28:52 +00:00
2013-12-20 16:05:28 +00:00
# Look for Aparat videos
2014-04-21 10:37:41 +00:00
mobj = re . search ( r ' <iframe .*?src= " (http://www \ .aparat \ .com/video/[^ " ]+) " ' , webpage )
2013-12-20 16:05:28 +00:00
if mobj is not None :
return self . url_result ( mobj . group ( 1 ) , ' Aparat ' )
2014-01-07 07:07:46 +00:00
# Look for MPORA videos
2014-01-29 21:26:46 +00:00
mobj = re . search ( r ' <iframe .*?src= " (http://mpora \ .(?:com|de)/videos/[^ " ]+) " ' , webpage )
2014-01-07 07:07:46 +00:00
if mobj is not None :
return self . url_result ( mobj . group ( 1 ) , ' Mpora ' )
2014-01-08 01:11:46 +00:00
2014-04-05 10:20:05 +00:00
# Look for embedded NovaMov-based player
2014-01-08 01:07:11 +00:00
mobj = re . search (
2014-05-17 11:12:12 +00:00
r ''' (?x)<(?:pagespeed_)?iframe[^>]+?src=([ " \ ' ])
2014-04-05 10:20:05 +00:00
( ? P < url > http : / / ( ? : ( ? : embed | www ) \. ) ?
( ? : novamov \. com |
nowvideo \. ( ? : ch | sx | eu | at | ag | co ) |
videoweed \. ( ? : es | com ) |
movshare \. ( ? : net | sx | ag ) |
divxstage \. ( ? : eu | net | ch | co | at | ag ) )
/ embed \. php . + ? ) \1 ''' , webpage)
2014-01-08 01:07:11 +00:00
if mobj is not None :
2014-04-05 10:20:05 +00:00
return self . url_result ( mobj . group ( ' url ' ) )
2014-04-05 08:49:45 +00:00
2014-01-21 17:10:14 +00:00
# Look for embedded Facebook player
mobj = re . search (
2014-01-27 04:47:30 +00:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https://www \ .facebook \ .com/video/embed.+?) \ 1 ' , webpage )
2014-01-21 17:10:14 +00:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Facebook ' )
2014-02-28 16:51:54 +00:00
# Look for embedded VK player
mobj = re . search ( r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://vk \ .com/video_ext \ .php.+?) \ 1 ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' VK ' )
2014-06-29 13:18:23 +00:00
# Look for embedded ivi player
mobj = re . search ( r ' <embed[^>]+?src=([ " \' ])(?P<url>https?://(?:www \ .)?ivi \ .ru/video/player.+?) \ 1 ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Ivi ' )
2014-01-27 04:47:30 +00:00
# Look for embedded Huffington Post player
mobj = re . search (
2014-01-29 21:26:46 +00:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://embed \ .live \ .huffingtonpost \ .com/.+?) \ 1 ' , webpage )
2014-01-27 04:47:30 +00:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' HuffPost ' )
2014-02-24 00:15:51 +00:00
# Look for embed.ly
mobj = re . search ( r ' class=[ " \' ]embedly-card[ " \' ][^>]href=[ " \' ](?P<url>[^ " \' ]+) ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) )
mobj = re . search ( r ' class=[ " \' ]embedly-embed[ " \' ][^>]src=[ " \' ][^ " \' ]*url=(?P<url>[^&]+) ' , webpage )
if mobj is not None :
return self . url_result ( compat_urllib_parse . unquote ( mobj . group ( ' url ' ) ) )
2014-03-11 15:51:36 +00:00
# Look for funnyordie embed
matches = re . findall ( r ' <iframe[^>]+?src= " (https?://(?:www \ .)?funnyordie \ .com/embed/[^ " ]+) " ' , webpage )
if matches :
2014-08-22 16:19:56 +00:00
return _playlist_from_matches (
matches , getter = unescapeHTML , ie = ' FunnyOrDie ' )
2014-03-11 15:51:36 +00:00
2014-03-16 19:00:31 +00:00
# Look for embedded RUTV player
rutv_url = RUTVIE . _extract_url ( webpage )
if rutv_url :
return self . url_result ( rutv_url , ' RUTV ' )
2014-03-22 09:20:44 +00:00
# Look for embedded TED player
mobj = re . search (
r ' <iframe[^>]+?src=([ " \' ])(?P<url>http://embed \ .ted \ .com/.+?) \ 1 ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' TED ' )
2014-04-04 14:23:09 +00:00
# Look for embedded Ustream videos
mobj = re . search (
r ' <iframe[^>]+?src=([ " \' ])(?P<url>http://www \ .ustream \ .tv/embed/.+?) \ 1 ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Ustream ' )
2014-03-24 21:01:47 +00:00
# Look for embedded arte.tv player
mobj = re . search (
r ' <script [^>]*?src= " (?P<url>http://www \ .arte \ .tv/playerv2/embed[^ " ]+) " ' ,
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' ArteTVEmbed ' )
2014-03-28 12:58:49 +00:00
# Look for embedded smotri.com player
smotri_url = SmotriIE . _extract_url ( webpage )
if smotri_url :
return self . url_result ( smotri_url , ' Smotri ' )
2014-05-05 01:12:41 +00:00
# Look for embeded soundcloud player
mobj = re . search (
r ' <iframe src= " (?P<url>https?://(?:w \ .)?soundcloud \ .com/player[^ " ]+) " ' ,
webpage )
if mobj is not None :
url = unescapeHTML ( mobj . group ( ' url ' ) )
return self . url_result ( url )
2014-06-09 21:06:25 +00:00
# Look for embedded vulture.com player
mobj = re . search (
r ' <iframe src= " (?P<url>https?://video \ .vulture \ .com/[^ " ]+) " ' ,
webpage )
if mobj is not None :
url = unescapeHTML ( mobj . group ( ' url ' ) )
return self . url_result ( url , ie = ' Vulture ' )
2014-06-22 19:38:04 +00:00
# Look for embedded mtvservices player
mobj = re . search (
r ' <iframe src= " (?P<url>https?://media \ .mtvnservices \ .com/embed/[^ " ]+) " ' ,
webpage )
if mobj is not None :
url = unescapeHTML ( mobj . group ( ' url ' ) )
return self . url_result ( url , ie = ' MTVServicesEmbedded ' )
2014-08-16 06:56:22 +00:00
# Look for embedded yahoo player
mobj = re . search (
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://(?:screen|movies) \ .yahoo \ .com/.+? \ .html \ ?format=embed) \ 1 ' ,
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Yahoo ' )
2014-08-23 13:20:49 +00:00
# Look for embedded sbs.com.au player
mobj = re . search (
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://(?:www \ .)sbs \ .com \ .au/ondemand/video/single/.+?) \ 1 ' ,
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' SBS ' )
2013-06-23 18:31:45 +00:00
# Start with something easy: JW Player in SWFObject
2014-04-30 00:23:51 +00:00
found = re . findall ( r ' flashvars: [ \' " ](?:.*&)?file=(http[^ \' " &]*) ' , webpage )
if not found :
2014-01-05 04:34:06 +00:00
# Look for gorilla-vid style embedding
2014-04-30 00:23:51 +00:00
found = re . findall ( r ''' (?sx)
2014-04-21 14:16:53 +00:00
( ? :
jw_plugins |
JWPlayerOptions |
jwplayer \s * \( \s * [ " ' ][^ ' " ] + [ " ' ] \ s* \ ) \ s* \ .setup
)
. * ? file \s * : \s * [ " \' ](.*?)[ " \' ] ' ' ' , webpage )
2014-04-30 00:23:51 +00:00
if not found :
2013-06-23 18:31:45 +00:00
# Broaden the search a little bit
2014-04-30 00:23:51 +00:00
found = re . findall ( r ' [^A-Za-z0-9]?(?:file|source)=(http[^ \' " &]*) ' , webpage )
if not found :
# Broaden the findall a little bit: JWPlayer JS loader
found = re . findall ( r ' [^A-Za-z0-9]?file[ " \' ]?: \ s*[ " \' ](http(?![^ \' " ]+ \ .[0-9]+[ \' " ])[^ \' " ]+)[ " \' ] ' , webpage )
2014-08-24 03:31:32 +00:00
if not found :
# Flow player
found = re . findall ( r ''' (?xs)
flowplayer \( " [^ " ] + " , \ s*
\{ [ ^ } ] + ? \} \s * ,
\s * { [ ^ } ] + ? [ " ' ]?clip[ " ' ]? \ s*: \ s* \ { \ s*
[ " ' ]?url[ " ' ]? \ s*: \ s*[ " ' ] ( [ ^ " ' ]+)[ " ' ]
''' , webpage)
2014-04-30 00:23:51 +00:00
if not found :
2013-06-23 18:31:45 +00:00
# Try to find twitter cards info
2014-04-30 00:23:51 +00:00
found = re . findall ( r ' <meta (?:property|name)= " twitter:player:stream " (?:content|value)= " (.+?) " ' , webpage )
if not found :
2013-06-23 18:31:45 +00:00
# We look for Open Graph info:
# We have to match any number spaces between elements, some sites try to align them (eg.: statigr.am)
2014-04-30 00:23:51 +00:00
m_video_type = re . findall ( r ' <meta.*?property= " og:video:type " .*?content= " video/(.*?) " ' , webpage )
2013-06-23 18:31:45 +00:00
# We only look in og:video if the MIME type is a video, don't try if it's a Flash player:
if m_video_type is not None :
2014-08-24 00:24:47 +00:00
def check_video ( vurl ) :
vpath = compat_urlparse . urlparse ( vurl ) . path
2014-08-27 00:07:11 +00:00
vext = determine_ext ( vpath )
return ' . ' in vpath and vext not in ( ' swf ' , ' png ' , ' jpg ' )
2014-08-24 00:24:47 +00:00
found = list ( filter (
check_video ,
re . findall ( r ' <meta.*?property= " og:video " .*?content= " (.*?) " ' , webpage ) ) )
2014-04-30 00:23:51 +00:00
if not found :
2013-08-21 02:32:22 +00:00
# HTML5 video
2014-08-27 00:09:59 +00:00
found = re . findall ( r ' (?s)<video[^<]*(?:>.*?<source[^>]+)? src= " ([^ " ]+) " ' , webpage )
2014-04-30 00:23:51 +00:00
if not found :
2014-05-16 13:32:53 +00:00
found = re . search (
2014-02-27 06:21:59 +00:00
r ' (?i)<meta \ s+(?=(?:[a-z-]+= " [^ " ]+ " \ s+)*http-equiv= " refresh " ) '
r ' (?:[a-z-]+= " [^ " ]+ " \ s+)*?content= " [0-9] { ,2};url= \' ([^ \' ]+) \' " ' ,
webpage )
2014-04-30 00:23:51 +00:00
if found :
new_url = found . group ( 1 )
2014-02-27 06:21:59 +00:00
self . report_following_redirect ( new_url )
return {
' _type ' : ' url ' ,
' url ' : new_url ,
}
2014-04-30 00:23:51 +00:00
if not found :
2014-01-06 00:47:52 +00:00
raise ExtractorError ( ' Unsupported URL: %s ' % url )
2013-06-23 18:31:45 +00:00
2014-04-30 00:23:51 +00:00
entries = [ ]
for video_url in found :
video_url = compat_urlparse . urljoin ( url , video_url )
video_id = compat_urllib_parse . unquote ( os . path . basename ( video_url ) )
2013-06-23 18:31:45 +00:00
2014-04-30 00:23:51 +00:00
# Sometimes, jwplayer extraction will result in a YouTube URL
if YoutubeIE . suitable ( video_url ) :
entries . append ( self . url_result ( video_url , ' Youtube ' ) )
continue
2013-06-23 18:31:45 +00:00
2014-04-30 00:23:51 +00:00
# here's a fun little line of code for you:
video_id = os . path . splitext ( video_id ) [ 0 ]
2014-01-06 00:42:58 +00:00
2014-04-30 00:23:51 +00:00
entries . append ( {
' id ' : video_id ,
' url ' : video_url ,
' uploader ' : video_uploader ,
' title ' : video_title ,
2014-08-24 03:31:32 +00:00
' age_limit ' : age_limit ,
2014-04-30 00:23:51 +00:00
} )
if len ( entries ) == 1 :
2014-05-01 09:28:37 +00:00
return entries [ 0 ]
2014-04-30 00:23:51 +00:00
else :
for num , e in enumerate ( entries , start = 1 ) :
e [ ' title ' ] = ' %s ( %d ) ' % ( e [ ' title ' ] , num )
return {
' _type ' : ' playlist ' ,
' entries ' : entries ,
}
2013-06-23 18:31:45 +00:00