2013-07-10 17:49:11 +02:00
# encoding: utf-8
2014-01-06 01:47:52 +01:00
from __future__ import unicode_literals
2013-06-23 20:31:45 +02:00
import os
import re
from . common import InfoExtractor
2014-01-06 01:42:58 +01:00
from . youtube import YoutubeIE
2013-06-23 20:31:45 +02:00
from . . utils import (
compat_urllib_error ,
compat_urllib_parse ,
compat_urllib_request ,
2013-08-28 12:47:27 +02:00
compat_urlparse ,
2014-02-21 16:59:10 +01:00
compat_xml_parse_error ,
2013-06-23 20:31:45 +02:00
ExtractorError ,
2013-12-20 17:05:28 +01:00
HEADRequest ,
2014-03-10 17:31:32 +01:00
parse_xml ,
2013-10-15 12:05:13 +02:00
smuggle_url ,
unescapeHTML ,
2013-12-17 12:33:55 +01:00
unified_strdate ,
url_basename ,
2013-06-23 20:31:45 +02:00
)
2013-07-10 17:49:11 +02:00
from . brightcove import BrightcoveIE
2013-12-19 20:28:52 +01:00
from . ooyala import OoyalaIE
2014-03-17 02:00:31 +07:00
from . rutv import RUTVIE
2014-03-28 19:58:49 +07:00
from . smotri import SmotriIE
2013-06-23 20:31:45 +02:00
2013-08-24 22:49:52 +02:00
2013-06-23 20:31:45 +02:00
class GenericIE ( InfoExtractor ) :
2014-01-06 01:47:52 +01:00
IE_DESC = ' Generic downloader that works on some sites '
2013-06-23 20:31:45 +02:00
_VALID_URL = r ' .* '
2014-01-06 01:47:52 +01:00
IE_NAME = ' generic '
2013-07-10 17:49:11 +02:00
_TESTS = [
{
2014-01-06 01:47:52 +01:00
' url ' : ' http://www.hodiho.fr/2013/02/regis-plante-sa-jeep.html ' ,
2014-04-14 18:51:46 +07:00
' md5 ' : ' 85b90ccc9d73b4acd9138d3af4c27f89 ' ,
2014-01-06 01:47:52 +01:00
' info_dict ' : {
2014-04-14 18:51:46 +07:00
' id ' : ' 13601338388002 ' ,
' ext ' : ' mp4 ' ,
2014-01-06 01:47:52 +01:00
' uploader ' : ' www.hodiho.fr ' ,
' title ' : ' R \u00e9 gis plante sa Jeep ' ,
2013-07-10 17:49:11 +02:00
}
} ,
2013-10-27 14:40:25 +01:00
# bandcamp page with custom domain
{
2014-01-06 01:47:52 +01:00
' add_ie ' : [ ' Bandcamp ' ] ,
' url ' : ' http://bronyrock.com/track/the-pony-mash ' ,
' info_dict ' : {
2014-04-14 18:56:29 +07:00
' id ' : ' 3235767654 ' ,
' ext ' : ' mp3 ' ,
2014-01-06 01:47:52 +01:00
' title ' : ' The Pony Mash ' ,
' uploader ' : ' M_Pallante ' ,
2013-10-27 14:40:25 +01:00
} ,
2014-01-06 01:47:52 +01:00
' skip ' : ' There is a limit of 200 free downloads / month for the test song ' ,
2013-10-27 14:40:25 +01:00
} ,
2013-11-06 16:40:24 +01:00
# embedded brightcove video
2013-11-07 21:06:48 +01:00
# it also tests brightcove videos that need to set the 'Referer' in the
# http requests
2013-11-06 16:40:24 +01:00
{
2014-01-06 01:47:52 +01: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 16:40:24 +01:00
} ,
2014-01-06 01:47:52 +01:00
' params ' : {
' skip_download ' : True ,
2013-11-06 16:40:24 +01:00
} ,
} ,
2014-01-28 03:35:32 +01:00
{
# https://github.com/rg3/youtube-dl/issues/2253
' url ' : ' http://bcove.me/i6nfkrc3 ' ,
' md5 ' : ' 0ba9446db037002366bab3b3eb30c88c ' ,
' info_dict ' : {
2014-04-14 18:56:29 +07:00
' id ' : ' 3101154703001 ' ,
' ext ' : ' mp4 ' ,
2014-01-28 03:35:32 +01: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 20:17:35 +07: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 12:33:55 +01:00
# Direct link to a video
{
2014-01-06 01:47:52 +01:00
' url ' : ' http://media.w3.org/2010/05/sintel/trailer.mp4 ' ,
' md5 ' : ' 67d406c2bcb6af27fa886f31aa934bbe ' ,
' info_dict ' : {
' id ' : ' trailer ' ,
2014-02-27 07:21:59 +01:00
' ext ' : ' mp4 ' ,
2014-01-06 01:47:52 +01:00
' title ' : ' trailer ' ,
' upload_date ' : ' 20100513 ' ,
2013-12-17 12:33:55 +01:00
}
2013-12-19 20:28:52 +01:00
} ,
# ooyala video
{
2014-01-06 01:47:52 +01: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 01:40:34 +01:00
' title ' : ' 2cc213299525360.mov ' , # that's what we get
2013-12-19 20:28:52 +01:00
} ,
} ,
2014-02-27 07:21:59 +01: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 14:01:53 +01:00
} ,
2014-02-24 01:15:51 +01: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 14:05:44 +01:00
' title ' : ' Tested: Grinding Coffee at 2000 Frames Per Second ' ,
' upload_date ' : ' 20140225 ' ,
' description ' : ' md5:06a40fbf30b220468f1e0957c0f558ff ' ,
' uploader ' : ' Tested ' ,
' uploader_id ' : ' testedcom ' ,
2014-02-24 01:15:51 +01:00
} ,
# No need to test YoutubeIE here
' params ' : {
' skip_download ' : True ,
} ,
} ,
2014-03-11 16:51:36 +01: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-17 02:00:31 +07:00
} ,
2014-03-11 16:51:36 +01:00
} ,
2014-03-17 02:00:31 +07: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 16:33:23 +01: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 16:51:36 +01:00
} ,
2014-04-05 00:53:09 +10:30
# Embeded Ustream video
{
' url ' : ' http://www.american.edu/spa/pti/nsa-privacy-janus-2014.cfm ' ,
' md5 ' : ' 27b99cdb639c9b12a79bca876a073417 ' ,
' info_dict ' : {
2014-04-05 03:26:29 +10:30
' id ' : ' 45734260 ' ,
' ext ' : ' flv ' ,
' uploader ' : ' AU SPA: The NSA and Privacy ' ,
2014-04-05 00:53:09 +10:30
' title ' : ' NSA and Privacy Forum Debate featuring General Hayden and Barton Gellman '
}
} ,
2014-03-15 04:39:53 +07: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 22:14:24 +01:00
} ,
2014-03-24 22:01:47 +01: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 19:58:49 +07: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 05:47:52 +02: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-30 01:46:06 +02: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 22:06:45 +02: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 21:38:04 +02: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. ' ,
} ,
} ,
2013-07-10 17:49:11 +02:00
]
2013-06-23 20:31:45 +02:00
def report_download_webpage ( self , video_id ) :
""" Report webpage download. """
if not self . _downloader . params . get ( ' test ' , False ) :
2014-01-06 01:47:52 +01:00
self . _downloader . report_warning ( ' Falling back on generic information extractor. ' )
2013-06-23 20:31:45 +02:00
super ( GenericIE , self ) . report_download_webpage ( video_id )
def report_following_redirect ( self , new_url ) :
""" Report information extraction. """
2014-01-06 01:47:52 +01:00
self . _downloader . to_screen ( ' [redirect] Following redirect to %s ' % new_url )
2013-06-23 20:31:45 +02:00
2013-12-17 12:33:55 +01:00
def _send_head ( self , url ) :
2013-06-23 20:31:45 +02:00
""" Check if it is a redirect, like url shorteners, in case return the new url. """
class HEADRedirectHandler ( compat_urllib_request . HTTPRedirectHandler ) :
"""
Subclass the HTTPRedirectHandler to make it use our
2013-12-20 17:05:28 +01:00
HEADRequest also on the redirected URL
2013-06-23 20:31:45 +02:00
"""
def redirect_request ( self , req , fp , code , msg , headers , newurl ) :
if code in ( 301 , 302 , 303 , 307 ) :
newurl = newurl . replace ( ' ' , ' % 20 ' )
newheaders = dict ( ( k , v ) for k , v in req . headers . items ( )
if k . lower ( ) not in ( " content-length " , " content-type " ) )
2014-03-17 21:59:21 +01:00
try :
# This function was deprecated in python 3.3 and removed in 3.4
origin_req_host = req . get_origin_req_host ( )
except AttributeError :
origin_req_host = req . origin_req_host
2013-12-20 17:05:28 +01:00
return HEADRequest ( newurl ,
2013-06-23 20:31:45 +02:00
headers = newheaders ,
2014-03-17 21:59:21 +01:00
origin_req_host = origin_req_host ,
2013-06-23 20:31:45 +02:00
unverifiable = True )
else :
raise compat_urllib_error . HTTPError ( req . get_full_url ( ) , code , msg , headers , fp )
class HTTPMethodFallback ( compat_urllib_request . BaseHandler ) :
"""
Fallback to GET if HEAD is not allowed ( 405 HTTP error )
"""
def http_error_405 ( self , req , fp , code , msg , headers ) :
fp . read ( )
fp . close ( )
newheaders = dict ( ( k , v ) for k , v in req . headers . items ( )
if k . lower ( ) not in ( " content-length " , " content-type " ) )
return self . parent . open ( compat_urllib_request . Request ( req . get_full_url ( ) ,
headers = newheaders ,
origin_req_host = req . get_origin_req_host ( ) ,
unverifiable = True ) )
# Build our opener
opener = compat_urllib_request . OpenerDirector ( )
for handler in [ compat_urllib_request . HTTPHandler , compat_urllib_request . HTTPDefaultErrorHandler ,
HTTPMethodFallback , HEADRedirectHandler ,
compat_urllib_request . HTTPErrorProcessor , compat_urllib_request . HTTPSHandler ] :
opener . add_handler ( handler ( ) )
2013-12-20 17:05:28 +01:00
response = opener . open ( HEADRequest ( url ) )
2013-06-23 20:31:45 +02:00
if response is None :
2014-01-06 01:47:52 +01:00
raise ExtractorError ( ' Invalid URL protocol ' )
2013-12-17 12:33:55 +01:00
return response
2013-06-23 20:31:45 +02:00
2014-02-20 13:14:05 +01: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 ,
}
2013-06-23 20:31:45 +02:00
def _real_extract ( self , url ) :
2014-04-30 01:46:06 +02:00
if url . startswith ( ' // ' ) :
return {
' _type ' : ' url ' ,
2014-05-05 03:12:41 +02:00
' url ' : self . http_scheme ( ) + url ,
2014-04-30 01:46:06 +02:00
}
2013-09-06 18:39:35 +02:00
parsed_url = compat_urlparse . urlparse ( url )
if not parsed_url . scheme :
2014-01-22 14:16:43 +01:00
default_search = self . _downloader . params . get ( ' default_search ' )
if default_search is None :
2014-07-06 11:22:44 +02:00
default_search = ' error '
2014-01-22 14:16:43 +01:00
2014-03-30 15:57:31 +02:00
if default_search in ( ' auto ' , ' auto_warning ' ) :
2014-01-22 14:16:43 +01:00
if ' / ' in url :
self . _downloader . report_warning ( ' The url doesn \' t specify the protocol, trying with http ' )
return self . url_result ( ' http:// ' + url )
else :
2014-03-30 15:57:31 +02:00
if default_search == ' auto_warning ' :
2014-05-19 17:10:11 +02: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 11:22:44 +02:00
' Falling back to youtube search for %s . Set --default-search " auto " to suppress this warning. ' % url )
2014-01-22 14:16:43 +01:00
return self . url_result ( ' ytsearch: ' + url )
2014-07-06 11:22:44 +02:00
elif default_search == ' error ' :
raise ExtractorError (
( ' %r is not a valid URL. '
' Set --default-search " ytseach " (or run youtube-dl " ytsearch: %s " ) to search YouTube '
) % ( url , url ) , expected = True )
2014-01-22 14:16:43 +01:00
else :
assert ' : ' in default_search
return self . url_result ( default_search + url )
2014-03-05 14:02:14 +01:00
video_id = os . path . splitext ( url . rstrip ( ' / ' ) . split ( ' / ' ) [ - 1 ] ) [ 0 ]
2013-09-06 18:39:35 +02:00
2014-01-06 01:47:52 +01:00
self . to_screen ( ' %s : Requesting header ' % video_id )
2013-12-27 08:38:42 +01:00
2013-08-21 04:31:57 +02:00
try :
2013-12-17 12:33:55 +01:00
response = self . _send_head ( url )
# Check for redirect
new_url = response . geturl ( )
if url != new_url :
self . report_following_redirect ( new_url )
2013-12-17 12:04:33 +01:00
return self . url_result ( new_url )
2013-12-17 12:33:55 +01:00
# Check for direct link to a video
content_type = response . headers . get ( ' Content-Type ' , ' ' )
2013-12-17 16:26:32 +01:00
m = re . match ( r ' ^(?P<type>audio|video|application(?=/ogg$))/(?P<format_id>.+)$ ' , content_type )
2013-12-17 12:33:55 +01: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 01:47:52 +01:00
' vcodec ' : ' none ' if m . group ( ' type ' ) == ' audio ' else None
2013-12-17 12:33:55 +01:00
} ] ,
' upload_date ' : upload_date ,
}
2013-08-21 04:31:57 +02:00
except compat_urllib_error . HTTPError :
# This may be a stupid server that doesn't like HEAD, our UA, or so
pass
2013-06-23 20:31:45 +02: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 01:47:52 +01:00
raise ExtractorError ( ' Failed to download URL: %s ' % url )
2013-06-23 20:31:45 +02:00
self . report_extraction ( video_id )
2013-11-18 13:28:26 +01:00
2014-02-20 13:14:05 +01:00
# Is it an RSS feed?
try :
2014-03-10 17:31:32 +01:00
doc = parse_xml ( webpage )
2014-02-20 13:14:05 +01:00
if doc . tag == ' rss ' :
return self . _extract_rss ( url , video_id , doc )
2014-02-21 16:59:10 +01:00
except compat_xml_parse_error :
2014-02-20 13:14:05 +01:00
pass
2014-03-15 04:38:49 +07: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 23:44:31 +07:00
webpage = compat_urllib_parse . unquote ( webpage )
2013-11-18 13:28:26 +01: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 09:15:04 +01:00
video_title = self . _html_search_regex (
2014-01-06 01:47:52 +01:00
r ' (?s)<title>(.*?)</title> ' , webpage , ' video title ' ,
default = ' video ' )
2013-12-06 09:15:04 +01:00
# video uploader is domain name
video_uploader = self . _search_regex (
2014-01-06 01:47:52 +01:00
r ' ^(?:https?://)?([^/]*)/.* ' , url , ' video uploader ' )
2013-11-18 13:28:26 +01:00
2013-08-26 21:29:31 +02:00
# Look for BrightCove:
2014-02-03 15:19:40 +01:00
bc_urls = BrightcoveIE . _extract_brightcove_urls ( webpage )
if bc_urls :
2014-01-06 01:47:52 +01:00
self . to_screen ( ' Brightcove video detected. ' )
2014-02-03 15:19:40 +01: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 17:49:11 +02:00
2013-12-22 03:34:13 +01:00
# Look for embedded (iframe) Vimeo player
2013-10-15 12:05:13 +02:00
mobj = re . search (
2014-03-16 00:47:04 +07:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>(?:https?:)?//player \ .vimeo \ .com/video/.+?) \ 1 ' , webpage )
2013-10-15 12:05:13 +02:00
if mobj :
2014-03-16 00:47:04 +07:00
player_url = unescapeHTML ( mobj . group ( ' url ' ) )
2013-10-15 12:05:13 +02:00
surl = smuggle_url ( player_url , { ' Referer ' : url } )
return self . url_result ( surl , ' Vimeo ' )
2013-12-22 03:34:13 +01:00
# Look for embedded (swf embed) Vimeo player
mobj = re . search (
2014-01-30 04:26:46 +07:00
r ' <embed[^>]+?src= " (https?://(?:www \ .)?vimeo \ .com/moogaloop \ .swf.+?) " ' , webpage )
2013-12-22 03:34:13 +01:00
if mobj :
return self . url_result ( mobj . group ( 1 ) , ' Vimeo ' )
2013-10-18 11:44:57 +02:00
# Look for embedded YouTube player
2013-12-19 20:44:30 +01:00
matches = re . findall ( r ''' (?x)
2014-06-09 22:06:45 +02:00
( ? :
< iframe [ ^ > ] + ? src = |
< embed [ ^ > ] + ? src = |
embedSWF \( ? : \s *
)
( [ " \' ])
( ? P < url > ( ? : https ? : ) ? / / ( ? : www \. ) ? youtube \. com /
2013-12-19 20:44:30 +01:00
( ? : embed | v ) / . + ? )
\1 ''' , webpage)
2013-11-18 13:28:26 +01:00
if matches :
urlrs = [ self . url_result ( unescapeHTML ( tuppl [ 1 ] ) , ' Youtube ' )
for tuppl in matches ]
return self . playlist_result (
urlrs , playlist_id = video_id , playlist_title = video_title )
2013-10-18 11:44:57 +02:00
2013-12-01 01:21:33 +01:00
# Look for embedded Dailymotion player
matches = re . findall (
2013-12-06 09:15:04 +01:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>(?:https?:)?//(?:www \ .)?dailymotion \ .com/embed/video/.+?) \ 1 ' , webpage )
2013-12-01 01:21:33 +01:00
if matches :
2014-04-30 01:46:06 +02:00
urlrs = [ self . url_result ( unescapeHTML ( tuppl [ 1 ] ) )
2013-12-01 01:21:33 +01:00
for tuppl in matches ]
return self . playlist_result (
urlrs , playlist_id = video_id , playlist_title = video_title )
2013-12-06 09:15:04 +01: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 20:08:23 +01:00
# Look for embedded blip.tv player
2013-12-30 06:15:02 +01:00
mobj = re . search ( r ' <meta \ s[^>]*https?://api \ .blip \ .tv/ \ w+/redirect/ \ w+/( \ d+) ' , webpage )
2013-12-16 20:08:23 +01:00
if mobj :
2013-12-30 06:15:02 +01:00
return self . url_result ( ' http://blip.tv/a/a- ' + mobj . group ( 1 ) , ' BlipTV ' )
mobj = re . search ( r ' <(?:iframe|embed|object) \ s[^>]*(https?://(?: \ w+ \ .)?blip \ .tv/(?:play/|api \ .swf#)[a-zA-Z0-9]+) ' , webpage )
2013-12-16 20:08:23 +01:00
if mobj :
2013-12-30 06:15:02 +01:00
return self . url_result ( mobj . group ( 1 ) , ' BlipTV ' )
2013-12-16 20:08:23 +01:00
2014-04-21 05:47:52 +02: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 14:40:25 +01: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 16:05:14 +01:00
# Don't set the extractor because it can be a track url or an album
return self . url_result ( burl )
2013-10-27 14:40:25 +01:00
2013-12-16 21:45:21 +01: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 20:28:52 +01:00
# Look for Ooyala videos
2014-03-21 21:51:33 +01: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 20:28:52 +01:00
if mobj is not None :
2014-03-21 21:51:33 +01:00
return OoyalaIE . _build_url_result ( mobj . group ( ' ec ' ) )
2013-12-19 20:28:52 +01:00
2013-12-20 17:05:28 +01:00
# Look for Aparat videos
2014-04-21 12:37:41 +02:00
mobj = re . search ( r ' <iframe .*?src= " (http://www \ .aparat \ .com/video/[^ " ]+) " ' , webpage )
2013-12-20 17:05:28 +01:00
if mobj is not None :
return self . url_result ( mobj . group ( 1 ) , ' Aparat ' )
2014-01-07 08:07:46 +01:00
# Look for MPORA videos
2014-01-30 04:26:46 +07:00
mobj = re . search ( r ' <iframe .*?src= " (http://mpora \ .(?:com|de)/videos/[^ " ]+) " ' , webpage )
2014-01-07 08:07:46 +01:00
if mobj is not None :
return self . url_result ( mobj . group ( 1 ) , ' Mpora ' )
2014-01-08 08:11:46 +07:00
2014-04-05 17:20:05 +07:00
# Look for embedded NovaMov-based player
2014-01-08 08:07:11 +07:00
mobj = re . search (
2014-05-17 18:12:12 +07:00
r ''' (?x)<(?:pagespeed_)?iframe[^>]+?src=([ " \ ' ])
2014-04-05 17:20:05 +07: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 08:07:11 +07:00
if mobj is not None :
2014-04-05 17:20:05 +07:00
return self . url_result ( mobj . group ( ' url ' ) )
2014-04-05 15:49:45 +07:00
2014-01-21 18:10:14 +01:00
# Look for embedded Facebook player
mobj = re . search (
2014-01-27 05:47:30 +01:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https://www \ .facebook \ .com/video/embed.+?) \ 1 ' , webpage )
2014-01-21 18:10:14 +01:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Facebook ' )
2014-02-28 23:51:54 +07: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 20:18:23 +07: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 05:47:30 +01:00
# Look for embedded Huffington Post player
mobj = re . search (
2014-01-30 04:26:46 +07:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://embed \ .live \ .huffingtonpost \ .com/.+?) \ 1 ' , webpage )
2014-01-27 05:47:30 +01:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' HuffPost ' )
2014-02-24 01:15:51 +01: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 16:51:36 +01:00
# Look for funnyordie embed
matches = re . findall ( r ' <iframe[^>]+?src= " (https?://(?:www \ .)?funnyordie \ .com/embed/[^ " ]+) " ' , webpage )
if matches :
urlrs = [ self . url_result ( unescapeHTML ( eurl ) , ' FunnyOrDie ' )
for eurl in matches ]
return self . playlist_result (
urlrs , playlist_id = video_id , playlist_title = video_title )
2014-03-17 02:00:31 +07: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 10:20:44 +01: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-05 00:53:09 +10:30
# 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 22:01:47 +01: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 19:58:49 +07: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 03:12:41 +02: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 23:06:25 +02: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 21:38:04 +02: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 ' )
2013-06-23 20:31:45 +02:00
# Start with something easy: JW Player in SWFObject
2014-04-30 02:23:51 +02:00
found = re . findall ( r ' flashvars: [ \' " ](?:.*&)?file=(http[^ \' " &]*) ' , webpage )
if not found :
2014-01-05 05:34:06 +01:00
# Look for gorilla-vid style embedding
2014-04-30 02:23:51 +02:00
found = re . findall ( r ''' (?sx)
2014-04-21 16:16:53 +02:00
( ? :
jw_plugins |
JWPlayerOptions |
jwplayer \s * \( \s * [ " ' ][^ ' " ] + [ " ' ] \ s* \ ) \ s* \ .setup
)
. * ? file \s * : \s * [ " \' ](.*?)[ " \' ] ' ' ' , webpage )
2014-04-30 02:23:51 +02:00
if not found :
2013-06-23 20:31:45 +02:00
# Broaden the search a little bit
2014-04-30 02:23:51 +02: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 )
if not found :
2013-06-23 20:31:45 +02:00
# Try to find twitter cards info
2014-04-30 02:23:51 +02:00
found = re . findall ( r ' <meta (?:property|name)= " twitter:player:stream " (?:content|value)= " (.+?) " ' , webpage )
if not found :
2013-06-23 20:31:45 +02: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 02:23:51 +02:00
m_video_type = re . findall ( r ' <meta.*?property= " og:video:type " .*?content= " video/(.*?) " ' , webpage )
2013-06-23 20:31:45 +02: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-04-30 02:23:51 +02:00
found = re . findall ( r ' <meta.*?property= " og:video " .*?content= " (.*?) " ' , webpage )
if not found :
2013-08-21 04:32:22 +02:00
# HTML5 video
2014-04-30 02:23:51 +02:00
found = re . findall ( r ' (?s)<video[^<]*(?:>.*?<source.*?)? src= " ([^ " ]+) " ' , webpage )
if not found :
2014-05-16 20:32:53 +07:00
found = re . search (
2014-02-27 07:21:59 +01:00
r ' (?i)<meta \ s+(?=(?:[a-z-]+= " [^ " ]+ " \ s+)*http-equiv= " refresh " ) '
r ' (?:[a-z-]+= " [^ " ]+ " \ s+)*?content= " [0-9] { ,2};url= \' ([^ \' ]+) \' " ' ,
webpage )
2014-04-30 02:23:51 +02:00
if found :
new_url = found . group ( 1 )
2014-02-27 07:21:59 +01:00
self . report_following_redirect ( new_url )
return {
' _type ' : ' url ' ,
' url ' : new_url ,
}
2014-04-30 02:23:51 +02:00
if not found :
2014-01-06 01:47:52 +01:00
raise ExtractorError ( ' Unsupported URL: %s ' % url )
2013-06-23 20:31:45 +02:00
2014-04-30 02:23:51 +02: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 20:31:45 +02:00
2014-04-30 02:23:51 +02: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 20:31:45 +02:00
2014-04-30 02:23:51 +02:00
# here's a fun little line of code for you:
video_id = os . path . splitext ( video_id ) [ 0 ]
2014-01-06 01:42:58 +01:00
2014-04-30 02:23:51 +02:00
entries . append ( {
' id ' : video_id ,
' url ' : video_url ,
' uploader ' : video_uploader ,
' title ' : video_title ,
} )
if len ( entries ) == 1 :
2014-05-01 16:28:37 +07:00
return entries [ 0 ]
2014-04-30 02:23:51 +02: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 20:31:45 +02:00