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
2015-09-20 07:16:12 +02:00
import sys
2013-06-23 20:31:45 +02:00
from . common import InfoExtractor
2014-01-06 01:42:58 +01:00
from . youtube import YoutubeIE
2014-11-02 11:23:40 +01:00
from . . compat import (
2015-10-26 16:41:24 +01:00
compat_etree_fromstring ,
2015-05-30 21:23:58 +02:00
compat_urllib_parse_unquote ,
2013-08-28 12:47:27 +02:00
compat_urlparse ,
2014-02-21 16:59:10 +01:00
compat_xml_parse_error ,
2014-11-02 11:23:40 +01:00
)
from . . utils import (
2014-08-28 01:00:59 +02:00
determine_ext ,
2013-06-23 20:31:45 +02:00
ExtractorError ,
2014-08-24 02:02:17 +02:00
float_or_none ,
2013-12-20 17:05:28 +01:00
HEADRequest ,
2015-01-23 01:21:30 +01:00
is_html ,
2014-08-22 18:19:56 +02:00
orderedSet ,
2015-11-21 17:18:17 +01:00
sanitized_Request ,
2013-10-15 12:05:13 +02:00
smuggle_url ,
unescapeHTML ,
2013-12-17 12:33:55 +01:00
unified_strdate ,
2014-08-24 04:47:18 +02:00
unsmuggle_url ,
2014-12-30 19:35:35 +01:00
UnsupportedError ,
2013-12-17 12:33:55 +01:00
url_basename ,
2015-03-02 15:21:11 +01:00
xpath_text ,
2013-06-23 20:31:45 +02:00
)
2015-09-11 05:46:21 +02:00
from . brightcove import (
2015-11-14 00:54:16 +01:00
BrightcoveLegacyIE ,
2015-11-14 00:55:59 +01:00
BrightcoveNewIE ,
2015-09-11 05:46:21 +02:00
)
2015-03-30 21:36:09 +02:00
from . nbc import NBCSportsVPlayerIE
2013-12-19 20:28:52 +01:00
from . ooyala import OoyalaIE
2014-03-16 20:00:31 +01:00
from . rutv import RUTVIE
2015-06-12 12:24:13 +02:00
from . tvc import TVCIE
2015-05-15 19:09:34 +02:00
from . sportbox import SportBoxEmbedIE
2014-03-28 13:58:49 +01:00
from . smotri import SmotriIE
2015-07-09 20:27:44 +02:00
from . myvi import MyviIE
2014-10-13 14:59:35 +02:00
from . condenast import CondeNastIE
2015-04-08 11:26:51 +02:00
from . udn import UDNEmbedIE
2015-04-20 21:18:38 +02:00
from . senateisvp import SenateISVPIE
2015-05-08 20:23:35 +02:00
from . svt import SVTIE
2015-06-12 23:36:16 +02:00
from . pornhub import PornHubIE
2015-06-21 19:11:25 +02:00
from . xhamster import XHamsterEmbedIE
2016-02-27 12:15:49 +01:00
from . tnaflix import TNAFlixNetworkEmbedIE
2015-06-21 12:23:58 +02:00
from . vimeo import VimeoIE
2016-07-06 17:37:54 +02:00
from . dailymotion import (
DailymotionIE ,
DailymotionCloudIE ,
)
2015-06-24 19:19:50 +02:00
from . onionstudios import OnionStudiosIE
2016-04-29 12:14:42 +02:00
from . viewlift import ViewLiftEmbedIE
2015-08-29 15:07:31 +02:00
from . screenwavemedia import ScreenwaveMediaIE
2015-09-26 15:47:20 +02:00
from . mtv import MTVServicesEmbeddedIE
2015-12-07 17:03:21 +01:00
from . pladform import PladformIE
2015-12-29 18:58:23 +01:00
from . videomore import VideomoreIE
2015-12-21 03:05:34 +01:00
from . googledrive import GoogleDriveIE
2015-12-21 04:24:58 +01:00
from . jwplatform import JWPlatformIE
2016-01-19 16:51:46 +01:00
from . digiteka import DigitekaIE
2016-03-24 09:32:27 +01:00
from . instagram import InstagramIE
2016-03-31 20:42:55 +02:00
from . liveleak import LiveLeakIE
2016-05-14 19:40:34 +02:00
from . threeqsdn import ThreeQSDNIE
2016-05-22 02:52:39 +02:00
from . theplatform import ThePlatformIE
2016-06-08 23:02:27 +02:00
from . vessel import VesselIE
2016-06-27 17:45:26 +02:00
from . kaltura import KalturaIE
2016-06-29 18:01:34 +02:00
from . eagleplatform import EaglePlatformIE
2016-07-02 15:33:23 +02:00
from . facebook import FacebookIE
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 = [
2015-05-30 22:22:29 +02:00
# Direct link to a video
{
' url ' : ' http://media.w3.org/2010/05/sintel/trailer.mp4 ' ,
' md5 ' : ' 67d406c2bcb6af27fa886f31aa934bbe ' ,
' info_dict ' : {
' id ' : ' trailer ' ,
' ext ' : ' mp4 ' ,
' title ' : ' trailer ' ,
' upload_date ' : ' 20100513 ' ,
}
} ,
2015-05-30 22:36:20 +02:00
# Direct link to media delivered compressed (until Accept-Encoding is *)
2015-05-30 22:22:29 +02:00
{
' url ' : ' http://calimero.tk/muzik/FictionJunction-Parallel_Hearts.flac ' ,
' md5 ' : ' 128c42e68b13950268b648275386fc74 ' ,
' info_dict ' : {
' id ' : ' FictionJunction-Parallel_Hearts ' ,
' ext ' : ' flac ' ,
' title ' : ' FictionJunction-Parallel_Hearts ' ,
' upload_date ' : ' 20140522 ' ,
} ,
' expected_warnings ' : [
' URL could be a direct video link, returning it as such. '
]
} ,
# Direct download with broken HEAD
{
' url ' : ' http://ai-radio.org:8000/radio.opus ' ,
' info_dict ' : {
' id ' : ' radio ' ,
' ext ' : ' opus ' ,
' title ' : ' radio ' ,
} ,
' params ' : {
' skip_download ' : True , # infinite live stream
} ,
' expected_warnings ' : [
2016-04-16 13:44:12 +02:00
r ' 501.*Not Implemented ' ,
r ' 400.*Bad Request ' ,
2015-05-30 22:22:29 +02:00
] ,
} ,
# Direct link with incorrect MIME type
{
' url ' : ' http://ftp.nluug.nl/video/nluug/2014-11-20_nj14/zaal-2/5_Lennart_Poettering_-_Systemd.webm ' ,
' md5 ' : ' 4ccbebe5f36706d85221f204d7eb5913 ' ,
' info_dict ' : {
' url ' : ' http://ftp.nluug.nl/video/nluug/2014-11-20_nj14/zaal-2/5_Lennart_Poettering_-_Systemd.webm ' ,
' id ' : ' 5_Lennart_Poettering_-_Systemd ' ,
' ext ' : ' webm ' ,
' title ' : ' 5_Lennart_Poettering_-_Systemd ' ,
' upload_date ' : ' 20141120 ' ,
} ,
' expected_warnings ' : [
' URL could be a direct video link, returning it as such. '
]
} ,
# 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:.*groundbreaking video review series.* '
} ,
' playlist_mincount ' : 11 ,
} ,
# RSS feed with enclosure
{
' url ' : ' http://podcastfeeds.nbcnews.com/audio/podcast/MSNBC-MADDOW-NETCAST-M4V.xml ' ,
' info_dict ' : {
' id ' : ' pdv_maddow_netcast_m4v-02-27-2015-201624 ' ,
' ext ' : ' m4v ' ,
' upload_date ' : ' 20150228 ' ,
' title ' : ' pdv_maddow_netcast_m4v-02-27-2015-201624 ' ,
}
} ,
2015-08-01 21:16:21 +02:00
# SMIL from http://videolectures.net/promogram_igor_mekjavic_eng
{
' url ' : ' http://videolectures.net/promogram_igor_mekjavic_eng/video/1/smil.xml ' ,
' info_dict ' : {
' id ' : ' smil ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Automatics, robotics and biocybernetics ' ,
' description ' : ' md5:815fc1deb6b3a2bff99de2d5325be482 ' ,
2015-10-31 18:05:30 +01:00
' upload_date ' : ' 20130627 ' ,
2015-08-01 21:16:21 +02:00
' formats ' : ' mincount:16 ' ,
' subtitles ' : ' mincount:1 ' ,
} ,
' params ' : {
' force_generic_extractor ' : True ,
' skip_download ' : True ,
} ,
} ,
# SMIL from http://www1.wdr.de/mediathek/video/livestream/index.html
{
' url ' : ' http://metafilegenerator.de/WDR/WDR_FS/hds/hds.smil ' ,
' info_dict ' : {
' id ' : ' hds ' ,
' ext ' : ' flv ' ,
' title ' : ' hds ' ,
' formats ' : ' mincount:1 ' ,
} ,
' params ' : {
' skip_download ' : True ,
} ,
} ,
# SMIL from https://www.restudy.dk/video/play/id/1637
{
' url ' : ' https://www.restudy.dk/awsmedia/SmilDirectory/video_1637.xml ' ,
' info_dict ' : {
' id ' : ' video_1637 ' ,
' ext ' : ' flv ' ,
' title ' : ' video_1637 ' ,
' formats ' : ' mincount:3 ' ,
} ,
' params ' : {
' skip_download ' : True ,
} ,
} ,
# SMIL from http://adventure.howstuffworks.com/5266-cool-jobs-iditarod-musher-video.htm
{
' url ' : ' http://services.media.howstuffworks.com/videos/450221/smil-service.smil ' ,
' info_dict ' : {
' id ' : ' smil-service ' ,
' ext ' : ' flv ' ,
' title ' : ' smil-service ' ,
' formats ' : ' mincount:1 ' ,
} ,
' params ' : {
' skip_download ' : True ,
} ,
} ,
# SMIL from http://new.livestream.com/CoheedandCambria/WebsterHall/videos/4719370
{
' url ' : ' http://api.new.livestream.com/accounts/1570303/events/1585861/videos/4719370.smil ' ,
' info_dict ' : {
' id ' : ' 4719370 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' 571de1fd-47bc-48db-abf9-238872a58d1f ' ,
' formats ' : ' mincount:3 ' ,
} ,
' params ' : {
' skip_download ' : True ,
} ,
} ,
2015-08-09 15:47:08 +02:00
# XSPF playlist from http://www.telegraaf.nl/tv/nieuws/binnenland/24353229/__Tikibad_ontruimd_wegens_brand__.html
{
' url ' : ' http://www.telegraaf.nl/xml/playlist/2015/8/7/mZlp2ctYIUEB.xspf ' ,
' info_dict ' : {
' id ' : ' mZlp2ctYIUEB ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Tikibad ontruimd wegens brand ' ,
' description ' : ' md5:05ca046ff47b931f9b04855015e163a4 ' ,
' thumbnail ' : ' re:^https?://.* \ .jpg$ ' ,
' duration ' : 33 ,
} ,
' params ' : {
' skip_download ' : True ,
} ,
} ,
2016-02-12 19:36:47 +01:00
# MPD from http://dash-mse-test.appspot.com/media.html
{
' url ' : ' http://yt-dash-mse-test.commondatastorage.googleapis.com/media/car-20120827-manifest.mpd ' ,
' md5 ' : ' 4b57baab2e30d6eb3a6a09f0ba57ef53 ' ,
' info_dict ' : {
' id ' : ' car-20120827-manifest ' ,
' ext ' : ' mp4 ' ,
' title ' : ' car-20120827-manifest ' ,
' formats ' : ' mincount:9 ' ,
2016-04-24 14:44:52 +02:00
' upload_date ' : ' 20130904 ' ,
2016-02-12 19:36:47 +01:00
} ,
' params ' : {
' format ' : ' bestvideo ' ,
} ,
} ,
2016-03-18 16:54:33 +01:00
# m3u8 served with Content-Type: audio/x-mpegURL; charset=utf-8
{
' url ' : ' http://once.unicornmedia.com/now/master/playlist/bb0b18ba-64f5-4b1b-a29f-0ac252f06b68/77a785f3-5188-4806-b788-0893a61634ed/93677179-2d99-4ef4-9e17-fe70d49abfbf/content.m3u8 ' ,
' info_dict ' : {
' id ' : ' content ' ,
' ext ' : ' mp4 ' ,
' title ' : ' content ' ,
' formats ' : ' mincount:8 ' ,
} ,
' params ' : {
# m3u8 downloads
' skip_download ' : True ,
}
} ,
2016-03-18 17:49:11 +01:00
# m3u8 served with Content-Type: text/plain
{
' url ' : ' http://www.nacentapps.com/m3u8/index.m3u8 ' ,
' info_dict ' : {
' id ' : ' index ' ,
' ext ' : ' mp4 ' ,
' title ' : ' index ' ,
' upload_date ' : ' 20140720 ' ,
' formats ' : ' mincount:11 ' ,
} ,
' params ' : {
# m3u8 downloads
' skip_download ' : True ,
}
} ,
2015-05-30 22:22:29 +02: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 ' : ' re:^Chris Ziegler takes a look at the \ .* ' ,
' uploader ' : ' The Verge ' ,
' title ' : ' First Firefox OS phones side-by-side ' ,
} ,
' params ' : {
' skip_download ' : False ,
}
} ,
2015-09-20 07:16:12 +02:00
{
# redirect in Refresh HTTP header
' url ' : ' https://www.facebook.com/l.php?u=https % 3A %2F %2F www.youtube.com %2F watch %3F v % 3DpO8h3EaFRdo&h=TAQHsoToz&enc=AZN16h-b6o4Zq9pZkCCdOLNKMN96BbGMNtcFwHSaazus4JHT_MFYkAA-WARTX2kvsCIdlAIyHZjl6d33ILIJU7Jzwk_K3mcenAXoAzBNoZDI_Q7EXGDJnIhrGkLXo_LJ_pAa2Jzbx17UHMd3jAs--6j2zaeto5w9RTn8T_1kKg3fdC5WPX9Dbb18vzH7YFX0eSJmoa6SP114rvlkw6pkS1-T&s=1 ' ,
' info_dict ' : {
' id ' : ' pO8h3EaFRdo ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Tripeo Boiler Room x Dekmantel Festival DJ Set ' ,
' description ' : ' md5:6294cc1af09c4049e0652b51a2df10d5 ' ,
' upload_date ' : ' 20150917 ' ,
' uploader_id ' : ' brtvofficial ' ,
' uploader ' : ' Boiler Room ' ,
} ,
' params ' : {
' skip_download ' : False ,
} ,
} ,
2013-07-10 17:49:11 +02:00
{
2014-01-06 01:47:52 +01:00
' url ' : ' http://www.hodiho.fr/2013/02/regis-plante-sa-jeep.html ' ,
2014-04-14 13:51:46 +02:00
' md5 ' : ' 85b90ccc9d73b4acd9138d3af4c27f89 ' ,
2014-01-06 01:47:52 +01:00
' info_dict ' : {
2014-04-14 13:51:46 +02: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 13:56:29 +02: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
{
2015-11-14 01:05:46 +01:00
' add_ie ' : [ ' BrightcoveLegacy ' ] ,
2014-01-06 01:47:52 +01:00
' 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 13:56:29 +02: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. ' ,
} ,
2015-11-14 01:05:46 +01:00
' add_ie ' : [ ' BrightcoveLegacy ' ] ,
2014-01-28 03:35:32 +01:00
} ,
2014-04-01 15:17:35 +02:00
{
' url ' : ' http://www.championat.com/video/football/v/87/87499.html ' ,
' md5 ' : ' fb973ecf6e4a78a67453647444222983 ' ,
' info_dict ' : {
' id ' : ' 3414141473001 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Видео. Удаление Дзагоева (ЦСКА) ' ,
' description ' : ' Онлайн-трансляция матча ЦСКА - " Волга " ' ,
' uploader ' : ' Championat ' ,
} ,
} ,
2014-11-03 22:13:46 +01:00
{
2014-11-05 23:14:33 +01:00
# https://github.com/rg3/youtube-dl/issues/3541
2015-11-14 01:05:46 +01:00
' add_ie ' : [ ' BrightcoveLegacy ' ] ,
2014-11-03 22:13:46 +01:00
' url ' : ' http://www.kijk.nl/sbs6/leermijvrouwenkennen/videos/jqMiXKAYan2S/aflevering-1 ' ,
' info_dict ' : {
' id ' : ' 3866516442001 ' ,
2014-11-05 23:14:33 +01:00
' ext ' : ' mp4 ' ,
2014-11-03 22:13:46 +01:00
' title ' : ' Leer mij vrouwen kennen: Aflevering 1 ' ,
' description ' : ' Leer mij vrouwen kennen: Aflevering 1 ' ,
' uploader ' : ' SBS Broadcasting ' ,
} ,
2014-11-05 23:14:33 +01:00
' skip ' : ' Restricted to Netherlands ' ,
2014-11-03 22:13:46 +01:00
' params ' : {
2014-11-05 23:14:33 +01:00
' skip_download ' : True , # m3u8 download
2014-11-03 22:13:46 +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 ' ,
2015-01-05 13:07:24 +01:00
' md5 ' : ' 166dd577b433b4d4ebfee10b0824d8ff ' ,
2014-01-06 01:47:52 +01:00
' info_dict ' : {
' id ' : ' BwY2RxaTrTkslxOfcan0UCf0YqyvWysJ ' ,
' ext ' : ' mp4 ' ,
2014-01-21 01:40:34 +01:00
' title ' : ' 2cc213299525360.mov ' , # that's what we get
2015-12-04 16:18:02 +01:00
' duration ' : 238.231 ,
2013-12-19 20:28:52 +01:00
} ,
2015-01-05 13:07:24 +01:00
' add_ie ' : [ ' Ooyala ' ] ,
2013-12-19 20:28:52 +01:00
} ,
2015-08-07 22:00:49 +02:00
{
# ooyala video embedded with http://player.ooyala.com/iframe.js
' url ' : ' http://www.macrumors.com/2015/07/24/steve-jobs-the-man-in-the-machine-first-trailer/ ' ,
' info_dict ' : {
' id ' : ' p0MGJndjoG5SOKqO_hZJuZFPB-Tr5VgB ' ,
' ext ' : ' mp4 ' ,
' title ' : ' " Steve Jobs: Man in the Machine " trailer ' ,
' description ' : ' The first trailer for the Alex Gibney documentary " Steve Jobs: Man in the Machine. " ' ,
2015-12-04 16:18:02 +01:00
' duration ' : 135.427 ,
2015-08-07 22:00:49 +02:00
} ,
' params ' : {
' skip_download ' : True ,
} ,
} ,
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 ' ,
' 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 20:00:31 +01:00
} ,
2014-03-11 16:51:36 +01:00
} ,
2014-03-16 20:00:31 +01: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
} ,
2015-06-12 12:28:45 +02:00
# TVC embed
{
' url ' : ' http://sch1298sz.mskobr.ru/dou_edu/karamel_ki/filial_galleries/video/iframe_src_http_tvc_ru_video_iframe_id_55304_isplay_false_acc_video_id_channel_brand_id_11_show_episodes_episode_id_32307_frameb/ ' ,
' info_dict ' : {
' id ' : ' 55304 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Дошкольное воспитание ' ,
} ,
} ,
2015-05-15 19:18:21 +02:00
# SportBox embed
{
' url ' : ' http://www.vestifinance.ru/articles/25753 ' ,
' info_dict ' : {
' id ' : ' 25753 ' ,
' title ' : ' Вести Экономика ― Прямые трансляции с Форума-выставки " Госзаказ-2013 " ' ,
} ,
' playlist ' : [ {
' info_dict ' : {
' id ' : ' 370908 ' ,
' title ' : ' Госзаказ. День 3 ' ,
' ext ' : ' mp4 ' ,
}
} , {
' info_dict ' : {
' id ' : ' 370905 ' ,
' title ' : ' Госзаказ. День 2 ' ,
' ext ' : ' mp4 ' ,
}
} , {
' info_dict ' : {
' id ' : ' 370902 ' ,
' title ' : ' Госзаказ. День 1 ' ,
' ext ' : ' mp4 ' ,
}
} ] ,
' params ' : {
# m3u8 download
' skip_download ' : True ,
} ,
} ,
2015-07-09 21:15:55 +02:00
# Myvi.ru embed
{
' url ' : ' http://www.kinomyvi.tv/news/detail/Pervij-dublirovannij-trejler--Uzhastikov-_nOw1 ' ,
' info_dict ' : {
' id ' : ' f4dafcad-ff21-423d-89b5-146cfd89fa1e ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Ужастики, русский трейлер (2015) ' ,
' thumbnail ' : ' re:^https?://.* \ .jpg$ ' ,
' duration ' : 153 ,
}
} ,
2015-06-21 19:18:28 +02:00
# XHamster embed
{
' url ' : ' http://www.numisc.com/forum/showthread.php?11696-FM15-which-pumiscer-was-this- % 28-vid- % 29- % 28-alfa-as-fuck-srx- % 29&s=711f5db534502e22260dec8c5e2d66d8 ' ,
' info_dict ' : {
' id ' : ' showthread ' ,
' title ' : ' [NSFL] [FM15] which pumiscer was this ( vid ) ( alfa as fuck srx ) ' ,
} ,
' playlist_mincount ' : 7 ,
} ,
2014-03-20 16:33:23 +01:00
# Embedded TED video
{
' url ' : ' http://en.support.wordpress.com/videos/ted-talks/ ' ,
2014-09-29 05:12:57 +02:00
' md5 ' : ' 65fdff94098e4a607385a60c5177c638 ' ,
2014-03-20 16:33:23 +01:00
' info_dict ' : {
2014-09-29 05:12:57 +02:00
' id ' : ' 1969 ' ,
2014-03-20 16:33:23 +01:00
' ext ' : ' mp4 ' ,
2014-09-29 05:12:57 +02:00
' title ' : ' Hidden miracles of the natural world ' ,
' uploader ' : ' Louie Schwartzberg ' ,
' description ' : ' md5:8145d19d320ff3e52f28401f4c4283b9 ' ,
2014-03-20 16:33:23 +01:00
}
2014-03-11 16:51:36 +01:00
} ,
2016-01-10 16:17:47 +01:00
# Embedded Ustream video
2014-04-04 16:23:09 +02:00
{
' url ' : ' http://www.american.edu/spa/pti/nsa-privacy-janus-2014.cfm ' ,
' md5 ' : ' 27b99cdb639c9b12a79bca876a073417 ' ,
' info_dict ' : {
2014-04-04 18:56:29 +02:00
' id ' : ' 45734260 ' ,
' ext ' : ' flv ' ,
' uploader ' : ' AU SPA: The NSA and Privacy ' ,
2014-04-04 16:23:09 +02:00
' title ' : ' NSA and Privacy Forum Debate featuring General Hayden and Barton Gellman '
}
} ,
2014-03-14 22:39:53 +01: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 '
}
} ,
2015-07-18 18:56:00 +02:00
# francetv embed
{
' url ' : ' http://www.tsprod.com/replay-du-concert-alcaline-de-calogero ' ,
' info_dict ' : {
' id ' : ' EV_30231 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Alcaline, le concert avec Calogero ' ,
' description ' : ' md5:61f08036dcc8f47e9cfc33aed08ffaff ' ,
' upload_date ' : ' 20150226 ' ,
' timestamp ' : 1424989860 ,
' duration ' : 5400 ,
} ,
' params ' : {
# m3u8 downloads
' skip_download ' : True ,
} ,
' expected_warnings ' : [
' Forbidden '
]
} ,
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 ' ,
2016-04-24 14:44:52 +02:00
' description ' : ' md5:faf028e48a461b8b7fad38f1e104b119 ' ,
2014-04-30 01:46:06 +02:00
' uploader ' : ' Spi0n ' ,
2016-04-24 14:44:52 +02:00
' uploader_id ' : ' xgditw ' ,
' upload_date ' : ' 20140425 ' ,
' timestamp ' : 1398441542 ,
2014-04-30 01:46:06 +02:00
} ,
' 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
{
2016-06-10 10:39:31 +02:00
' url ' : ' http://www.vulture.com/2016/06/new-key-peele-sketches-released.html ' ,
' md5 ' : ' ca1aef97695ef2c1d6973256a57e5252 ' ,
2014-06-22 21:38:04 +02:00
' info_dict ' : {
2016-06-10 10:39:31 +02:00
' id ' : ' 769f7ec0-0692-4d62-9b45-0d88074bffc1 ' ,
2014-06-22 21:38:04 +02:00
' ext ' : ' mp4 ' ,
2016-06-10 10:39:31 +02:00
' title ' : ' Key and Peele|October 10, 2012|2|203|Liam Neesons - Uncensored ' ,
' description ' : ' Two valets share their love for movie star Liam Neesons. ' ,
2014-06-22 21:38:04 +02:00
} ,
} ,
2014-05-21 11:55:37 +02:00
# YouTube embed via <data-embed-url="">
{
' url ' : ' https://play.google.com/store/apps/details?id=com.gameloft.android.ANMP.GloftA8HM ' ,
' info_dict ' : {
2014-09-29 05:12:57 +02:00
' id ' : ' 4vAffPZIT44 ' ,
2014-05-21 11:55:37 +02:00
' ext ' : ' mp4 ' ,
2014-09-29 05:12:57 +02:00
' title ' : ' Asphalt 8: Airborne - Update - Welcome to Dubai! ' ,
2014-08-22 18:19:56 +02:00
' uploader ' : ' Gameloft ' ,
' uploader_id ' : ' gameloft ' ,
2014-09-29 05:12:57 +02:00
' upload_date ' : ' 20140828 ' ,
' description ' : ' md5:c80da9ed3d83ae6d1876c834de03e1c4 ' ,
2014-08-22 18:19:56 +02:00
} ,
' params ' : {
' skip_download ' : True ,
2014-05-21 11:55:37 +02:00
}
2014-08-24 02:02:17 +02: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 05:31:32 +02: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 ' ,
2014-10-27 00:45:15 +01:00
' title ' : ' Busty Blonde Siri Tit Fuck While Wank at HandjobHub.com ' ,
2014-08-24 05:31:32 +02:00
}
2014-08-25 18:03:01 +02:00
} ,
2014-08-28 00:58:24 +02:00
# Multiple brightcove videos
# https://github.com/rg3/youtube-dl/issues/2283
{
' url ' : ' http://www.newyorker.com/online/blogs/newsdesk/2014/01/always-never-nuclear-command-and-control.html ' ,
' info_dict ' : {
' id ' : ' always-never ' ,
' title ' : ' Always / Never - The New Yorker ' ,
} ,
' playlist_count ' : 3 ,
' params ' : {
' extract_flat ' : False ,
' skip_download ' : True ,
}
2014-09-02 15:19:28 +02:00
} ,
# MLB embed
{
' url ' : ' http://umpire-empire.com/index.php/topic/58125-laz-decides-no-thats-low/ ' ,
' md5 ' : ' 96f09a37e44da40dd083e12d9a683327 ' ,
' info_dict ' : {
' id ' : ' 33322633 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Ump changes call to ball ' ,
' description ' : ' md5:71c11215384298a172a6dcb4c2e20685 ' ,
' duration ' : 48 ,
' timestamp ' : 1401537900 ,
' upload_date ' : ' 20140531 ' ,
' thumbnail ' : ' re:^https?://.* \ .jpg$ ' ,
} ,
} ,
2014-09-20 02:02:11 +02:00
# Wistia embed
{
2016-05-20 17:55:35 +02:00
' url ' : ' http://study.com/academy/lesson/north-american-exploration-failed-colonies-of-spain-france-england.html#lesson ' ,
' md5 ' : ' 1953f3a698ab51cfc948ed3992a0b7ff ' ,
2014-09-20 02:02:11 +02:00
' info_dict ' : {
2016-05-20 17:55:35 +02:00
' id ' : ' 6e2wtrbdaf ' ,
2014-09-20 02:02:11 +02:00
' ext ' : ' mov ' ,
2016-05-20 17:55:35 +02:00
' title ' : ' paywall_north-american-exploration-failed-colonies-of-spain-france-england ' ,
' description ' : ' a Paywall Videos video from Remilon ' ,
' duration ' : 644.072 ,
' uploader ' : ' study.com ' ,
' timestamp ' : 1459678540 ,
' upload_date ' : ' 20160403 ' ,
' filesize ' : 24687186 ,
2014-09-20 02:02:11 +02:00
} ,
} ,
2014-10-23 16:57:13 +02:00
{
' url ' : ' http://thoughtworks.wistia.com/medias/uxjb0lwrcz ' ,
' md5 ' : ' baf49c2baa8a7de5f3fc145a8506dcd4 ' ,
' info_dict ' : {
' id ' : ' uxjb0lwrcz ' ,
' ext ' : ' mp4 ' ,
2016-05-20 17:55:35 +02:00
' title ' : ' Conversation about Hexagonal Rails Part 1 ' ,
2016-04-24 14:44:52 +02:00
' description ' : ' a Martin Fowler video from ThoughtWorks ' ,
2014-10-23 16:57:13 +02:00
' duration ' : 1715.0 ,
2014-10-23 17:53:56 +02:00
' uploader ' : ' thoughtworks.wistia.com ' ,
2016-04-24 14:44:52 +02:00
' timestamp ' : 1401832161 ,
2016-05-20 17:55:35 +02:00
' upload_date ' : ' 20140603 ' ,
2014-10-26 20:49:51 +01:00
} ,
2014-10-23 16:57:13 +02:00
} ,
2016-05-20 17:43:36 +02:00
# Wistia standard embed (async)
{
' url ' : ' https://www.getdrip.com/university/brennan-dunn-drip-workshop/ ' ,
' info_dict ' : {
' id ' : ' 807fafadvk ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Drip Brennan Dunn Workshop ' ,
' description ' : ' a JV Webinars video from getdrip-1 ' ,
' duration ' : 4986.95 ,
' timestamp ' : 1463607249 ,
2016-05-20 17:55:35 +02:00
' upload_date ' : ' 20160518 ' ,
2016-05-20 17:43:36 +02:00
} ,
' params ' : {
' skip_download ' : True ,
}
} ,
2014-10-29 20:27:58 +01:00
# Soundcloud embed
{
' url ' : ' http://nakedsecurity.sophos.com/2014/10/29/sscc-171-are-you-sure-that-1234-is-a-bad-password-podcast/ ' ,
' info_dict ' : {
' id ' : ' 174391317 ' ,
' ext ' : ' mp3 ' ,
' description ' : ' md5:ff867d6b555488ad3c52572bb33d432c ' ,
' uploader ' : ' Sophos Security ' ,
' title ' : ' Chet Chat 171 - Oct 29, 2014 ' ,
' upload_date ' : ' 20141029 ' ,
}
2014-11-13 16:12:51 +01:00
} ,
# Livestream embed
{
' url ' : ' http://www.esa.int/Our_Activities/Space_Science/Rosetta/Philae_comet_touch-down_webcast ' ,
' info_dict ' : {
' id ' : ' 67864563 ' ,
' ext ' : ' flv ' ,
' upload_date ' : ' 20141112 ' ,
' title ' : ' Rosetta #CometLanding webcast HL 10 ' ,
}
} ,
2016-05-22 19:39:09 +02:00
# Another Livestream embed, without 'new.' in URL
{
' url ' : ' https://www.freespeech.org/ ' ,
' info_dict ' : {
' id ' : ' 123537347 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' re:^FSTV [0-9] {4} -[0-9] {2} -[0-9] {2} [0-9] {2} :[0-9] {2} $ ' ,
} ,
' params ' : {
# Live stream
' skip_download ' : True ,
} ,
} ,
2014-11-25 14:34:19 +01:00
# LazyYT
{
' url ' : ' http://discourse.ubuntu.com/t/unity-8-desktop-mode-windows-on-mir/1986 ' ,
' info_dict ' : {
2015-02-18 00:49:10 +01:00
' id ' : ' 1986 ' ,
2014-11-25 14:34:19 +01:00
' title ' : ' Unity 8 desktop-mode windows on Mir! - Ubuntu Discourse ' ,
} ,
' playlist_mincount ' : 2 ,
2014-11-26 10:44:39 +01:00
} ,
2014-12-12 02:57:36 +01:00
# Cinchcast embed
{
' url ' : ' http://undergroundwellness.com/podcasts/306-5-steps-to-permanent-gut-healing/ ' ,
' info_dict ' : {
' id ' : ' 7141703 ' ,
' ext ' : ' mp3 ' ,
' upload_date ' : ' 20141126 ' ,
' title ' : ' Jack Tips: 5 Steps to Permanent Gut Healing ' ,
}
} ,
2015-01-23 12:00:25 +01:00
# Cinerama player
{
' url ' : ' http://www.abc.net.au/7.30/content/2015/s4164797.htm ' ,
' info_dict ' : {
' id ' : ' 730m_DandD_1901_512k ' ,
' ext ' : ' mp4 ' ,
' uploader ' : ' www.abc.net.au ' ,
' title ' : ' Game of Thrones with dice - Dungeons and Dragons fantasy role-playing game gets new life - 19/01/2015 ' ,
}
2015-01-28 06:08:19 +01:00
} ,
# embedded viddler video
{
' url ' : ' http://deadspin.com/i-cant-stop-watching-john-wall-chop-the-nuggets-with-th-1681801597 ' ,
' info_dict ' : {
' id ' : ' 4d03aad9 ' ,
' ext ' : ' mp4 ' ,
' uploader ' : ' deadspin ' ,
' title ' : ' WALL-TO-GORTAT ' ,
' timestamp ' : 1422285291 ,
' upload_date ' : ' 20150126 ' ,
} ,
' add_ie ' : [ ' Viddler ' ] ,
2015-02-09 10:42:25 +01:00
} ,
2015-03-22 03:20:27 +01:00
# Libsyn embed
{
' url ' : ' http://thedailyshow.cc.com/podcast/episodetwelve ' ,
' info_dict ' : {
' id ' : ' 3377616 ' ,
' ext ' : ' mp3 ' ,
' title ' : " The Daily Show Podcast without Jon Stewart - Episode 12: Bassem Youssef: Egypt ' s Jon Stewart " ,
' description ' : ' md5:601cb790edd05908957dae8aaa866465 ' ,
' upload_date ' : ' 20150220 ' ,
} ,
} ,
2015-02-09 10:42:25 +01:00
# jwplayer YouTube
{
' url ' : ' http://media.nationalarchives.gov.uk/index.php/webinar-using-discovery-national-archives-online-catalogue/ ' ,
' info_dict ' : {
' id ' : ' Mrj4DVp2zeA ' ,
' ext ' : ' mp4 ' ,
2015-02-19 02:00:50 +01:00
' upload_date ' : ' 20150212 ' ,
2015-02-09 10:42:25 +01:00
' uploader ' : ' The National Archives UK ' ,
' description ' : ' md5:a236581cd2449dd2df4f93412f3f01c6 ' ,
' uploader_id ' : ' NationalArchives08 ' ,
' title ' : ' Webinar: Using Discovery, The National Archives’ online catalogue ' ,
} ,
2015-02-16 15:45:01 +01:00
} ,
# rtl.nl embed
{
' url ' : ' http://www.rtlnieuws.nl/nieuws/buitenland/aanslagen-kopenhagen ' ,
' playlist_mincount ' : 5 ,
' info_dict ' : {
' id ' : ' aanslagen-kopenhagen ' ,
' title ' : ' Aanslagen Kopenhagen | RTL Nieuws ' ,
}
2015-02-21 20:39:26 +01:00
} ,
# Zapiks embed
{
' url ' : ' http://www.skipass.com/news/116090-bon-appetit-s5ep3-baqueira-mi-cor.html ' ,
' info_dict ' : {
' id ' : ' 118046 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' EP3S5 - Bon Appétit - Baqueira Mi Corazon ! ' ,
}
} ,
2015-07-02 17:39:46 +02:00
# Kaltura embed (different embed code)
{
' url ' : ' http://www.premierchristianradio.com/Shows/Saturday/Unbelievable/Conference-Videos/Os-Guinness-Is-It-Fools-Talk-Unbelievable-Conference-2014 ' ,
' info_dict ' : {
' id ' : ' 1_a52wc67y ' ,
' ext ' : ' flv ' ,
' upload_date ' : ' 20150127 ' ,
' uploader_id ' : ' PremierMedia ' ,
' timestamp ' : int ,
' title ' : ' Os Guinness // Is It Fools Talk? // Unbelievable? Conference 2014 ' ,
} ,
} ,
2015-11-20 20:40:28 +01:00
# Kaltura embed protected with referrer
{
' url ' : ' http://www.disney.nl/disney-channel/filmpjes/achter-de-schermen#/videoId/violetta-achter-de-schermen-ruggero ' ,
' info_dict ' : {
' id ' : ' 1_g4fbemnq ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Violetta - Achter De Schermen - Ruggero ' ,
' description ' : ' Achter de schermen met Ruggero ' ,
' timestamp ' : 1435133761 ,
' upload_date ' : ' 20150624 ' ,
' uploader_id ' : ' echojecka ' ,
} ,
} ,
2016-06-01 12:37:34 +02:00
# Kaltura embed with single quotes
{
' url ' : ' http://fod.infobase.com/p_ViewPlaylist.aspx?AssignmentID=NUN8ZY ' ,
' info_dict ' : {
' id ' : ' 0_izeg5utt ' ,
' ext ' : ' mp4 ' ,
' title ' : ' 35871 ' ,
' timestamp ' : 1355743100 ,
' upload_date ' : ' 20121217 ' ,
' uploader_id ' : ' batchUser ' ,
} ,
' add_ie ' : [ ' Kaltura ' ] ,
} ,
2016-06-26 23:11:53 +02:00
{
# Kaltura embedded via quoted entry_id
' url ' : ' https://www.oreilly.com/ideas/my-cloud-makes-pretty-pictures ' ,
' info_dict ' : {
' id ' : ' 0_utuok90b ' ,
' ext ' : ' mp4 ' ,
' title ' : ' 06_matthew_brender_raj_dutt ' ,
' timestamp ' : 1466638791 ,
' upload_date ' : ' 20160622 ' ,
} ,
' add_ie ' : [ ' Kaltura ' ] ,
' expected_warnings ' : [
' Could not send HEAD request '
] ,
' params ' : {
' skip_download ' : True ,
}
} ,
2015-03-07 17:22:57 +01:00
# Eagle.Platform embed (generic URL)
{
' url ' : ' http://lenta.ru/news/2015/03/06/navalny/ ' ,
2016-04-25 16:48:17 +02:00
# Not checking MD5 as sometimes the direct HTTP link results in 404 and HLS is used
2015-03-07 17:22:57 +01:00
' info_dict ' : {
' id ' : ' 227304 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Навальный вышел на свободу ' ,
' description ' : ' md5:d97861ac9ae77377f3f20eaf9d04b4f5 ' ,
' thumbnail ' : ' re:^https?://.* \ .jpg$ ' ,
' duration ' : 87 ,
' view_count ' : int ,
' age_limit ' : 0 ,
} ,
} ,
2015-03-07 17:34:44 +01:00
# ClipYou (Eagle.Platform) embed (custom URL)
{
' url ' : ' http://muz-tv.ru/play/7129/ ' ,
2016-04-25 16:48:17 +02:00
# Not checking MD5 as sometimes the direct HTTP link results in 404 and HLS is used
2015-03-07 17:34:44 +01:00
' info_dict ' : {
' id ' : ' 12820 ' ,
' ext ' : ' mp4 ' ,
' title ' : " ' O Sole Mio " ,
' thumbnail ' : ' re:^https?://.* \ .jpg$ ' ,
' duration ' : 216 ,
' view_count ' : int ,
} ,
} ,
2015-03-08 13:07:10 +01:00
# Pladform embed
{
' url ' : ' http://muz-tv.ru/kinozal/view/7400/ ' ,
' info_dict ' : {
' id ' : ' 100183293 ' ,
' ext ' : ' mp4 ' ,
2015-04-16 17:37:15 +02:00
' title ' : ' Тайны перевала Дятлова • 1 серия 2 часть ' ,
2015-03-08 13:07:10 +01:00
' description ' : ' Документальный сериал-расследование одной из самых жутких тайн Х Х века ' ,
' thumbnail ' : ' re:^https?://.* \ .jpg$ ' ,
' duration ' : 694 ,
' age_limit ' : 0 ,
} ,
} ,
2015-04-15 18:13:01 +02:00
# Playwire embed
{
' url ' : ' http://www.cinemablend.com/new/First-Joe-Dirt-2-Trailer-Teaser-Stupid-Greatness-70874.html ' ,
' info_dict ' : {
' id ' : ' 3519514 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Joe Dirt 2 Beautiful Loser Teaser Trailer ' ,
' thumbnail ' : ' re:^https?://.* \ .png$ ' ,
' duration ' : 45.115 ,
} ,
} ,
2015-03-29 03:57:37 +02:00
# 5min embed
{
' url ' : ' http://techcrunch.com/video/facebook-creates-on-this-day-crunch-report/518726732/ ' ,
' md5 ' : ' 4c6f127a30736b59b3e2c19234ee2bf7 ' ,
' info_dict ' : {
' id ' : ' 518726732 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Facebook Creates " On This Day " | Crunch Report ' ,
} ,
} ,
2015-05-08 20:27:37 +02:00
# SVT embed
{
' url ' : ' http://www.svt.se/sport/ishockey/jagr-tacklar-giroux-under-intervjun ' ,
' info_dict ' : {
' id ' : ' 2900353 ' ,
' ext ' : ' flv ' ,
' title ' : ' Här trycker Jagr till Giroux (under SVT-intervjun) ' ,
' duration ' : 27 ,
' age_limit ' : 0 ,
} ,
} ,
2015-04-11 16:26:42 +02:00
# Crooks and Liars embed
{
' url ' : ' http://crooksandliars.com/2015/04/fox-friends-says-protecting-atheists ' ,
' info_dict ' : {
' id ' : ' 8RUoRhRi ' ,
' ext ' : ' mp4 ' ,
' title ' : " Fox & Friends Says Protecting Atheists From Discrimination Is Anti-Christian! " ,
' description ' : ' md5:e1a46ad1650e3a5ec7196d432799127f ' ,
' timestamp ' : 1428207000 ,
' upload_date ' : ' 20150405 ' ,
' uploader ' : ' Heather ' ,
} ,
} ,
# Crooks and Liars external embed
{
' url ' : ' http://theothermccain.com/2010/02/02/video-proves-that-bill-kristol-has-been-watching-glenn-beck/comment-page-1/ ' ,
' info_dict ' : {
' id ' : ' MTE3MjUtMzQ2MzA ' ,
' ext ' : ' mp4 ' ,
' title ' : ' md5:5e3662a81a4014d24c250d76d41a08d5 ' ,
' description ' : ' md5:9b8e9542d6c3c5de42d6451b7d780cec ' ,
' timestamp ' : 1265032391 ,
' upload_date ' : ' 20100201 ' ,
' uploader ' : ' Heather ' ,
} ,
} ,
2015-03-31 16:11:14 +02:00
# NBC Sports vplayer embed
2015-03-30 21:36:09 +02:00
{
2015-03-31 16:11:14 +02:00
' url ' : ' http://www.riderfans.com/forum/showthread.php?121827-Freeman&s=e98fa1ea6dc08e886b1678d35212494a ' ,
2015-03-30 21:36:09 +02:00
' info_dict ' : {
2015-03-31 16:11:14 +02:00
' id ' : ' ln7x1qSThw4k ' ,
' ext ' : ' flv ' ,
' title ' : " PFT Live: New leader in the ' new-look ' defense " ,
' description ' : ' md5:65a19b4bbfb3b0c0c5768bed1dfad74e ' ,
2016-04-24 14:44:52 +02:00
' uploader ' : ' NBCU-SPORTS ' ,
' upload_date ' : ' 20140107 ' ,
' timestamp ' : 1389118457 ,
2015-03-30 21:36:09 +02:00
} ,
2015-04-08 11:26:51 +02:00
} ,
2016-06-10 07:32:59 +02:00
# NBC News embed
{
' url ' : ' http://www.vulture.com/2016/06/letterman-couldnt-care-less-about-late-night.html ' ,
' md5 ' : ' 1aa589c675898ae6d37a17913cf68d66 ' ,
' info_dict ' : {
' id ' : ' 701714499682 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' PREVIEW: On Assignment: David Letterman ' ,
' description ' : ' A preview of Tom Brokaw \' s interview with David Letterman as part of the On Assignment series powered by Dateline. Airs Sunday June 12 at 7/6c. ' ,
} ,
} ,
2015-04-08 11:26:51 +02:00
# UDN embed
{
2016-06-01 13:23:44 +02:00
' url ' : ' https://video.udn.com/news/300346 ' ,
2015-04-14 07:10:10 +02:00
' md5 ' : ' fd2060e988c326991037b9aff9df21a6 ' ,
2015-04-08 11:26:51 +02:00
' info_dict ' : {
2015-04-14 07:10:10 +02:00
' id ' : ' 300346 ' ,
2015-04-08 11:26:51 +02:00
' ext ' : ' mp4 ' ,
2015-04-14 07:10:10 +02:00
' title ' : ' 中一中男師變性 全校師生力挺 ' ,
2015-04-08 11:26:51 +02:00
' thumbnail ' : ' re:^https?://.* \ .jpg$ ' ,
2016-06-01 13:23:44 +02:00
} ,
' params ' : {
# m3u8 download
' skip_download ' : True ,
} ,
2015-04-14 06:45:43 +02:00
} ,
# Ooyala embed
{
' url ' : ' http://www.businessinsider.com/excel-index-match-vlookup-video-how-to-2015-2?IR=T ' ,
' info_dict ' : {
' id ' : ' 50YnY4czr4ms1vJ7yz3xzq0excz_pUMs ' ,
' ext ' : ' mp4 ' ,
2015-10-16 17:02:40 +02:00
' description ' : ' VIDEO: INDEX/MATCH versus VLOOKUP. ' ,
2015-04-14 06:45:43 +02:00
' title ' : ' This is what separates the Excel masters from the wannabes ' ,
2015-12-04 16:18:02 +01:00
' duration ' : 191.933 ,
2015-04-14 06:45:43 +02:00
} ,
' params ' : {
# m3u8 downloads
' skip_download ' : True ,
}
2015-04-16 11:16:11 +02:00
} ,
2015-06-07 09:29:17 +02:00
# Brightcove URL in single quotes
{
' url ' : ' http://www.sportsnet.ca/baseball/mlb/sn-presents-russell-martin-world-citizen/ ' ,
' md5 ' : ' 4ae374f1f8b91c889c4b9203c8c752af ' ,
' info_dict ' : {
' id ' : ' 4255764656001 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' SN Presents: Russell Martin, World Citizen ' ,
' description ' : ' To understand why he was the Toronto Blue Jays’ top off-season priority is to appreciate his background and upbringing in Montreal, where he first developed his baseball skills. Written and narrated by Stephen Brunt. ' ,
' uploader ' : ' Rogers Sportsnet ' ,
2016-04-24 14:44:52 +02:00
' uploader_id ' : ' 1704050871 ' ,
' upload_date ' : ' 20150525 ' ,
' timestamp ' : 1432570283 ,
2015-06-07 09:29:17 +02:00
} ,
2015-06-21 15:30:34 +02:00
} ,
# Dailymotion Cloud video
{
' url ' : ' http://replay.publicsenat.fr/vod/le-debat/florent-kolandjian,dominique-cena,axel-decourtye,laurence-abeille,bruno-parmentier/175910 ' ,
2016-06-26 05:54:52 +02:00
' md5 ' : ' dcaf23ad0c67a256f4278bce6e0bae38 ' ,
2015-06-21 15:30:34 +02:00
' info_dict ' : {
2016-06-26 05:54:52 +02:00
' id ' : ' x2uy8t3 ' ,
2015-06-21 15:30:34 +02:00
' ext ' : ' mp4 ' ,
2016-06-26 05:54:52 +02:00
' title ' : ' Sauvons les abeilles ! - Le débat ' ,
' description ' : ' md5:d9082128b1c5277987825d684939ca26 ' ,
2015-06-21 15:30:34 +02:00
' thumbnail ' : ' re:^https?://.* \ .jpe?g$ ' ,
2016-06-26 05:54:52 +02:00
' timestamp ' : 1434970506 ,
' upload_date ' : ' 20150622 ' ,
' uploader ' : ' Public Sénat ' ,
' uploader_id ' : ' xa9gza ' ,
2015-06-21 15:30:34 +02:00
}
2015-06-22 09:02:53 +02:00
} ,
2015-06-24 19:23:16 +02:00
# OnionStudios embed
{
' url ' : ' http://www.clickhole.com/video/dont-understand-bitcoin-man-will-mumble-explanatio-2537 ' ,
' info_dict ' : {
' id ' : ' 2855 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Don’ t Understand Bitcoin? This Man Will Mumble An Explanation At You ' ,
' thumbnail ' : ' re:^https?://.* \ .jpe?g$ ' ,
' uploader ' : ' ClickHole ' ,
' uploader_id ' : ' clickhole ' ,
}
} ,
2015-06-27 14:28:10 +02:00
# SnagFilms embed
{
' url ' : ' http://whilewewatch.blogspot.ru/2012/06/whilewewatch-whilewewatch-gripping.html ' ,
' info_dict ' : {
' id ' : ' 74849a00-85a9-11e1-9660-123139220831 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' #whilewewatch ' ,
}
} ,
2015-06-22 09:02:53 +02:00
# AdobeTVVideo embed
{
' url ' : ' https://helpx.adobe.com/acrobat/how-to/new-experience-acrobat-dc.html?set=acrobat--get-started--essential-beginners ' ,
' md5 ' : ' 43662b577c018ad707a63766462b1e87 ' ,
' info_dict ' : {
' id ' : ' 2456 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' New experience with Acrobat DC ' ,
' description ' : ' New experience with Acrobat DC ' ,
' duration ' : 248.667 ,
} ,
2015-08-29 15:12:38 +02:00
} ,
# ScreenwaveMedia embed
{
' url ' : ' http://www.thecinemasnob.com/the-cinema-snob/a-nightmare-on-elm-street-2-freddys-revenge1 ' ,
' md5 ' : ' 24ace5baba0d35d55c6810b51f34e9e0 ' ,
' info_dict ' : {
' id ' : ' cinemasnob-55d26273809dd ' ,
' ext ' : ' mp4 ' ,
' title ' : ' cinemasnob ' ,
} ,
2015-09-11 05:46:21 +02:00
} ,
# BrightcoveInPageEmbed embed
{
' url ' : ' http://www.geekandsundry.com/tabletop-bonus-wils-final-thoughts-on-dread/ ' ,
' info_dict ' : {
' id ' : ' 4238694884001 ' ,
' ext ' : ' flv ' ,
' title ' : ' Tabletop: Dread, Last Thoughts ' ,
' description ' : ' Tabletop: Dread, Last Thoughts ' ,
' duration ' : 51690 ,
} ,
2015-11-21 09:43:01 +01:00
} ,
# JWPlayer with M3U8
{
' url ' : ' http://ren.tv/novosti/2015-09-25/sluchaynyy-prohozhiy-poymal-avtougonshchika-v-murmanske-video ' ,
' info_dict ' : {
' id ' : ' playlist ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Случайный прохожий поймал автоугонщика в Мурманске. ВИДЕО | Р Е Н Т В ' ,
' uploader ' : ' ren.tv ' ,
} ,
' params ' : {
# m3u8 downloads
' skip_download ' : True ,
}
2016-03-26 11:30:43 +01:00
} ,
# Brightcove embed, with no valid 'renditions' but valid 'IOSRenditions'
# This video can't be played in browsers if Flash disabled and UA set to iPhone, which is actually a false alarm
{
' url ' : ' https://dl.dropboxusercontent.com/u/29092637/interview.html ' ,
' info_dict ' : {
' id ' : ' 4785848093001 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' The Cardinal Pell Interview ' ,
' description ' : ' Sky News Contributor Andrew Bolt interviews George Pell in Rome, following the Cardinal \' s evidence before the Royal Commission into Child Abuse. ' ,
' uploader ' : ' GlobeCast Australia - GlobeStream ' ,
2016-04-24 14:44:52 +02:00
' uploader_id ' : ' 2733773828001 ' ,
' upload_date ' : ' 20160304 ' ,
' timestamp ' : 1457083087 ,
2016-03-26 11:30:43 +01:00
} ,
' params ' : {
# m3u8 downloads
' skip_download ' : True ,
} ,
} ,
2016-04-11 13:17:11 +02:00
# Another form of arte.tv embed
{
' url ' : ' http://www.tv-replay.fr/redirection/09-04-16/arte-reportage-arte-11508975.html ' ,
' md5 ' : ' 850bfe45417ddf221288c88a0cffe2e2 ' ,
' info_dict ' : {
' id ' : ' 030273-562_PLUS7-F ' ,
' ext ' : ' mp4 ' ,
' title ' : ' ARTE Reportage - Nulle part, en France ' ,
' description ' : ' md5:e3a0e8868ed7303ed509b9e3af2b870d ' ,
' upload_date ' : ' 20160409 ' ,
} ,
} ,
2016-03-31 20:42:55 +02:00
# LiveLeak embed
{
' url ' : ' http://www.wykop.pl/link/3088787/ ' ,
' md5 ' : ' ace83b9ed19b21f68e1b50e844fdf95d ' ,
' info_dict ' : {
' id ' : ' 874_1459135191 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Man shows poor quality of new apartment building ' ,
' description ' : ' The wall is like a sand pile. ' ,
' uploader ' : ' Lake8737 ' ,
}
} ,
2016-05-22 16:22:27 +02:00
# Duplicated embedded video URLs
{
' url ' : ' http://www.hudl.com/athlete/2538180/highlights/149298443 ' ,
' info_dict ' : {
' id ' : ' 149298443_480_16c25b74_2 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' vs. Blue Orange Spring Game ' ,
' uploader ' : ' www.hudl.com ' ,
} ,
} ,
2016-07-16 10:59:43 +02:00
# twitter:player:stream embed
{
' url ' : ' http://www.rtl.be/info/video/589263.aspx?CategoryID=288 ' ,
' info_dict ' : {
' id ' : ' master ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Une nouvelle espèce de dinosaure découverte en Argentine ' ,
' uploader ' : ' www.rtl.be ' ,
} ,
' params ' : {
# m3u8 downloads
' skip_download ' : True ,
} ,
} ,
2016-06-30 13:01:30 +02:00
# twitter:player embed
{
' url ' : ' http://www.theatlantic.com/video/index/484130/what-do-black-holes-sound-like/ ' ,
' md5 ' : ' a3e0df96369831de324f0778e126653c ' ,
' info_dict ' : {
' id ' : ' 4909620399001 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' What Do Black Holes Sound Like? ' ,
' description ' : ' what do black holes sound like ' ,
' upload_date ' : ' 20160524 ' ,
' uploader_id ' : ' 29913724001 ' ,
' timestamp ' : 1464107587 ,
' uploader ' : ' TheAtlantic ' ,
} ,
' add_ie ' : [ ' BrightcoveLegacy ' ] ,
2016-07-02 15:33:23 +02:00
} ,
# Facebook <iframe> embed
{
' url ' : ' https://www.hostblogger.de/blog/archives/6181-Auto-jagt-Betonmischer.html ' ,
2016-07-02 15:57:06 +02:00
' md5 ' : ' fbcde74f534176ecb015849146dd3aee ' ,
2016-07-02 15:33:23 +02:00
' info_dict ' : {
' id ' : ' 599637780109885 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Facebook video #599637780109885 ' ,
} ,
} ,
# Facebook API embed
{
' url ' : ' http://www.lothype.com/blue-stars-2016-preview-standstill-full-show/ ' ,
2016-07-02 15:57:06 +02:00
' md5 ' : ' a47372ee61b39a7b90287094d447d94e ' ,
2016-07-02 15:33:23 +02:00
' info_dict ' : {
' id ' : ' 10153467542406923 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Facebook video #10153467542406923 ' ,
} ,
2016-07-02 15:50:17 +02:00
} ,
# Wordpress "YouTube Video Importer" plugin
{
' url ' : ' http://www.lothype.com/blue-devils-drumline-stanford-lot-2016/ ' ,
2016-07-02 15:57:06 +02:00
' md5 ' : ' d16797741b560b485194eddda8121b48 ' ,
2016-07-02 15:50:17 +02:00
' info_dict ' : {
' id ' : ' HNTXWDXV9Is ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Blue Devils Drumline Stanford lot 2016 ' ,
' upload_date ' : ' 20160627 ' ,
' uploader_id ' : ' GENOCIDE8GENERAL10 ' ,
' uploader ' : ' cylus cyrus ' ,
} ,
} ,
2016-07-04 18:57:44 +02:00
{
# video stored on custom kaltura server
' url ' : ' http://www.expansion.com/multimedia/videos.html?media=EQcM30NHIPv ' ,
' md5 ' : ' 537617d06e64dfed891fa1593c4b30cc ' ,
' info_dict ' : {
' id ' : ' 0_1iotm5bh ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Elecciones británicas: 5 lecciones para Rajoy ' ,
' description ' : ' md5:435a89d68b9760b92ce67ed227055f16 ' ,
' uploader_id ' : ' videos.expansion@el-mundo.net ' ,
' upload_date ' : ' 20150429 ' ,
' timestamp ' : 1430303472 ,
} ,
' add_ie ' : [ ' Kaltura ' ] ,
} ,
2016-07-09 08:39:01 +02:00
{
# Non-standard Vimeo embed
' url ' : ' https://openclassrooms.com/courses/understanding-the-web ' ,
' md5 ' : ' 64d86f1c7d369afd9a78b38cbb88d80a ' ,
' info_dict ' : {
' id ' : ' 148867247 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Understanding the web - Teaser ' ,
' description ' : ' This is " Understanding the web - Teaser " by openclassrooms on Vimeo, the home for high quality videos and the people who love them. ' ,
' upload_date ' : ' 20151214 ' ,
' uploader ' : ' OpenClassrooms ' ,
' uploader_id ' : ' openclassrooms ' ,
} ,
' add_ie ' : [ ' Vimeo ' ] ,
} ,
2016-07-09 03:32:55 +02:00
# {
# # TODO: find another test
# # http://schema.org/VideoObject
# 'url': 'https://flipagram.com/f/nyvTSJMKId',
# 'md5': '888dcf08b7ea671381f00fab74692755',
# 'info_dict': {
# 'id': 'nyvTSJMKId',
# 'ext': 'mp4',
# 'title': 'Flipagram by sjuria101 featuring Midnight Memories by One Direction',
# 'description': '#love for cats.',
# 'timestamp': 1461244995,
# 'upload_date': '20160421',
# },
# 'params': {
# 'force_generic_extractor': True,
# },
# }
2013-07-10 17:49:11 +02:00
]
2013-06-23 20:31:45 +02:00
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
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
2015-03-02 15:21:11 +01:00
entries = [ ]
for it in doc . findall ( ' ./channel/item ' ) :
next_url = xpath_text ( it , ' link ' , fatal = False )
if not next_url :
enclosure_nodes = it . findall ( ' ./enclosure ' )
for e in enclosure_nodes :
next_url = e . attrib . get ( ' url ' )
if next_url :
break
if not next_url :
continue
entries . append ( {
' _type ' : ' url ' ,
' url ' : next_url ,
' title ' : it . find ( ' title ' ) . text ,
} )
2014-02-20 13:14:05 +01:00
return {
' _type ' : ' playlist ' ,
' id ' : url ,
' title ' : playlist_title ,
' description ' : playlist_desc ,
' entries ' : entries ,
}
2014-08-24 02:02:17 +02: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 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-29 17:17:43 +02:00
default_search = ' fixup_error '
2014-01-22 14:16:43 +01:00
2014-07-29 17:17:43 +02:00
if default_search in ( ' auto ' , ' auto_warning ' , ' fixup_error ' ) :
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 )
2014-07-29 17:17:43 +02:00
elif default_search != ' fixup_error ' :
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-29 17:17:43 +02:00
if default_search in ( ' error ' , ' fixup_error ' ) :
2014-07-06 11:22:44 +02:00
raise ExtractorError (
2014-11-23 22:21:46 +01:00
' %r is not a valid URL. '
' Set --default-search " ytsearch " (or run youtube-dl " ytsearch: %s " ) to search YouTube '
% ( url , url ) , expected = True )
2014-01-22 14:16:43 +01:00
else :
2014-10-23 21:13:45 +02:00
if ' : ' not in default_search :
default_search + = ' : '
2014-01-22 14:16:43 +01:00
return self . url_result ( default_search + url )
2014-08-24 04:47:18 +02:00
url , smuggled_data = unsmuggle_url ( url )
force_videoid = None
2014-09-28 12:14:16 +02:00
is_intentional = smuggled_data and smuggled_data . get ( ' to_generic ' )
2014-08-24 04:47:18 +02:00
if smuggled_data and ' force_videoid ' in smuggled_data :
force_videoid = smuggled_data [ ' force_videoid ' ]
video_id = force_videoid
else :
2015-05-30 21:23:58 +02:00
video_id = compat_urllib_parse_unquote ( 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
2014-08-24 06:58:11 +02:00
head_req = HEADRequest ( url )
2014-10-26 17:05:44 +01:00
head_response = self . _request_webpage (
2014-08-24 06:58:11 +02:00
head_req , video_id ,
note = False , errnote = ' Could not send HEAD request to %s ' % url ,
fatal = False )
2013-12-17 12:33:55 +01:00
2014-10-26 17:05:44 +01:00
if head_response is not False :
2013-12-17 12:33:55 +01:00
# Check for redirect
2014-10-26 17:05:44 +01:00
new_url = head_response . geturl ( )
2013-12-17 12:33:55 +01:00
if url != new_url :
self . report_following_redirect ( new_url )
2014-08-24 04:47:18 +02:00
if force_videoid :
new_url = smuggle_url (
new_url , { ' force_videoid ' : force_videoid } )
2013-12-17 12:04:33 +01:00
return self . url_result ( new_url )
2013-12-17 12:33:55 +01:00
2014-10-26 17:05:44 +01:00
full_response = None
if head_response is False :
2015-11-21 17:18:17 +01:00
request = sanitized_Request ( url )
2015-05-30 20:44:54 +02:00
request . add_header ( ' Accept-Encoding ' , ' * ' )
full_response = self . _request_webpage ( request , video_id )
2014-10-26 17:05:44 +01:00
head_response = full_response
2016-03-12 22:17:25 +01:00
info_dict = {
' id ' : video_id ,
' title ' : compat_urllib_parse_unquote ( os . path . splitext ( url_basename ( url ) ) [ 0 ] ) ,
2016-03-18 17:41:16 +01:00
' upload_date ' : unified_strdate ( head_response . headers . get ( ' Last-Modified ' ) )
2016-03-12 22:17:25 +01:00
}
2014-10-26 17:05:44 +01:00
# Check for direct link to a video
2016-03-18 16:50:44 +01:00
content_type = head_response . headers . get ( ' Content-Type ' , ' ' ) . lower ( )
2016-03-18 16:50:10 +01:00
m = re . match ( r ' ^(?P<type>audio|video|application(?=/(?:ogg$|(?:vnd \ .apple \ .|x-)?mpegurl)))/(?P<format_id>[^; \ s]+) ' , content_type )
2014-10-26 17:05:44 +01:00
if m :
2016-03-12 22:17:25 +01:00
format_id = m . group ( ' format_id ' )
if format_id . endswith ( ' mpegurl ' ) :
2016-02-04 01:25:36 +01:00
formats = self . _extract_m3u8_formats ( url , video_id , ' mp4 ' )
2016-03-12 22:17:25 +01:00
elif format_id == ' f4m ' :
formats = self . _extract_f4m_formats ( url , video_id )
2016-02-04 01:25:36 +01:00
else :
formats = [ {
' format_id ' : m . group ( ' format_id ' ) ,
' url ' : url ,
' vcodec ' : ' none ' if m . group ( ' type ' ) == ' audio ' else None
} ]
2016-03-18 17:43:07 +01:00
info_dict [ ' direct ' ] = True
2016-03-27 03:03:08 +02:00
self . _sort_formats ( formats )
2016-03-18 17:43:07 +01:00
info_dict [ ' formats ' ] = formats
2016-03-12 22:17:25 +01:00
return info_dict
2013-12-17 12:33:55 +01:00
2014-09-28 12:14:16 +02:00
if not self . _downloader . params . get ( ' test ' , False ) and not is_intentional :
2015-06-23 23:08:24 +02:00
force = self . _downloader . params . get ( ' force_generic_extractor ' , False )
self . _downloader . report_warning (
' %s on generic information extractor. ' % ( ' Forcing ' if force else ' Falling back ' ) )
2014-09-28 12:14:16 +02:00
2014-11-26 10:44:39 +01:00
if not full_response :
2015-11-21 17:18:17 +01:00
request = sanitized_Request ( url )
2015-05-30 20:44:54 +02:00
# Some webservers may serve compressed content of rather big size (e.g. gzipped flac)
# making it impossible to download only chunk of the file (yet we need only 512kB to
# test whether it's HTML or not). According to youtube-dl default Accept-Encoding
# that will always result in downloading the whole file that is not desirable.
# Therefore for extraction pass we have to override Accept-Encoding to any in order
# to accept raw bytes and being able to download only a chunk.
# It may probably better to solve this by checking Content-Type for application/octet-stream
# after HEAD request finishes, but not sure if we can rely on this.
request . add_header ( ' Accept-Encoding ' , ' * ' )
full_response = self . _request_webpage ( request , video_id )
2014-11-26 10:44:39 +01:00
2016-03-18 17:45:28 +01:00
first_bytes = full_response . read ( 512 )
# Is it an M3U playlist?
2016-03-19 00:43:43 +01:00
if first_bytes . startswith ( b ' #EXTM3U ' ) :
2016-03-18 17:45:28 +01:00
info_dict [ ' formats ' ] = self . _extract_m3u8_formats ( url , video_id , ' mp4 ' )
2016-03-27 03:03:08 +02:00
self . _sort_formats ( info_dict [ ' formats ' ] )
2016-03-18 17:45:28 +01:00
return info_dict
2014-11-26 10:44:39 +01:00
# Maybe it's a direct link to a video?
# Be careful not to download the whole thing!
2015-01-23 01:21:30 +01:00
if not is_html ( first_bytes ) :
2014-11-26 10:44:39 +01:00
self . _downloader . report_warning (
' URL could be a direct video link, returning it as such. ' )
2016-03-12 22:17:25 +01:00
info_dict . update ( {
2014-11-26 10:44:39 +01:00
' direct ' : True ,
' url ' : url ,
2016-03-12 22:17:25 +01:00
} )
return info_dict
2014-11-26 10:44:39 +01:00
webpage = self . _webpage_read_content (
full_response , url , video_id , prefix = first_bytes )
2013-06-23 20:31:45 +02:00
self . report_extraction ( video_id )
2013-11-18 13:28:26 +01:00
2016-02-06 14:35:32 +01:00
# Is it an RSS feed, a SMIL file, an XSPF playlist or a MPD manifest?
2014-02-20 13:14:05 +01:00
try :
2015-10-26 16:41:24 +01:00
doc = compat_etree_fromstring ( webpage . encode ( ' utf-8 ' ) )
2014-02-20 13:14:05 +01:00
if doc . tag == ' rss ' :
return self . _extract_rss ( url , video_id , doc )
2015-08-01 21:13:59 +02:00
elif re . match ( r ' ^(?: { [^}]+})?smil$ ' , doc . tag ) :
2016-03-27 03:03:08 +02:00
smil = self . _parse_smil ( doc , url , video_id )
self . _sort_formats ( smil [ ' formats ' ] )
return smil
2015-08-09 15:43:42 +02:00
elif doc . tag == ' { http://xspf.org/ns/0/}playlist ' :
return self . playlist_result ( self . _parse_xspf ( doc , video_id ) , video_id )
2016-02-06 14:35:32 +01:00
elif re . match ( r ' (?i)^(?: { [^}]+})?MPD$ ' , doc . tag ) :
2016-03-12 22:17:25 +01:00
info_dict [ ' formats ' ] = self . _parse_mpd_formats (
doc , video_id , mpd_base_url = url . rpartition ( ' / ' ) [ 0 ] )
2016-03-27 03:03:08 +02:00
self . _sort_formats ( info_dict [ ' formats ' ] )
2016-03-12 22:17:25 +01:00
return info_dict
elif re . match ( r ' ^ { http://ns \ .adobe \ .com/f4m/[12] \ .0}manifest$ ' , doc . tag ) :
info_dict [ ' formats ' ] = self . _parse_f4m_formats ( doc , url , video_id )
2016-03-27 03:03:08 +02:00
self . _sort_formats ( info_dict [ ' formats ' ] )
2016-03-12 22:17:25 +01:00
return info_dict
2014-02-21 16:59:10 +01:00
except compat_xml_parse_error :
2014-02-20 13:14:05 +01:00
pass
2014-08-24 02:02:17 +02: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 22:38:49 +01: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
2015-07-15 22:30:47 +02:00
webpage = compat_urllib_parse_unquote ( webpage )
2014-02-24 17:44:31 +01:00
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
2016-05-14 19:58:25 +02:00
video_title = self . _og_search_title (
webpage , default = None ) or 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
2014-08-24 05:31:32 +02: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 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
2016-05-14 19:58:25 +02:00
video_description = self . _og_search_description ( webpage , default = None )
video_thumbnail = self . _og_search_thumbnail ( webpage , default = None )
2014-08-22 18:19:56 +02:00
# Helper method
2015-01-02 15:54:30 +01:00
def _playlist_from_matches ( matches , getter = None , ie = None ) :
2014-09-24 11:05:14 +02:00
urlrs = orderedSet (
2015-01-02 15:54:30 +01:00
self . url_result ( self . _proto_relative_url ( getter ( m ) if getter else m ) , ie )
2014-09-24 11:05:14 +02:00
for m in matches )
2014-08-22 18:19:56 +02:00
return self . playlist_result (
urlrs , playlist_id = video_id , playlist_title = video_title )
2015-11-14 01:03:32 +01:00
# Look for Brightcove Legacy Studio embeds
2015-11-14 00:54:16 +01:00
bc_urls = BrightcoveLegacyIE . _extract_brightcove_urls ( webpage )
2014-02-03 15:19:40 +01:00
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 } ) ,
2015-11-14 01:05:46 +01:00
' ie_key ' : ' BrightcoveLegacy '
2014-02-03 15:19:40 +01:00
} 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
2015-11-14 01:03:07 +01:00
# Look for Brightcove New Studio embeds
bc_urls = BrightcoveNewIE . _extract_urls ( webpage )
if bc_urls :
return _playlist_from_matches ( bc_urls , ie = ' BrightcoveNew ' )
2015-09-11 05:46:21 +02:00
2016-05-22 02:52:39 +02:00
# Look for ThePlatform embeds
tp_urls = ThePlatformIE . _extract_urls ( webpage )
if tp_urls :
return _playlist_from_matches ( tp_urls , ie = ' ThePlatform ' )
2016-06-08 23:02:27 +02:00
# Look for Vessel embeds
vessel_urls = VesselIE . _extract_urls ( webpage )
if vessel_urls :
return _playlist_from_matches ( vessel_urls , ie = VesselIE . ie_key ( ) )
2015-02-16 15:45:01 +01:00
# Look for embedded rtl.nl player
matches = re . findall (
2015-06-11 15:04:12 +02:00
r ' <iframe[^>]+?src= " ((?:https?:)?//(?:www \ .)?rtl \ .nl/system/videoplayer/[^ " ]+(?:video_)?embed[^ " ]+) " ' ,
2015-02-16 15:45:01 +01:00
webpage )
if matches :
return _playlist_from_matches ( matches , ie = ' RtlNl ' )
2015-06-21 12:23:58 +02:00
vimeo_url = VimeoIE . _extract_vimeo_url ( url , webpage )
if vimeo_url is not None :
return self . url_result ( vimeo_url )
2013-12-22 03:34:13 +01:00
2015-07-20 12:49:53 +02:00
vid_me_embed_url = self . _search_regex (
r ' src=[ \' " ](https?://vid \ .me/[^ \' " ]+)[ \' " ] ' ,
webpage , ' vid.me embed ' , default = None )
if vid_me_embed_url is not None :
return self . url_result ( vid_me_embed_url , ' Vidme ' )
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 = |
2014-08-22 17:40:36 +02:00
data - video - url = |
2014-06-09 22:06:45 +02:00
< embed [ ^ > ] + ? src = |
2014-10-26 14:15:48 +01:00
embedSWF \( ? : \s * |
new \s + SWFObject \(
2014-06-09 22:06:45 +02:00
)
( [ " \' ])
2014-09-10 13:29:20 +02:00
( ? P < url > ( ? : https ? : ) ? / / ( ? : www \. ) ? youtube ( ? : - nocookie ) ? \. com /
2014-09-25 01:58:49 +02:00
( ? : embed | v | p ) / . + ? )
2013-12-19 20:44:30 +01:00
\1 ''' , webpage)
2013-11-18 13:28:26 +01:00
if matches :
2014-08-22 18:19:56 +02:00
return _playlist_from_matches (
2014-09-24 11:05:14 +02:00
matches , lambda m : unescapeHTML ( m [ 1 ] ) )
2013-10-18 11:44:57 +02:00
2014-11-25 14:34:19 +01:00
# Look for lazyYT YouTube embed
matches = re . findall (
r ' class= " lazyYT " data-youtube-id= " ([^ " ]+) " ' , webpage )
if matches :
return _playlist_from_matches ( matches , lambda m : unescapeHTML ( m ) )
2016-07-02 15:50:17 +02:00
# Look for Wordpress "YouTube Video Importer" plugin
matches = re . findall ( r ''' (?x)<div[^>]+
class = ( ? P < q1 > [ \' " ])[^ \' " ]* \b yvii_single_video_player \b [^ \' " ]*(?P=q1)[^>]+
data - video_id = ( ? P < q2 > [ \' " ])([^ \' " ]+)(?P=q2) ' ' ' , webpage )
if matches :
return _playlist_from_matches ( matches , lambda m : m [ - 1 ] )
2016-07-06 17:37:54 +02:00
matches = DailymotionIE . _extract_urls ( webpage )
2013-12-01 01:21:33 +01:00
if matches :
2016-07-06 17:37:54 +02:00
return _playlist_from_matches ( matches )
2013-12-01 01:21:33 +01:00
2014-10-02 20:42:45 +02:00
# Look for embedded Dailymotion playlist player (#3822)
m = re . search (
r ' <iframe[^>]+?src=([ " \' ])(?P<url>(?:https?:)?//(?:www \ .)?dailymotion \ .[a-z] { 2,3}/widget/jukebox \ ?.+?) \ 1 ' , webpage )
if m :
playlists = re . findall (
r ' list \ [ \ ]=/playlist/([^/]+)/ ' , unescapeHTML ( m . group ( ' url ' ) ) )
if playlists :
return _playlist_from_matches (
playlists , lambda p : ' //dailymotion.com/playlist/ %s ' % p )
2013-12-06 09:15:04 +01:00
# Look for embedded Wistia player
match = re . search (
2014-10-23 18:03:07 +02:00
r ' <(?:meta[^>]+?content|iframe[^>]+?src)=([ " \' ])(?P<url>(?:https?:)?//(?:fast \ .)?wistia \ .net/embed/iframe/.+?) \ 1 ' , webpage )
2013-12-06 09:15:04 +01:00
if match :
2014-10-18 00:52:55 +02:00
embed_url = self . _proto_relative_url (
unescapeHTML ( match . group ( ' url ' ) ) )
2013-12-06 09:15:04 +01:00
return {
' _type ' : ' url_transparent ' ,
2014-10-18 00:52:55 +02:00
' url ' : embed_url ,
2013-12-06 09:15:04 +01:00
' ie_key ' : ' Wistia ' ,
' uploader ' : video_uploader ,
}
2014-11-23 20:41:03 +01:00
2014-10-18 00:52:55 +02:00
match = re . search ( r ' (?:id=[ " \' ]wistia_|data-wistia-?id=[ " \' ]|Wistia \ .embed \ ([ " \' ])(?P<id>[^ " \' ]+) ' , webpage )
2014-09-20 02:02:11 +02:00
if match :
return {
' _type ' : ' url_transparent ' ,
2016-05-20 17:55:35 +02:00
' url ' : ' wistia: %s ' % match . group ( ' id ' ) ,
2014-09-20 02:02:11 +02:00
' ie_key ' : ' Wistia ' ,
' uploader ' : video_uploader ,
}
2013-12-06 09:15:04 +01:00
2016-05-20 17:33:31 +02:00
match = re . search (
r ''' (?sx)
< script [ ^ > ] + src = ( [ " ' ])(?:https?:)?//fast \ .wistia \ .com/assets/external/E-v1 \ .js \1 [^>]*>.*?
< div [ ^ > ] + class = ( [ " ' ]).*? \b wistia_async_(?P<id>[a-z0-9]+) \b .*? \2
''' , webpage)
if match :
return self . url_result ( self . _proto_relative_url (
' wistia: %s ' % match . group ( ' id ' ) ) , ' Wistia ' )
2015-05-08 20:23:35 +02:00
# Look for SVT player
svt_url = SVTIE . _extract_url ( webpage )
if svt_url :
return self . url_result ( svt_url , ' SVT ' )
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 ' ) )
2015-01-28 06:08:19 +01:00
# Look for embedded Viddler player
2015-01-28 18:07:37 +01:00
mobj = re . search (
r ' <(?:iframe[^>]+?src|param[^>]+?value)=([ " \' ])(?P<url>(?:https?:)?//(?:www \ .)?viddler \ .com/(?:embed|player)/.+?) \ 1 ' ,
webpage )
2015-01-28 06:08:19 +01:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) )
2013-12-16 21:45:21 +01:00
2015-03-19 16:26:57 +01:00
# Look for NYTimes player
mobj = re . search (
r ' <iframe[^>]+src=([ " \' ])(?P<url>(?:https?:)?//graphics8 \ .nytimes \ .com/bcvideo/[^/]+/iframe/embed \ .html.+?) \ 1> ' ,
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) )
2015-03-22 03:18:13 +01:00
# Look for Libsyn player
mobj = re . search (
r ' <iframe[^>]+src=([ " \' ])(?P<url>(?:https?:)?//html5-player \ .libsyn \ .com/embed/.+?) \ 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
2015-08-07 21:55:59 +02:00
mobj = ( re . search ( r ' player \ .ooyala \ .com/[^ " ?]+[?#][^ " ]*?(?:embedCode|ec)=(?P<ec>[^ " &]+) ' , webpage ) or
2015-02-04 14:33:37 +01:00
re . search ( r ' OO \ .Player \ .create \ ([ \' " ].*?[ \' " ], \ s*[ \' " ](?P<ec>. {32} )[ \' " ] ' , webpage ) or
2015-04-14 06:45:43 +02:00
re . search ( r ' SBN \ .VideoLinkset \ .ooyala \ ([ \' " ](?P<ec>. {32} )[ \' " ] \ ) ' , webpage ) or
re . search ( r ' data-ooyala-video-id \ s*= \ s*[ \' " ](?P<ec>. {32} )[ \' " ] ' , webpage ) )
2013-12-19 20:28:52 +01:00
if mobj is not None :
2015-10-16 17:02:40 +02:00
return OoyalaIE . _build_url_result ( smuggle_url ( mobj . group ( ' ec ' ) , { ' domain ' : url } ) )
2013-12-19 20:28:52 +01:00
2015-02-04 14:33:37 +01:00
# Look for multiple Ooyala embeds on SBN network websites
mobj = re . search ( r ' SBN \ .VideoLinkset \ .entryGroup \ (( \ [.*? \ ]) ' , webpage )
if mobj is not None :
embeds = self . _parse_json ( mobj . group ( 1 ) , video_id , fatal = False )
if embeds :
return _playlist_from_matches (
2015-10-16 17:02:40 +02:00
embeds , getter = lambda v : OoyalaIE . _url_for_embed_code ( smuggle_url ( v [ ' provider_video_id ' ] , { ' domain ' : url } ) ) , ie = ' Ooyala ' )
2015-02-04 14:33:37 +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-29 22:26:46 +01: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 02:11:46 +01:00
2014-04-05 12:20:05 +02:00
# Look for embedded NovaMov-based player
2014-01-08 02:07:11 +01:00
mobj = re . search (
2014-05-17 13:12:12 +02:00
r ''' (?x)<(?:pagespeed_)?iframe[^>]+?src=([ " \ ' ])
2014-04-05 12:20:05 +02: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 02:07:11 +01:00
if mobj is not None :
2014-04-05 12:20:05 +02:00
return self . url_result ( mobj . group ( ' url ' ) )
2014-04-05 10:49:45 +02:00
2014-01-21 18:10:14 +01:00
# Look for embedded Facebook player
2016-07-02 15:33:23 +02:00
facebook_url = FacebookIE . _extract_url ( webpage )
if facebook_url is not None :
return self . url_result ( facebook_url , ' Facebook ' )
2014-01-21 18:10:14 +01:00
2014-02-28 17:51:54 +01: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 ' )
2016-02-21 04:51:54 +01:00
# Look for embedded Odnoklassniki player
mobj = re . search ( r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://(?:odnoklassniki|ok) \ .ru/videoembed/.+?) \ 1 ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Odnoklassniki ' )
2014-06-29 15:18:23 +02: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-29 22:26:46 +01: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 :
2015-07-17 19:39:32 +02:00
return self . url_result ( compat_urllib_parse_unquote ( mobj . group ( ' url ' ) ) )
2014-02-24 01:15:51 +01:00
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 :
2014-08-22 18:19:56 +02:00
return _playlist_from_matches (
matches , getter = unescapeHTML , ie = ' FunnyOrDie ' )
2014-03-11 16:51:36 +01:00
2015-01-02 15:46:17 +01:00
# Look for BBC iPlayer embed
matches = re . findall ( r ' setPlaylist \ ( " (https?://www \ .bbc \ .co \ .uk/iplayer/[^/]+/[ \ da-z] {8} ) " \ ) ' , webpage )
if matches :
2015-01-02 15:55:09 +01:00
return _playlist_from_matches ( matches , ie = ' BBCCoUk ' )
2015-01-02 15:46:17 +01:00
2014-03-16 20:00:31 +01:00
# Look for embedded RUTV player
rutv_url = RUTVIE . _extract_url ( webpage )
if rutv_url :
return self . url_result ( rutv_url , ' RUTV ' )
2015-06-12 12:22:46 +02:00
# Look for embedded TVC player
2015-06-12 14:15:30 +02:00
tvc_url = TVCIE . _extract_url ( webpage )
if tvc_url :
return self . url_result ( tvc_url , ' TVC ' )
2015-06-12 12:22:46 +02:00
2015-05-15 19:09:34 +02:00
# Look for embedded SportBox player
sportbox_urls = SportBoxEmbedIE . _extract_urls ( webpage )
if sportbox_urls :
return _playlist_from_matches ( sportbox_urls , ie = ' SportBoxEmbed ' )
2015-06-12 23:39:14 +02:00
# Look for embedded PornHub player
2015-06-12 23:36:16 +02:00
pornhub_url = PornHubIE . _extract_url ( webpage )
if pornhub_url :
return self . url_result ( pornhub_url , ' PornHub ' )
2015-06-21 19:11:25 +02:00
# Look for embedded XHamster player
xhamster_urls = XHamsterEmbedIE . _extract_urls ( webpage )
if xhamster_urls :
return _playlist_from_matches ( xhamster_urls , ie = ' XHamsterEmbed ' )
2016-02-27 12:15:49 +01:00
# Look for embedded TNAFlixNetwork player
tnaflix_urls = TNAFlixNetworkEmbedIE . _extract_urls ( webpage )
if tnaflix_urls :
return _playlist_from_matches ( tnaflix_urls , ie = TNAFlixNetworkEmbedIE . ie_key ( ) )
2015-06-12 14:37:09 +02:00
# Look for embedded Tvigle player
mobj = re . search (
r ' <iframe[^>]+?src=([ " \' ])(?P<url>(?:https?:)?//cloud \ .tvigle \ .ru/video/.+?) \ 1 ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Tvigle ' )
2014-03-22 10:20:44 +01:00
# Look for embedded TED player
mobj = re . search (
2015-01-05 18:16:47 +01:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://embed(?:-ssl)? \ .ted \ .com/.+?) \ 1 ' , webpage )
2014-03-22 10:20:44 +01:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' TED ' )
2014-04-04 16:23:09 +02: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 22:01:47 +01:00
# Look for embedded arte.tv player
mobj = re . search (
2016-04-11 13:17:11 +02:00
r ' <(?:script|iframe) [^>]*?src= " (?P<url>http://www \ .arte \ .tv/(?:playerv2/embed|arte_vp/index)[^ " ]+) " ' ,
2014-03-24 22:01:47 +01:00
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' ArteTVEmbed ' )
2015-07-18 18:56:00 +02:00
# Look for embedded francetv player
mobj = re . search (
r ' <iframe[^>]+?src=([ " \' ])(?P<url>(?:https?://)?embed \ .francetv \ .fr/ \ ?ue=.+?) \ 1 ' ,
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) )
2014-03-28 13:58:49 +01:00
# Look for embedded smotri.com player
smotri_url = SmotriIE . _extract_url ( webpage )
if smotri_url :
return self . url_result ( smotri_url , ' Smotri ' )
2015-07-09 20:25:36 +02:00
# Look for embedded Myvi.ru player
2015-07-09 20:27:44 +02:00
myvi_url = MyviIE . _extract_url ( webpage )
2015-07-09 20:25:36 +02:00
if myvi_url :
return self . url_result ( myvi_url )
2016-01-10 16:17:47 +01:00
# Look for embedded soundcloud player
2014-05-05 03:12:41 +02:00
mobj = re . search (
2014-10-29 20:27:58 +01:00
r ' <iframe \ s+(?:[a-zA-Z0-9_-]+= " [^ " ]+ " \ s+)*src= " (?P<url>https?://(?:w \ .)?soundcloud \ .com/player[^ " ]+) " ' ,
2014-05-05 03:12:41 +02:00
webpage )
if mobj is not None :
url = unescapeHTML ( mobj . group ( ' url ' ) )
return self . url_result ( url )
2014-06-22 21:38:04 +02:00
# Look for embedded mtvservices player
2015-09-26 15:47:20 +02:00
mtvservices_url = MTVServicesEmbeddedIE . _extract_url ( webpage )
if mtvservices_url :
return self . url_result ( mtvservices_url , ie = ' MTVServicesEmbedded ' )
2014-06-22 21:38:04 +02:00
2014-08-16 08:56:22 +02: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 15:20:49 +02:00
# Look for embedded sbs.com.au player
mobj = re . search (
2015-02-09 14:46:10 +01:00
r ''' (?x)
( ? :
< meta \s + property = " og:video " \s + content = |
< iframe [ ^ > ] + ? src =
)
( [ " \' ])(?P<url>https?://(?:www \ .)?sbs \ .com \ .au/ondemand/video/.+?) \1 ' ' ' ,
2014-08-23 15:20:49 +02:00
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' SBS ' )
2014-12-12 02:57:36 +01:00
# Look for embedded Cinchcast player
mobj = re . search (
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://player \ .cinchcast \ .com/.+?) \ 1 ' ,
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Cinchcast ' )
2014-09-02 15:19:28 +02:00
mobj = re . search (
2014-10-31 22:01:58 +01:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://m(?:lb)? \ .mlb \ .com/shared/video/embed/embed \ .html \ ?.+?) \ 1 ' ,
2014-09-02 15:19:28 +02:00
webpage )
2015-05-03 20:20:07 +02:00
if not mobj :
mobj = re . search (
r ' data-video-link=[ " \' ](?P<url>http://m.mlb.com/video/[^ " \' ]+) ' ,
webpage )
2014-09-02 15:19:28 +02:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' MLB ' )
2014-10-13 14:59:35 +02:00
mobj = re . search (
2015-09-27 01:55:48 +02:00
r ' <(?:iframe|script)[^>]+?src=([ " \' ])(?P<url> %s ) \ 1 ' % CondeNastIE . EMBED_URL ,
2014-10-13 14:59:35 +02:00
webpage )
if mobj is not None :
return self . url_result ( self . _proto_relative_url ( mobj . group ( ' url ' ) , scheme = ' http: ' ) , ' CondeNast ' )
2014-11-13 16:12:51 +01:00
mobj = re . search (
2016-05-22 19:39:09 +02:00
r ' <iframe[^>]+src= " (?P<url>https?://(?:new \ .)?livestream \ .com/[^ " ]+/player[^ " ]+) " ' ,
2014-11-13 16:12:51 +01:00
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Livestream ' )
2015-02-21 20:39:26 +01:00
# Look for Zapiks embed
mobj = re . search (
r ' <iframe[^>]+src= " (?P<url>https?://(?:www \ .)?zapiks \ .fr/index \ .php \ ?.+?) " ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Zapiks ' )
2015-02-26 23:34:19 +01:00
# Look for Kaltura embeds
2016-06-27 17:45:26 +02:00
kaltura_url = KalturaIE . _extract_url ( webpage )
if kaltura_url :
return self . url_result ( smuggle_url ( kaltura_url , { ' source_url ' : url } ) , KalturaIE . ie_key ( ) )
2015-02-26 23:34:19 +01:00
2015-03-07 17:22:57 +01:00
# Look for Eagle.Platform embeds
2016-06-29 18:01:34 +02:00
eagleplatform_url = EaglePlatformIE . _extract_url ( webpage )
if eagleplatform_url :
return self . url_result ( eagleplatform_url , EaglePlatformIE . ie_key ( ) )
2015-03-07 17:22:57 +01:00
2015-03-07 17:34:44 +01:00
# Look for ClipYou (uses Eagle.Platform) embeds
mobj = re . search (
r ' <iframe[^>]+src= " https?://(?P<host>media \ .clipyou \ .ru)/index/player \ ?.* \ brecord_id=(?P<id> \ d+).* " ' , webpage )
if mobj is not None :
return self . url_result ( ' eagleplatform: %(host)s : %(id)s ' % mobj . groupdict ( ) , ' EaglePlatform ' )
2015-03-08 13:07:10 +01:00
# Look for Pladform embeds
2015-12-07 17:03:21 +01:00
pladform_url = PladformIE . _extract_url ( webpage )
if pladform_url :
return self . url_result ( pladform_url )
2015-03-08 13:07:10 +01:00
2015-12-29 18:58:23 +01:00
# Look for Videomore embeds
videomore_url = VideomoreIE . _extract_url ( webpage )
if videomore_url :
return self . url_result ( videomore_url )
2015-04-15 18:10:08 +02:00
# Look for Playwire embeds
mobj = re . search (
r ' <script[^>]+data-config=([ " \' ])(?P<url>(?:https?:)?//config \ .playwire \ .com/.+?) \ 1 ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) )
2015-03-29 03:57:37 +02:00
# Look for 5min embeds
mobj = re . search (
r ' <meta[^>]+property= " og:video " [^>]+content= " https?://embed \ .5min \ .com/(?P<id>[0-9]+)/? ' , webpage )
if mobj is not None :
return self . url_result ( ' 5min: %s ' % mobj . group ( ' id ' ) , ' FiveMin ' )
2015-04-11 16:20:20 +02:00
# Look for Crooks and Liars embeds
mobj = re . search (
r ' <(?:iframe[^>]+src|param[^>]+value)=([ " \' ])(?P<url>(?:https?:)?//embed \ .crooksandliars \ .com/(?:embed|v)/.+?) \ 1 ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) )
2015-03-30 21:36:09 +02:00
# Look for NBC Sports VPlayer embeds
nbc_sports_url = NBCSportsVPlayerIE . _extract_url ( webpage )
if nbc_sports_url :
return self . url_result ( nbc_sports_url , ' NBCSportsVPlayer ' )
2016-06-10 07:32:59 +02:00
# Look for NBC News embeds
nbc_news_embed_url = re . search (
r ' <iframe[^>]+src=([ " \' ])(?P<url>(?:https?:)?//www \ .nbcnews \ .com/widget/video-embed/[^ " \' ]+) \ 1 ' , webpage )
if nbc_news_embed_url :
return self . url_result ( nbc_news_embed_url . group ( ' url ' ) , ' NBCNews ' )
2015-06-29 09:01:30 +02:00
# Look for Google Drive embeds
2015-12-21 03:05:34 +01:00
google_drive_url = GoogleDriveIE . _extract_url ( webpage )
2015-06-29 09:01:30 +02:00
if google_drive_url :
return self . url_result ( google_drive_url , ' GoogleDrive ' )
2015-04-08 11:26:51 +02:00
# Look for UDN embeds
mobj = re . search (
2015-11-19 15:22:57 +01:00
r ' <iframe[^>]+src= " (?P<url> %s ) " ' % UDNEmbedIE . _PROTOCOL_RELATIVE_VALID_URL , webpage )
2015-04-08 11:26:51 +02:00
if mobj is not None :
return self . url_result (
2015-04-08 15:39:34 +02:00
compat_urlparse . urljoin ( url , mobj . group ( ' url ' ) ) , ' UDNEmbed ' )
2015-04-08 11:26:51 +02:00
2015-04-20 21:18:38 +02:00
# Look for Senate ISVP iframe
senate_isvp_url = SenateISVPIE . _search_iframe_url ( webpage )
if senate_isvp_url :
2015-05-15 19:23:51 +02:00
return self . url_result ( senate_isvp_url , ' SenateISVP ' )
2015-04-20 21:18:38 +02:00
2015-06-21 15:30:34 +02:00
# Look for Dailymotion Cloud videos
dmcloud_url = DailymotionCloudIE . _extract_dmcloud_url ( webpage )
if dmcloud_url :
return self . url_result ( dmcloud_url , ' DailymotionCloud ' )
2015-06-24 19:19:50 +02:00
# Look for OnionStudios embeds
onionstudios_url = OnionStudiosIE . _extract_url ( webpage )
if onionstudios_url :
return self . url_result ( onionstudios_url )
2016-04-29 12:14:42 +02:00
# Look for ViewLift embeds
viewlift_url = ViewLiftEmbedIE . _extract_url ( webpage )
if viewlift_url :
return self . url_result ( viewlift_url )
2015-06-27 14:26:14 +02:00
2015-12-21 04:24:58 +01:00
# Look for JWPlatform embeds
jwplatform_url = JWPlatformIE . _extract_url ( webpage )
if jwplatform_url :
return self . url_result ( jwplatform_url , ' JWPlatform ' )
2015-08-29 07:55:20 +02:00
# Look for ScreenwaveMedia embeds
2015-08-29 15:07:31 +02:00
mobj = re . search ( ScreenwaveMediaIE . EMBED_PATTERN , webpage )
2015-08-29 07:55:20 +02:00
if mobj is not None :
2015-08-29 15:07:31 +02:00
return self . url_result ( unescapeHTML ( mobj . group ( ' url ' ) ) , ' ScreenwaveMedia ' )
2015-08-29 07:55:20 +02:00
2016-01-19 16:51:46 +01:00
# Look for Digiteka embeds
digiteka_url = DigitekaIE . _extract_url ( webpage )
if digiteka_url :
return self . url_result ( self . _proto_relative_url ( digiteka_url ) , DigitekaIE . ie_key ( ) )
2015-09-25 11:52:48 +02:00
2016-01-30 20:45:56 +01:00
# Look for Limelight embeds
mobj = re . search ( r ' LimelightPlayer \ .doLoad(Media|Channel|ChannelList) \ ([ " \' ](?P<id>[a-z0-9] {32} ) ' , webpage )
if mobj :
lm = {
' Media ' : ' media ' ,
' Channel ' : ' channel ' ,
' ChannelList ' : ' channel_list ' ,
}
return self . url_result ( ' limelight: %s : %s ' % (
lm [ mobj . group ( 1 ) ] , mobj . group ( 2 ) ) , ' Limelight %s ' % mobj . group ( 1 ) , mobj . group ( 2 ) )
2015-06-22 09:02:53 +02:00
# Look for AdobeTVVideo embeds
mobj = re . search (
r ' <iframe[^>]+src=[ \' " ]((?:https?:)?//video \ .tv \ .adobe \ .com/v/ \ d+[^ " ]+)[ \' " ] ' ,
webpage )
if mobj is not None :
return self . url_result (
self . _proto_relative_url ( unescapeHTML ( mobj . group ( 1 ) ) ) ,
' AdobeTVVideo ' )
2016-03-23 16:55:08 +01:00
# Look for Vine embeds
mobj = re . search (
r ' <iframe[^>]+src=[ \' " ]((?:https?:)?//(?:www \ .)?vine \ .co/v/[^/]+/embed/(?:simple|postcard)) ' ,
webpage )
if mobj is not None :
return self . url_result (
self . _proto_relative_url ( unescapeHTML ( mobj . group ( 1 ) ) ) , ' Vine ' )
2016-03-24 09:32:27 +01:00
# Look for Instagram embeds
instagram_embed_url = InstagramIE . _extract_embed_url ( webpage )
if instagram_embed_url is not None :
2016-04-16 18:39:20 +02:00
return self . url_result (
self . _proto_relative_url ( instagram_embed_url ) , InstagramIE . ie_key ( ) )
2016-03-24 09:32:27 +01:00
2016-03-31 20:42:55 +02:00
# Look for LiveLeak embeds
liveleak_url = LiveLeakIE . _extract_url ( webpage )
if liveleak_url :
return self . url_result ( liveleak_url , ' LiveLeak ' )
2016-05-14 19:40:34 +02:00
# Look for 3Q SDN embeds
threeqsdn_url = ThreeQSDNIE . _extract_url ( webpage )
if threeqsdn_url :
2016-05-14 19:58:25 +02:00
return {
' _type ' : ' url_transparent ' ,
' ie_key ' : ThreeQSDNIE . ie_key ( ) ,
' url ' : self . _proto_relative_url ( threeqsdn_url ) ,
' title ' : video_title ,
' description ' : video_description ,
' thumbnail ' : video_thumbnail ,
' uploader ' : video_uploader ,
}
2016-05-14 19:40:34 +02:00
2016-07-08 22:29:07 +02:00
# Looking for http://schema.org/VideoObject
json_ld = self . _search_json_ld (
webpage , video_id , default = None , expected_type = ' VideoObject ' )
if json_ld and json_ld . get ( ' url ' ) :
info_dict . update ( {
' title ' : video_title or info_dict [ ' title ' ] ,
' description ' : video_description ,
' thumbnail ' : video_thumbnail ,
' age_limit ' : age_limit
} )
info_dict . update ( json_ld )
return info_dict
2014-10-09 14:26:23 +02:00
def check_video ( vurl ) :
2015-02-09 10:42:25 +01:00
if YoutubeIE . suitable ( vurl ) :
return True
2014-10-09 14:26:23 +02:00
vpath = compat_urlparse . urlparse ( vurl ) . path
vext = determine_ext ( vpath )
return ' . ' in vpath and vext not in ( ' swf ' , ' png ' , ' jpg ' , ' srt ' , ' sbv ' , ' sub ' , ' vtt ' , ' ttml ' )
def filter_video ( urls ) :
return list ( filter ( check_video , urls ) )
2013-06-23 20:31:45 +02:00
# Start with something easy: JW Player in SWFObject
2014-10-09 14:26:23 +02:00
found = filter_video ( re . findall ( r ' flashvars: [ \' " ](?:.*&)?file=(http[^ \' " &]*) ' , webpage ) )
2014-04-30 02:23:51 +02:00
if not found :
2014-01-05 05:34:06 +01:00
# Look for gorilla-vid style embedding
2014-10-09 14:26:23 +02:00
found = filter_video ( re . findall ( r ''' (?sx)
2014-04-21 16:16:53 +02:00
( ? :
jw_plugins |
JWPlayerOptions |
jwplayer \s * \( \s * [ " ' ][^ ' " ] + [ " ' ] \ s* \ ) \ s* \ .setup
)
2015-02-09 10:42:25 +01:00
. * ?
[ ' " ]?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-10-09 14:26:23 +02:00
found = filter_video ( re . findall ( r ' [^A-Za-z0-9]?(?:file|source)=(http[^ \' " &]*) ' , webpage ) )
2014-04-30 02:23:51 +02:00
if not found :
# Broaden the findall a little bit: JWPlayer JS loader
2014-10-09 14:26:23 +02:00
found = filter_video ( re . findall (
2015-08-05 17:19:52 +02:00
r ' [^A-Za-z0-9]?(?:file|video_url)[ " \' ]?: \ s*[ " \' ](http(?![^ \' " ]+ \ .[0-9]+[ \' " ])[^ \' " ]+)[ " \' ] ' , webpage ) )
2014-08-24 05:31:32 +02:00
if not found :
# Flow player
2014-10-09 14:26:23 +02:00
found = filter_video ( re . findall ( r ''' (?xs)
2014-08-24 05:31:32 +02:00
flowplayer \( " [^ " ] + " , \ s*
\{ [ ^ } ] + ? \} \s * ,
2015-01-30 08:41:40 +01:00
\s * \{ [ ^ } ] + ? [ " ' ]?clip[ " ' ]? \ s*: \ s* \ { \ s*
2014-08-24 05:31:32 +02:00
[ " ' ]?url[ " ' ]? \ s*: \ s*[ " ' ] ( [ ^ " ' ]+)[ " ' ]
2014-10-09 14:26:23 +02:00
''' , webpage))
2015-01-23 12:00:25 +01:00
if not found :
# Cinerama player
found = re . findall (
r " cinerama \ .embedPlayer \ ( \ s* \ ' [^ ' ]+ \ ' , \ s* ' ([^ ' ]+) ' " , webpage )
2014-04-30 02:23:51 +02:00
if not found :
2013-06-23 20:31:45 +02:00
# Try to find twitter cards info
2016-07-16 10:59:43 +02:00
# twitter:player:stream should be checked before twitter:player since
# it is expected to contain a raw stream (see
# https://dev.twitter.com/cards/types/player#On_twitter.com_via_desktop_browser)
2014-10-09 14:26:23 +02:00
found = filter_video ( re . findall (
r ' <meta (?:property|name)= " twitter:player:stream " (?:content|value)= " (.+?) " ' , webpage ) )
2014-04-30 02:23:51 +02:00
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-10-09 14:26:23 +02:00
found = filter_video ( re . findall ( r ' <meta.*?property= " og:video " .*?content= " (.*?) " ' , webpage ) )
2014-04-30 02:23:51 +02:00
if not found :
2013-08-21 04:32:22 +02:00
# HTML5 video
2015-09-03 07:13:05 +02:00
found = re . findall ( r ' (?s)<(?:video|audio)[^<]*(?:>.*?<source[^>]*)? \ s+src=[ " \' ](.*?)[ " \' ] ' , webpage )
2014-04-30 02:23:51 +02:00
if not found :
2015-03-17 19:05:40 +01:00
REDIRECT_REGEX = r ' [0-9] { ,2}; \ s*(?:URL|url)= \' ?([^ \' " ]+) '
2014-05-16 15:32:53 +02:00
found = re . search (
2014-02-27 07:21:59 +01:00
r ' (?i)<meta \ s+(?=(?:[a-z-]+= " [^ " ]+ " \ s+)*http-equiv= " refresh " ) '
2015-03-17 19:05:40 +01:00
r ' (?:[a-z-]+= " [^ " ]+ " \ s+)*?content= " %s ' % REDIRECT_REGEX ,
2014-02-27 07:21:59 +01:00
webpage )
2015-03-17 18:51:40 +01:00
if not found :
# Look also in Refresh HTTP header
refresh_header = head_response . headers . get ( ' Refresh ' )
if refresh_header :
2015-09-20 07:16:12 +02:00
# In python 2 response HTTP headers are bytestrings
if sys . version_info < ( 3 , 0 ) and isinstance ( refresh_header , str ) :
refresh_header = refresh_header . decode ( ' iso-8859-1 ' )
2015-03-17 19:05:40 +01:00
found = re . search ( REDIRECT_REGEX , refresh_header )
2014-04-30 02:23:51 +02:00
if found :
2015-07-22 23:33:49 +02:00
new_url = compat_urlparse . urljoin ( url , unescapeHTML ( found . group ( 1 ) ) )
2014-02-27 07:21:59 +01:00
self . report_following_redirect ( new_url )
return {
' _type ' : ' url ' ,
' url ' : new_url ,
}
2016-07-16 10:59:43 +02:00
if not found :
# twitter:player is a https URL to iframe player that may or may not
# be supported by youtube-dl thus this is checked the very last (see
# https://dev.twitter.com/cards/types/player#On_twitter.com_via_desktop_browser)
embed_url = self . _html_search_meta ( ' twitter:player ' , webpage , default = None )
if embed_url :
return self . url_result ( embed_url )
2014-04-30 02:23:51 +02:00
if not found :
2014-12-30 19:35:35 +01:00
raise UnsupportedError ( url )
2013-06-23 20:31:45 +02:00
2014-04-30 02:23:51 +02:00
entries = [ ]
2016-05-22 16:22:27 +02:00
for video_url in orderedSet ( found ) :
2016-04-24 10:23:21 +02:00
video_url = unescapeHTML ( video_url )
2015-11-21 07:12:34 +01:00
video_url = video_url . replace ( ' \\ / ' , ' / ' )
2014-04-30 02:23:51 +02:00
video_url = compat_urlparse . urljoin ( url , video_url )
2015-07-17 19:39:32 +02:00
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
2015-11-21 09:08:54 +01:00
entry_info_dict = {
' id ' : video_id ,
' uploader ' : video_uploader ,
' title ' : video_title ,
' age_limit ' : age_limit ,
}
2015-08-09 15:43:42 +02:00
ext = determine_ext ( video_url )
if ext == ' smil ' :
2015-11-21 09:08:54 +01:00
entry_info_dict [ ' formats ' ] = self . _extract_smil_formats ( video_url , video_id )
2015-08-09 15:43:42 +02:00
elif ext == ' xspf ' :
return self . playlist_result ( self . _extract_xspf_playlist ( video_url , video_id ) , video_id )
2015-11-21 09:43:01 +01:00
elif ext == ' m3u8 ' :
entry_info_dict [ ' formats ' ] = self . _extract_m3u8_formats ( video_url , video_id , ext = ' mp4 ' )
2016-02-06 14:42:03 +01:00
elif ext == ' mpd ' :
entry_info_dict [ ' formats ' ] = self . _extract_mpd_formats ( video_url , video_id )
2016-03-12 22:38:20 +01:00
elif ext == ' f4m ' :
entry_info_dict [ ' formats ' ] = self . _extract_f4m_formats ( video_url , video_id )
2015-04-16 11:16:11 +02:00
else :
2015-11-21 09:08:54 +01:00
entry_info_dict [ ' url ' ] = video_url
2016-03-27 03:03:08 +02:00
if entry_info_dict . get ( ' formats ' ) :
self . _sort_formats ( entry_info_dict [ ' formats ' ] )
2015-11-21 09:08:54 +01:00
entries . append ( entry_info_dict )
2014-04-30 02:23:51 +02:00
if len ( entries ) == 1 :
2014-05-01 11:28:37 +02:00
return entries [ 0 ]
2014-04-30 02:23:51 +02:00
else :
for num , e in enumerate ( entries , start = 1 ) :
2015-02-25 17:56:51 +01:00
# 'url' results don't have a title
if e . get ( ' title ' ) is not None :
e [ ' title ' ] = ' %s ( %d ) ' % ( e [ ' title ' ] , num )
2014-04-30 02:23:51 +02:00
return {
' _type ' : ' playlist ' ,
' entries ' : entries ,
}