2013-07-10 08:49:11 -07:00
# encoding: utf-8
2014-01-05 16:47:52 -08:00
from __future__ import unicode_literals
2013-06-23 11:31:45 -07:00
import os
import re
from . common import InfoExtractor
2014-01-05 16:42:58 -08:00
from . youtube import YoutubeIE
2014-11-02 02:23:40 -08:00
from . . compat import (
2015-05-30 12:23:58 -07:00
compat_urllib_parse_unquote ,
compat_urllib_request ,
2013-08-28 03:47:27 -07:00
compat_urlparse ,
2014-02-21 07:59:10 -08:00
compat_xml_parse_error ,
2014-11-02 02:23:40 -08:00
)
from . . utils import (
2014-08-27 16:00:59 -07:00
determine_ext ,
2013-06-23 11:31:45 -07:00
ExtractorError ,
2014-08-23 17:02:17 -07:00
float_or_none ,
2013-12-20 08:05:28 -08:00
HEADRequest ,
2015-01-22 16:21:30 -08:00
is_html ,
2014-08-22 09:19:56 -07:00
orderedSet ,
2014-03-10 09:31:32 -07:00
parse_xml ,
2013-10-15 03:05:13 -07:00
smuggle_url ,
unescapeHTML ,
2013-12-17 03:33:55 -08:00
unified_strdate ,
2014-08-23 19:47:18 -07:00
unsmuggle_url ,
2014-12-30 10:35:35 -08:00
UnsupportedError ,
2013-12-17 03:33:55 -08:00
url_basename ,
2015-03-02 06:21:11 -08:00
xpath_text ,
2013-06-23 11:31:45 -07:00
)
2013-07-10 08:49:11 -07:00
from . brightcove import BrightcoveIE
2015-03-30 12:36:09 -07:00
from . nbc import NBCSportsVPlayerIE
2013-12-19 11:28:52 -08:00
from . ooyala import OoyalaIE
2014-03-16 12:00:31 -07:00
from . rutv import RUTVIE
2015-06-12 03:24:13 -07:00
from . tvc import TVCIE
2015-05-15 10:09:34 -07:00
from . sportbox import SportBoxEmbedIE
2014-03-28 05:58:49 -07:00
from . smotri import SmotriIE
2015-07-09 11:27:44 -07:00
from . myvi import MyviIE
2014-10-13 05:59:35 -07:00
from . condenast import CondeNastIE
2015-04-08 02:26:51 -07:00
from . udn import UDNEmbedIE
2015-04-20 12:18:38 -07:00
from . senateisvp import SenateISVPIE
2015-04-20 22:48:02 -07:00
from . bliptv import BlipTVIE
2015-05-08 11:23:35 -07:00
from . svt import SVTIE
2015-06-12 14:36:16 -07:00
from . pornhub import PornHubIE
2015-06-21 10:11:25 -07:00
from . xhamster import XHamsterEmbedIE
2015-06-21 03:23:58 -07:00
from . vimeo import VimeoIE
2015-06-21 06:30:34 -07:00
from . dailymotion import DailymotionCloudIE
2015-06-24 10:19:50 -07:00
from . onionstudios import OnionStudiosIE
2015-06-27 05:26:14 -07:00
from . snagfilms import SnagFilmsEmbedIE
2015-06-29 00:01:30 -07:00
from . googledrive import GoogleDriveEmbedIE
2013-06-23 11:31:45 -07:00
2013-08-24 13:49:52 -07:00
2013-06-23 11:31:45 -07:00
class GenericIE ( InfoExtractor ) :
2014-01-05 16:47:52 -08:00
IE_DESC = ' Generic downloader that works on some sites '
2013-06-23 11:31:45 -07:00
_VALID_URL = r ' .* '
2014-01-05 16:47:52 -08:00
IE_NAME = ' generic '
2013-07-10 08:49:11 -07:00
_TESTS = [
2015-05-30 13:22:29 -07: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 13:36:20 -07:00
# Direct link to media delivered compressed (until Accept-Encoding is *)
2015-05-30 13:22:29 -07: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 ' : [
r ' 501.*Not Implemented '
] ,
} ,
# 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 ' ,
}
} ,
# 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 ,
}
} ,
2013-07-10 08:49:11 -07:00
{
2014-01-05 16:47:52 -08:00
' url ' : ' http://www.hodiho.fr/2013/02/regis-plante-sa-jeep.html ' ,
2014-04-14 04:51:46 -07:00
' md5 ' : ' 85b90ccc9d73b4acd9138d3af4c27f89 ' ,
2014-01-05 16:47:52 -08:00
' info_dict ' : {
2014-04-14 04:51:46 -07:00
' id ' : ' 13601338388002 ' ,
' ext ' : ' mp4 ' ,
2014-01-05 16:47:52 -08:00
' uploader ' : ' www.hodiho.fr ' ,
' title ' : ' R \u00e9 gis plante sa Jeep ' ,
2013-07-10 08:49:11 -07:00
}
} ,
2013-10-27 06:40:25 -07:00
# bandcamp page with custom domain
{
2014-01-05 16:47:52 -08:00
' add_ie ' : [ ' Bandcamp ' ] ,
' url ' : ' http://bronyrock.com/track/the-pony-mash ' ,
' info_dict ' : {
2014-04-14 04:56:29 -07:00
' id ' : ' 3235767654 ' ,
' ext ' : ' mp3 ' ,
2014-01-05 16:47:52 -08:00
' title ' : ' The Pony Mash ' ,
' uploader ' : ' M_Pallante ' ,
2013-10-27 06:40:25 -07:00
} ,
2014-01-05 16:47:52 -08:00
' skip ' : ' There is a limit of 200 free downloads / month for the test song ' ,
2013-10-27 06:40:25 -07:00
} ,
2013-11-06 07:40:24 -08:00
# embedded brightcove video
2013-11-07 12:06:48 -08:00
# it also tests brightcove videos that need to set the 'Referer' in the
# http requests
2013-11-06 07:40:24 -08:00
{
2014-01-05 16:47:52 -08:00
' add_ie ' : [ ' Brightcove ' ] ,
' url ' : ' http://www.bfmtv.com/video/bfmbusiness/cours-bourse/cours-bourse-l-analyse-technique-154522/ ' ,
' info_dict ' : {
' id ' : ' 2765128793001 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Le cours de bourse : l’ analyse technique ' ,
' description ' : ' md5:7e9ad046e968cb2d1114004aba466fd9 ' ,
' uploader ' : ' BFM BUSINESS ' ,
2013-11-06 07:40:24 -08:00
} ,
2014-01-05 16:47:52 -08:00
' params ' : {
' skip_download ' : True ,
2013-11-06 07:40:24 -08:00
} ,
} ,
2014-01-27 18:35:32 -08:00
{
# https://github.com/rg3/youtube-dl/issues/2253
' url ' : ' http://bcove.me/i6nfkrc3 ' ,
' md5 ' : ' 0ba9446db037002366bab3b3eb30c88c ' ,
' info_dict ' : {
2014-04-14 04:56:29 -07:00
' id ' : ' 3101154703001 ' ,
' ext ' : ' mp4 ' ,
2014-01-27 18:35:32 -08:00
' title ' : ' Still no power ' ,
' uploader ' : ' thestar.com ' ,
' description ' : ' Mississauga resident David Farmer is still out of power as a result of the ice storm a month ago. To keep the house warm, Farmer cuts wood from his property for a wood burning stove downstairs. ' ,
} ,
' add_ie ' : [ ' Brightcove ' ] ,
} ,
2014-04-01 06:17:35 -07:00
{
' url ' : ' http://www.championat.com/video/football/v/87/87499.html ' ,
' md5 ' : ' fb973ecf6e4a78a67453647444222983 ' ,
' info_dict ' : {
' id ' : ' 3414141473001 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Видео. Удаление Дзагоева (ЦСКА) ' ,
' description ' : ' Онлайн-трансляция матча ЦСКА - " Волга " ' ,
' uploader ' : ' Championat ' ,
} ,
} ,
2014-11-03 13:13:46 -08:00
{
2014-11-05 14:14:33 -08:00
# https://github.com/rg3/youtube-dl/issues/3541
2014-11-03 13:13:46 -08:00
' add_ie ' : [ ' Brightcove ' ] ,
' url ' : ' http://www.kijk.nl/sbs6/leermijvrouwenkennen/videos/jqMiXKAYan2S/aflevering-1 ' ,
' info_dict ' : {
' id ' : ' 3866516442001 ' ,
2014-11-05 14:14:33 -08:00
' ext ' : ' mp4 ' ,
2014-11-03 13:13:46 -08:00
' title ' : ' Leer mij vrouwen kennen: Aflevering 1 ' ,
' description ' : ' Leer mij vrouwen kennen: Aflevering 1 ' ,
' uploader ' : ' SBS Broadcasting ' ,
} ,
2014-11-05 14:14:33 -08:00
' skip ' : ' Restricted to Netherlands ' ,
2014-11-03 13:13:46 -08:00
' params ' : {
2014-11-05 14:14:33 -08:00
' skip_download ' : True , # m3u8 download
2014-11-03 13:13:46 -08:00
} ,
} ,
2013-12-19 11:28:52 -08:00
# ooyala video
{
2014-01-05 16:47:52 -08:00
' url ' : ' http://www.rollingstone.com/music/videos/norwegian-dj-cashmere-cat-goes-spartan-on-with-me-premiere-20131219 ' ,
2015-01-05 04:07:24 -08:00
' md5 ' : ' 166dd577b433b4d4ebfee10b0824d8ff ' ,
2014-01-05 16:47:52 -08:00
' info_dict ' : {
' id ' : ' BwY2RxaTrTkslxOfcan0UCf0YqyvWysJ ' ,
' ext ' : ' mp4 ' ,
2014-01-20 16:40:34 -08:00
' title ' : ' 2cc213299525360.mov ' , # that's what we get
2013-12-19 11:28:52 -08:00
} ,
2015-01-05 04:07:24 -08:00
' add_ie ' : [ ' Ooyala ' ] ,
2013-12-19 11:28:52 -08:00
} ,
2015-02-04 05:33:37 -08:00
# multiple ooyala embeds on SBN network websites
{
' url ' : ' http://www.sbnation.com/college-football-recruiting/2015/2/3/7970291/national-signing-day-rationalizations-itll-be-ok-itll-be-ok ' ,
' info_dict ' : {
' id ' : ' national-signing-day-rationalizations-itll-be-ok-itll-be-ok ' ,
' title ' : ' 25 lies you will tell yourself on National Signing Day - SBNation.com ' ,
} ,
' playlist_mincount ' : 3 ,
' params ' : {
' skip_download ' : True ,
} ,
' add_ie ' : [ ' Ooyala ' ] ,
} ,
2014-02-23 16:15:51 -08: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 05:05:44 -08:00
' title ' : ' Tested: Grinding Coffee at 2000 Frames Per Second ' ,
' upload_date ' : ' 20140225 ' ,
' description ' : ' md5:06a40fbf30b220468f1e0957c0f558ff ' ,
' uploader ' : ' Tested ' ,
' uploader_id ' : ' testedcom ' ,
2014-02-23 16:15:51 -08:00
} ,
# No need to test YoutubeIE here
' params ' : {
' skip_download ' : True ,
} ,
} ,
2014-03-11 08:51:36 -07: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 12:00:31 -07:00
} ,
2014-03-11 08:51:36 -07:00
} ,
2015-01-02 06:56:42 -08:00
# BBC iPlayer embeds
{
' url ' : ' http://www.bbc.co.uk/blogs/adamcurtis/posts/BUGGER ' ,
' info_dict ' : {
' title ' : ' BBC - Blogs - Adam Curtis - BUGGER ' ,
} ,
' playlist_mincount ' : 18 ,
} ,
2014-03-16 12:00:31 -07:00
# RUTV embed
{
' url ' : ' http://www.rg.ru/2014/03/15/reg-dfo/anklav-anons.html ' ,
' info_dict ' : {
' id ' : ' 776940 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Охотское море стало целиком российским ' ,
' description ' : ' md5:5ed62483b14663e2a95ebbe115eb8f43 ' ,
} ,
' params ' : {
# m3u8 download
' skip_download ' : True ,
} ,
2014-03-20 08:33:23 -07:00
} ,
2015-06-12 03:28:45 -07: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 10:18:21 -07: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 12:15:55 -07: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 10:18:28 -07: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 08:33:23 -07:00
# Embedded TED video
{
' url ' : ' http://en.support.wordpress.com/videos/ted-talks/ ' ,
2014-09-28 20:12:57 -07:00
' md5 ' : ' 65fdff94098e4a607385a60c5177c638 ' ,
2014-03-20 08:33:23 -07:00
' info_dict ' : {
2014-09-28 20:12:57 -07:00
' id ' : ' 1969 ' ,
2014-03-20 08:33:23 -07:00
' ext ' : ' mp4 ' ,
2014-09-28 20:12:57 -07:00
' title ' : ' Hidden miracles of the natural world ' ,
' uploader ' : ' Louie Schwartzberg ' ,
' description ' : ' md5:8145d19d320ff3e52f28401f4c4283b9 ' ,
2014-03-20 08:33:23 -07:00
}
2014-03-11 08:51:36 -07:00
} ,
2014-04-04 07:23:09 -07:00
# Embeded Ustream video
{
' url ' : ' http://www.american.edu/spa/pti/nsa-privacy-janus-2014.cfm ' ,
' md5 ' : ' 27b99cdb639c9b12a79bca876a073417 ' ,
' info_dict ' : {
2014-04-04 09:56:29 -07:00
' id ' : ' 45734260 ' ,
' ext ' : ' flv ' ,
' uploader ' : ' AU SPA: The NSA and Privacy ' ,
2014-04-04 07:23:09 -07:00
' title ' : ' NSA and Privacy Forum Debate featuring General Hayden and Barton Gellman '
}
} ,
2014-03-14 14:39:53 -07:00
# nowvideo embed hidden behind percent encoding
{
' url ' : ' http://www.waoanime.tv/the-super-dimension-fortress-macross-episode-1/ ' ,
' md5 ' : ' 2baf4ddd70f697d94b1c18cf796d5107 ' ,
' info_dict ' : {
' id ' : ' 06e53103ca9aa ' ,
' ext ' : ' flv ' ,
' title ' : ' Macross Episode 001 Watch Macross Episode 001 onl ' ,
' description ' : ' No description ' ,
} ,
2014-03-21 14:14:24 -07:00
} ,
2014-03-24 14:01:47 -07: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 09:56:00 -07: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-20 20:47:52 -07:00
# Condé Nast embed
{
' url ' : ' http://www.wired.com/2014/04/honda-asimo/ ' ,
' md5 ' : ' ba0dfe966fa007657bd1443ee672db0f ' ,
' info_dict ' : {
' id ' : ' 53501be369702d3275860000 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Honda’ s New Asimo Robot Is More Human Than Ever ' ,
}
2014-04-29 16:46:06 -07:00
} ,
# Dailymotion embed
{
' url ' : ' http://www.spi0n.com/zap-spi0n-com-n216/ ' ,
' md5 ' : ' 441aeeb82eb72c422c7f14ec533999cd ' ,
' info_dict ' : {
' id ' : ' k2mm4bCdJ6CQ2i7c8o2 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Le Zap de Spi0n n°216 - Zapping du Web ' ,
' uploader ' : ' Spi0n ' ,
} ,
' add_ie ' : [ ' Dailymotion ' ] ,
2014-06-09 13:06:45 -07: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 12:38:04 -07:00
# MTVSercices embed
{
' url ' : ' http://www.gametrailers.com/news-post/76093/north-america-europe-is-getting-that-mario-kart-8-mercedes-dlc-too ' ,
' md5 ' : ' 35727f82f58c76d996fc188f9755b0d5 ' ,
' info_dict ' : {
' id ' : ' 0306a69b-8adf-4fb5-aace-75f8e8cbfca9 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Review ' ,
' description ' : ' Mario \' s life in the fast lane has never looked so good. ' ,
} ,
} ,
2014-05-21 02:55:37 -07: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-28 20:12:57 -07:00
' id ' : ' 4vAffPZIT44 ' ,
2014-05-21 02:55:37 -07:00
' ext ' : ' mp4 ' ,
2014-09-28 20:12:57 -07:00
' title ' : ' Asphalt 8: Airborne - Update - Welcome to Dubai! ' ,
2014-08-22 09:19:56 -07:00
' uploader ' : ' Gameloft ' ,
' uploader_id ' : ' gameloft ' ,
2014-09-28 20:12:57 -07:00
' upload_date ' : ' 20140828 ' ,
' description ' : ' md5:c80da9ed3d83ae6d1876c834de03e1c4 ' ,
2014-08-22 09:19:56 -07:00
} ,
' params ' : {
' skip_download ' : True ,
2014-05-21 02:55:37 -07:00
}
2014-08-23 17:02:17 -07: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-23 20:31:32 -07: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-26 16:45:15 -07:00
' title ' : ' Busty Blonde Siri Tit Fuck While Wank at HandjobHub.com ' ,
2014-08-23 20:31:32 -07:00
}
2014-08-25 09:03:01 -07:00
} ,
2014-08-27 15:58:24 -07: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 06:19:28 -07: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-19 17:02:11 -07:00
# Wistia embed
{
' url ' : ' http://education-portal.com/academy/lesson/north-american-exploration-failed-colonies-of-spain-france-england.html#lesson ' ,
' md5 ' : ' 8788b683c777a5cf25621eaf286d0c23 ' ,
' info_dict ' : {
' id ' : ' 1cfaf6b7ea ' ,
' ext ' : ' mov ' ,
' title ' : ' md5:51364a8d3d009997ba99656004b5e20d ' ,
' duration ' : 643.0 ,
' filesize ' : 182808282 ,
' uploader ' : ' education-portal.com ' ,
} ,
} ,
2014-10-23 07:57:13 -07:00
{
' url ' : ' http://thoughtworks.wistia.com/medias/uxjb0lwrcz ' ,
' md5 ' : ' baf49c2baa8a7de5f3fc145a8506dcd4 ' ,
' info_dict ' : {
' id ' : ' uxjb0lwrcz ' ,
' ext ' : ' mp4 ' ,
2014-10-23 08:53:56 -07:00
' title ' : ' Conversation about Hexagonal Rails Part 1 - ThoughtWorks ' ,
2014-10-23 07:57:13 -07:00
' duration ' : 1715.0 ,
2014-10-23 08:53:56 -07:00
' uploader ' : ' thoughtworks.wistia.com ' ,
2014-10-26 12:49:51 -07:00
} ,
2014-10-23 07:57:13 -07:00
} ,
2014-10-29 12:27:58 -07: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 07:12:51 -08: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 ' ,
}
} ,
2014-11-25 05:34:19 -08:00
# LazyYT
{
' url ' : ' http://discourse.ubuntu.com/t/unity-8-desktop-mode-windows-on-mir/1986 ' ,
' info_dict ' : {
2015-02-17 15:49:10 -08:00
' id ' : ' 1986 ' ,
2014-11-25 05:34:19 -08:00
' title ' : ' Unity 8 desktop-mode windows on Mir! - Ubuntu Discourse ' ,
} ,
' playlist_mincount ' : 2 ,
2014-11-26 01:44:39 -08:00
} ,
2014-12-11 17:57:36 -08: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 03:00:25 -08: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-27 21:08:19 -08: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 01:42:25 -08:00
} ,
2015-03-21 19:20:27 -07: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 01:42:25 -08: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-18 17:00:50 -08:00
' upload_date ' : ' 20150212 ' ,
2015-02-09 01:42:25 -08:00
' uploader ' : ' The National Archives UK ' ,
' description ' : ' md5:a236581cd2449dd2df4f93412f3f01c6 ' ,
' uploader_id ' : ' NationalArchives08 ' ,
' title ' : ' Webinar: Using Discovery, The National Archives’ online catalogue ' ,
} ,
2015-02-16 06:45:01 -08: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 11:39:26 -08: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-02-26 14:34:19 -08:00
# Kaltura embed
{
' url ' : ' http://www.monumentalnetwork.com/videos/john-carlson-postgame-2-25-15 ' ,
' info_dict ' : {
' id ' : ' 1_eergr3h1 ' ,
' ext ' : ' mp4 ' ,
' upload_date ' : ' 20150226 ' ,
' uploader_id ' : ' MonumentalSports-Kaltura@perfectsensedigital.com ' ,
' timestamp ' : int ,
' title ' : ' John Carlson Postgame 2/25/15 ' ,
} ,
} ,
2015-07-02 08:39:46 -07: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-03-07 08:22:57 -08:00
# Eagle.Platform embed (generic URL)
{
' url ' : ' http://lenta.ru/news/2015/03/06/navalny/ ' ,
' 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 08:34:44 -08:00
# ClipYou (Eagle.Platform) embed (custom URL)
{
' url ' : ' http://muz-tv.ru/play/7129/ ' ,
' info_dict ' : {
' id ' : ' 12820 ' ,
' ext ' : ' mp4 ' ,
' title ' : " ' O Sole Mio " ,
' thumbnail ' : ' re:^https?://.* \ .jpg$ ' ,
' duration ' : 216 ,
' view_count ' : int ,
} ,
} ,
2015-03-08 05:07:10 -07:00
# Pladform embed
{
' url ' : ' http://muz-tv.ru/kinozal/view/7400/ ' ,
' info_dict ' : {
' id ' : ' 100183293 ' ,
' ext ' : ' mp4 ' ,
2015-04-16 08:37:15 -07:00
' title ' : ' Тайны перевала Дятлова • 1 серия 2 часть ' ,
2015-03-08 05:07:10 -07:00
' description ' : ' Документальный сериал-расследование одной из самых жутких тайн Х Х века ' ,
' thumbnail ' : ' re:^https?://.* \ .jpg$ ' ,
' duration ' : 694 ,
' age_limit ' : 0 ,
} ,
} ,
2015-04-15 09:13:01 -07: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-28 18:57:37 -07: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 11:27:37 -07: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 07:26:42 -07: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 07:11:14 -07:00
# NBC Sports vplayer embed
2015-03-30 12:36:09 -07:00
{
2015-03-31 07:11:14 -07:00
' url ' : ' http://www.riderfans.com/forum/showthread.php?121827-Freeman&s=e98fa1ea6dc08e886b1678d35212494a ' ,
2015-03-30 12:36:09 -07:00
' info_dict ' : {
2015-03-31 07:11:14 -07:00
' id ' : ' ln7x1qSThw4k ' ,
' ext ' : ' flv ' ,
' title ' : " PFT Live: New leader in the ' new-look ' defense " ,
' description ' : ' md5:65a19b4bbfb3b0c0c5768bed1dfad74e ' ,
2015-03-30 12:36:09 -07:00
} ,
2015-04-08 02:26:51 -07:00
} ,
# UDN embed
{
' url ' : ' http://www.udn.com/news/story/7314/822787 ' ,
2015-04-13 22:10:10 -07:00
' md5 ' : ' fd2060e988c326991037b9aff9df21a6 ' ,
2015-04-08 02:26:51 -07:00
' info_dict ' : {
2015-04-13 22:10:10 -07:00
' id ' : ' 300346 ' ,
2015-04-08 02:26:51 -07:00
' ext ' : ' mp4 ' ,
2015-04-13 22:10:10 -07:00
' title ' : ' 中一中男師變性 全校師生力挺 ' ,
2015-04-08 02:26:51 -07:00
' thumbnail ' : ' re:^https?://.* \ .jpg$ ' ,
}
2015-04-13 21:45:43 -07: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 ' ,
' description ' : ' VIDEO: Index/Match versus VLOOKUP. ' ,
' title ' : ' This is what separates the Excel masters from the wannabes ' ,
} ,
' params ' : {
# m3u8 downloads
' skip_download ' : True ,
}
2015-04-16 02:16:11 -07:00
} ,
# Contains a SMIL manifest
{
' url ' : ' http://www.telewebion.com/fa/1263668/ % D9 %82% D8 % B1 % D8 % B9 % D9 %87% E2 %80% 8C % DA % A9 % D8 % B4 % DB % 8C- % D9 %84% DB % 8C % DA % AF- % D9 %82% D9 %87% D8 % B1 % D9 %85% D8 % A7 % D9 %86% D8 % A7 % D9 % 86- % D8 % A7 % D8 % B1 % D9 %88% D9 % BE % D8 % A7/ % 2B- % D9 %81% D9 %88% D8 % AA % D8 % A8 % D8 % A7 % D9 % 84.html ' ,
' info_dict ' : {
' id ' : ' file ' ,
' ext ' : ' flv ' ,
' title ' : ' + Football: Lottery Champions League Europe ' ,
' uploader ' : ' www.telewebion.com ' ,
} ,
' params ' : {
# rtmpe downloads
' skip_download ' : True ,
}
2015-06-07 00:29:17 -07: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 ' ,
} ,
2015-06-21 06:30:34 -07:00
} ,
# Dailymotion Cloud video
{
' url ' : ' http://replay.publicsenat.fr/vod/le-debat/florent-kolandjian,dominique-cena,axel-decourtye,laurence-abeille,bruno-parmentier/175910 ' ,
' md5 ' : ' 49444254273501a64675a7e68c502681 ' ,
' info_dict ' : {
' id ' : ' 5585de919473990de4bee11b ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Le débat ' ,
' thumbnail ' : ' re:^https?://.* \ .jpe?g$ ' ,
}
2015-06-22 00:02:53 -07:00
} ,
2015-06-24 10:23:16 -07: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 05:28:10 -07: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 00:02:53 -07: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-03-02 06:21:11 -08:00
}
2013-07-10 08:49:11 -07:00
]
2013-06-23 11:31:45 -07:00
def report_following_redirect ( self , new_url ) :
""" Report information extraction. """
2014-01-05 16:47:52 -08:00
self . _downloader . to_screen ( ' [redirect] Following redirect to %s ' % new_url )
2013-06-23 11:31:45 -07:00
2014-02-20 04:14:05 -08: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 06:21:11 -08: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 04:14:05 -08:00
return {
' _type ' : ' playlist ' ,
' id ' : url ,
' title ' : playlist_title ,
' description ' : playlist_desc ,
' entries ' : entries ,
}
2014-08-23 17:02:17 -07: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 11:31:45 -07:00
def _real_extract ( self , url ) :
2014-04-29 16:46:06 -07:00
if url . startswith ( ' // ' ) :
return {
' _type ' : ' url ' ,
2014-05-04 18:12:41 -07:00
' url ' : self . http_scheme ( ) + url ,
2014-04-29 16:46:06 -07:00
}
2013-09-06 09:39:35 -07:00
parsed_url = compat_urlparse . urlparse ( url )
if not parsed_url . scheme :
2014-01-22 05:16:43 -08:00
default_search = self . _downloader . params . get ( ' default_search ' )
if default_search is None :
2014-07-29 08:17:43 -07:00
default_search = ' fixup_error '
2014-01-22 05:16:43 -08:00
2014-07-29 08:17:43 -07:00
if default_search in ( ' auto ' , ' auto_warning ' , ' fixup_error ' ) :
2014-01-22 05:16:43 -08: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 08:17:43 -07:00
elif default_search != ' fixup_error ' :
2014-03-30 06:57:31 -07:00
if default_search == ' auto_warning ' :
2014-05-19 08:10:11 -07: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 02:22:44 -07:00
' Falling back to youtube search for %s . Set --default-search " auto " to suppress this warning. ' % url )
2014-01-22 05:16:43 -08:00
return self . url_result ( ' ytsearch: ' + url )
2014-07-29 08:17:43 -07:00
if default_search in ( ' error ' , ' fixup_error ' ) :
2014-07-06 02:22:44 -07:00
raise ExtractorError (
2014-11-23 13:21:46 -08: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 05:16:43 -08:00
else :
2014-10-23 12:13:45 -07:00
if ' : ' not in default_search :
default_search + = ' : '
2014-01-22 05:16:43 -08:00
return self . url_result ( default_search + url )
2014-08-23 19:47:18 -07:00
url , smuggled_data = unsmuggle_url ( url )
force_videoid = None
2014-09-28 03:14:16 -07:00
is_intentional = smuggled_data and smuggled_data . get ( ' to_generic ' )
2014-08-23 19:47:18 -07:00
if smuggled_data and ' force_videoid ' in smuggled_data :
force_videoid = smuggled_data [ ' force_videoid ' ]
video_id = force_videoid
else :
2015-05-30 12:23:58 -07:00
video_id = compat_urllib_parse_unquote ( os . path . splitext ( url . rstrip ( ' / ' ) . split ( ' / ' ) [ - 1 ] ) [ 0 ] )
2013-09-06 09:39:35 -07:00
2014-01-05 16:47:52 -08:00
self . to_screen ( ' %s : Requesting header ' % video_id )
2013-12-26 23:38:42 -08:00
2014-08-23 21:58:11 -07:00
head_req = HEADRequest ( url )
2014-10-26 09:05:44 -07:00
head_response = self . _request_webpage (
2014-08-23 21:58:11 -07:00
head_req , video_id ,
note = False , errnote = ' Could not send HEAD request to %s ' % url ,
fatal = False )
2013-12-17 03:33:55 -08:00
2014-10-26 09:05:44 -07:00
if head_response is not False :
2013-12-17 03:33:55 -08:00
# Check for redirect
2014-10-26 09:05:44 -07:00
new_url = head_response . geturl ( )
2013-12-17 03:33:55 -08:00
if url != new_url :
self . report_following_redirect ( new_url )
2014-08-23 19:47:18 -07:00
if force_videoid :
new_url = smuggle_url (
new_url , { ' force_videoid ' : force_videoid } )
2013-12-17 03:04:33 -08:00
return self . url_result ( new_url )
2013-12-17 03:33:55 -08:00
2014-10-26 09:05:44 -07:00
full_response = None
if head_response is False :
2015-05-30 11:44:54 -07:00
request = compat_urllib_request . Request ( url )
request . add_header ( ' Accept-Encoding ' , ' * ' )
full_response = self . _request_webpage ( request , video_id )
2014-10-26 09:05:44 -07:00
head_response = full_response
# Check for direct link to a video
content_type = head_response . headers . get ( ' Content-Type ' , ' ' )
m = re . match ( r ' ^(?P<type>audio|video|application(?=/ogg$))/(?P<format_id>.+)$ ' , content_type )
if m :
upload_date = unified_strdate (
head_response . headers . get ( ' Last-Modified ' ) )
return {
' id ' : video_id ,
2015-05-30 12:23:58 -07:00
' title ' : compat_urllib_parse_unquote ( os . path . splitext ( url_basename ( url ) ) [ 0 ] ) ,
2014-11-01 04:04:15 -07:00
' direct ' : True ,
2014-10-26 09:05:44 -07:00
' formats ' : [ {
' format_id ' : m . group ( ' format_id ' ) ,
' url ' : url ,
' vcodec ' : ' none ' if m . group ( ' type ' ) == ' audio ' else None
} ] ,
' upload_date ' : upload_date ,
}
2013-12-17 03:33:55 -08:00
2014-09-28 03:14:16 -07:00
if not self . _downloader . params . get ( ' test ' , False ) and not is_intentional :
2015-06-23 14:08:24 -07: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 03:14:16 -07:00
2014-11-26 01:44:39 -08:00
if not full_response :
2015-05-30 11:44:54 -07:00
request = compat_urllib_request . Request ( url )
# 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 01:44:39 -08:00
# Maybe it's a direct link to a video?
# Be careful not to download the whole thing!
first_bytes = full_response . read ( 512 )
2015-01-22 16:21:30 -08:00
if not is_html ( first_bytes ) :
2014-11-26 01:44:39 -08:00
self . _downloader . report_warning (
' URL could be a direct video link, returning it as such. ' )
upload_date = unified_strdate (
head_response . headers . get ( ' Last-Modified ' ) )
return {
' id ' : video_id ,
2015-05-30 12:23:58 -07:00
' title ' : compat_urllib_parse_unquote ( os . path . splitext ( url_basename ( url ) ) [ 0 ] ) ,
2014-11-26 01:44:39 -08:00
' direct ' : True ,
' url ' : url ,
' upload_date ' : upload_date ,
}
webpage = self . _webpage_read_content (
full_response , url , video_id , prefix = first_bytes )
2013-06-23 11:31:45 -07:00
self . report_extraction ( video_id )
2013-11-18 04:28:26 -08:00
2014-02-20 04:14:05 -08:00
# Is it an RSS feed?
try :
2014-03-10 09:31:32 -07:00
doc = parse_xml ( webpage )
2014-02-20 04:14:05 -08:00
if doc . tag == ' rss ' :
return self . _extract_rss ( url , video_id , doc )
2014-02-21 07:59:10 -08:00
except compat_xml_parse_error :
2014-02-20 04:14:05 -08:00
pass
2014-08-23 17:02:17 -07: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 14:38:49 -07:00
# Sometimes embedded video player is hidden behind percent encoding
# (e.g. https://github.com/rg3/youtube-dl/issues/2448)
# Unescaping the whole page allows to handle those cases in a generic way
2015-07-15 13:30:47 -07:00
webpage = compat_urllib_parse_unquote ( webpage )
2014-02-24 08:44:31 -08:00
2013-11-18 04:28:26 -08:00
# it's tempting to parse this further, but you would
# have to take into account all the variations like
# Video Title - Site Name
# Site Name | Video Title
# Video Title - Tagline | Site Name
# and so on and so forth; it's just not practical
2013-12-06 00:15:04 -08:00
video_title = self . _html_search_regex (
2014-01-05 16:47:52 -08:00
r ' (?s)<title>(.*?)</title> ' , webpage , ' video title ' ,
default = ' video ' )
2013-12-06 00:15:04 -08:00
2014-08-23 20:31:32 -07: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 00:15:04 -08:00
# video uploader is domain name
video_uploader = self . _search_regex (
2014-01-05 16:47:52 -08:00
r ' ^(?:https?://)?([^/]*)/.* ' , url , ' video uploader ' )
2013-11-18 04:28:26 -08:00
2014-08-22 09:19:56 -07:00
# Helper method
2015-01-02 06:54:30 -08:00
def _playlist_from_matches ( matches , getter = None , ie = None ) :
2014-09-24 02:05:14 -07:00
urlrs = orderedSet (
2015-01-02 06:54:30 -08:00
self . url_result ( self . _proto_relative_url ( getter ( m ) if getter else m ) , ie )
2014-09-24 02:05:14 -07:00
for m in matches )
2014-08-22 09:19:56 -07:00
return self . playlist_result (
urlrs , playlist_id = video_id , playlist_title = video_title )
2013-08-26 12:29:31 -07:00
# Look for BrightCove:
2014-02-03 06:19:40 -08:00
bc_urls = BrightcoveIE . _extract_brightcove_urls ( webpage )
if bc_urls :
2014-01-05 16:47:52 -08:00
self . to_screen ( ' Brightcove video detected. ' )
2014-02-03 06:19:40 -08:00
entries = [ {
' _type ' : ' url ' ,
' url ' : smuggle_url ( bc_url , { ' Referer ' : url } ) ,
' ie_key ' : ' Brightcove '
} for bc_url in bc_urls ]
return {
' _type ' : ' playlist ' ,
' title ' : video_title ,
' id ' : video_id ,
' entries ' : entries ,
}
2013-07-10 08:49:11 -07:00
2015-02-16 06:45:01 -08:00
# Look for embedded rtl.nl player
matches = re . findall (
2015-06-11 06:04:12 -07:00
r ' <iframe[^>]+?src= " ((?:https?:)?//(?:www \ .)?rtl \ .nl/system/videoplayer/[^ " ]+(?:video_)?embed[^ " ]+) " ' ,
2015-02-16 06:45:01 -08:00
webpage )
if matches :
return _playlist_from_matches ( matches , ie = ' RtlNl ' )
2015-06-21 03:23:58 -07:00
vimeo_url = VimeoIE . _extract_vimeo_url ( url , webpage )
if vimeo_url is not None :
return self . url_result ( vimeo_url )
2013-12-21 18:34:13 -08:00
2015-07-20 03:49:53 -07: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 02:44:57 -07:00
# Look for embedded YouTube player
2013-12-19 11:44:30 -08:00
matches = re . findall ( r ''' (?x)
2014-06-09 13:06:45 -07:00
( ? :
< iframe [ ^ > ] + ? src = |
2014-08-22 08:40:36 -07:00
data - video - url = |
2014-06-09 13:06:45 -07:00
< embed [ ^ > ] + ? src = |
2014-10-26 06:15:48 -07:00
embedSWF \( ? : \s * |
new \s + SWFObject \(
2014-06-09 13:06:45 -07:00
)
( [ " \' ])
2014-09-10 04:29:20 -07:00
( ? P < url > ( ? : https ? : ) ? / / ( ? : www \. ) ? youtube ( ? : - nocookie ) ? \. com /
2014-09-24 16:58:49 -07:00
( ? : embed | v | p ) / . + ? )
2013-12-19 11:44:30 -08:00
\1 ''' , webpage)
2013-11-18 04:28:26 -08:00
if matches :
2014-08-22 09:19:56 -07:00
return _playlist_from_matches (
2014-09-24 02:05:14 -07:00
matches , lambda m : unescapeHTML ( m [ 1 ] ) )
2013-10-18 02:44:57 -07:00
2014-11-25 05:34:19 -08: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 ) )
2013-11-30 16:21:33 -08:00
# Look for embedded Dailymotion player
matches = re . findall (
2013-12-06 00:15:04 -08:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>(?:https?:)?//(?:www \ .)?dailymotion \ .com/embed/video/.+?) \ 1 ' , webpage )
2013-11-30 16:21:33 -08:00
if matches :
2014-08-22 09:19:56 -07:00
return _playlist_from_matches (
matches , lambda m : unescapeHTML ( m [ 1 ] ) )
2013-11-30 16:21:33 -08:00
2014-10-02 11:42:45 -07: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 00:15:04 -08:00
# Look for embedded Wistia player
match = re . search (
2014-10-23 09:03:07 -07:00
r ' <(?:meta[^>]+?content|iframe[^>]+?src)=([ " \' ])(?P<url>(?:https?:)?//(?:fast \ .)?wistia \ .net/embed/iframe/.+?) \ 1 ' , webpage )
2013-12-06 00:15:04 -08:00
if match :
2014-10-17 15:52:55 -07:00
embed_url = self . _proto_relative_url (
unescapeHTML ( match . group ( ' url ' ) ) )
2013-12-06 00:15:04 -08:00
return {
' _type ' : ' url_transparent ' ,
2014-10-17 15:52:55 -07:00
' url ' : embed_url ,
2013-12-06 00:15:04 -08:00
' ie_key ' : ' Wistia ' ,
' uploader ' : video_uploader ,
' title ' : video_title ,
' id ' : video_id ,
}
2014-11-23 11:41:03 -08:00
2014-10-17 15:52:55 -07:00
match = re . search ( r ' (?:id=[ " \' ]wistia_|data-wistia-?id=[ " \' ]|Wistia \ .embed \ ([ " \' ])(?P<id>[^ " \' ]+) ' , webpage )
2014-09-19 17:02:11 -07:00
if match :
return {
' _type ' : ' url_transparent ' ,
' url ' : ' http://fast.wistia.net/embed/iframe/ {0:} ' . format ( match . group ( ' id ' ) ) ,
' ie_key ' : ' Wistia ' ,
' uploader ' : video_uploader ,
' title ' : video_title ,
' id ' : match . group ( ' id ' )
}
2013-12-06 00:15:04 -08:00
2013-12-16 11:08:23 -08:00
# Look for embedded blip.tv player
2015-04-20 22:48:02 -07:00
bliptv_url = BlipTVIE . _extract_url ( webpage )
if bliptv_url :
return self . url_result ( bliptv_url , ' BlipTV ' )
2013-12-16 11:08:23 -08:00
2015-05-08 11:23:35 -07:00
# Look for SVT player
svt_url = SVTIE . _extract_url ( webpage )
if svt_url :
return self . url_result ( svt_url , ' SVT ' )
2014-04-20 20:47:52 -07: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 06:40:25 -07: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 07:05:14 -08:00
# Don't set the extractor because it can be a track url or an album
return self . url_result ( burl )
2013-10-27 06:40:25 -07:00
2013-12-16 12:45:21 -08: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-27 21:08:19 -08:00
# Look for embedded Viddler player
2015-01-28 09:07:37 -08:00
mobj = re . search (
r ' <(?:iframe[^>]+?src|param[^>]+?value)=([ " \' ])(?P<url>(?:https?:)?//(?:www \ .)?viddler \ .com/(?:embed|player)/.+?) \ 1 ' ,
webpage )
2015-01-27 21:08:19 -08:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) )
2013-12-16 12:45:21 -08:00
2015-03-19 08:26:57 -07: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-21 19:18:13 -07: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 11:28:52 -08:00
# Look for Ooyala videos
2015-01-28 09:07:37 -08:00
mobj = ( re . search ( r ' player \ .ooyala \ .com/[^ " ?]+ \ ?[^ " ]*?(?:embedCode|ec)=(?P<ec>[^ " &]+) ' , webpage ) or
2015-02-04 05:33:37 -08:00
re . search ( r ' OO \ .Player \ .create \ ([ \' " ].*?[ \' " ], \ s*[ \' " ](?P<ec>. {32} )[ \' " ] ' , webpage ) or
2015-04-13 21:45:43 -07: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 11:28:52 -08:00
if mobj is not None :
2014-03-21 13:51:33 -07:00
return OoyalaIE . _build_url_result ( mobj . group ( ' ec ' ) )
2013-12-19 11:28:52 -08:00
2015-02-04 05:33:37 -08: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 (
embeds , getter = lambda v : OoyalaIE . _url_for_embed_code ( v [ ' provider_video_id ' ] ) , ie = ' Ooyala ' )
2013-12-20 08:05:28 -08:00
# Look for Aparat videos
2014-04-21 03:37:41 -07:00
mobj = re . search ( r ' <iframe .*?src= " (http://www \ .aparat \ .com/video/[^ " ]+) " ' , webpage )
2013-12-20 08:05:28 -08:00
if mobj is not None :
return self . url_result ( mobj . group ( 1 ) , ' Aparat ' )
2014-01-06 23:07:46 -08:00
# Look for MPORA videos
2014-01-29 13:26:46 -08:00
mobj = re . search ( r ' <iframe .*?src= " (http://mpora \ .(?:com|de)/videos/[^ " ]+) " ' , webpage )
2014-01-06 23:07:46 -08:00
if mobj is not None :
return self . url_result ( mobj . group ( 1 ) , ' Mpora ' )
2014-01-07 17:11:46 -08:00
2014-04-05 03:20:05 -07:00
# Look for embedded NovaMov-based player
2014-01-07 17:07:11 -08:00
mobj = re . search (
2014-05-17 04:12:12 -07:00
r ''' (?x)<(?:pagespeed_)?iframe[^>]+?src=([ " \ ' ])
2014-04-05 03:20:05 -07:00
( ? P < url > http : / / ( ? : ( ? : embed | www ) \. ) ?
( ? : novamov \. com |
nowvideo \. ( ? : ch | sx | eu | at | ag | co ) |
videoweed \. ( ? : es | com ) |
movshare \. ( ? : net | sx | ag ) |
divxstage \. ( ? : eu | net | ch | co | at | ag ) )
/ embed \. php . + ? ) \1 ''' , webpage)
2014-01-07 17:07:11 -08:00
if mobj is not None :
2014-04-05 03:20:05 -07:00
return self . url_result ( mobj . group ( ' url ' ) )
2014-04-05 01:49:45 -07:00
2014-01-21 09:10:14 -08:00
# Look for embedded Facebook player
mobj = re . search (
2014-01-26 20:47:30 -08:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https://www \ .facebook \ .com/video/embed.+?) \ 1 ' , webpage )
2014-01-21 09:10:14 -08:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Facebook ' )
2014-02-28 08:51:54 -08:00
# Look for embedded VK player
mobj = re . search ( r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://vk \ .com/video_ext \ .php.+?) \ 1 ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' VK ' )
2014-06-29 06:18:23 -07:00
# Look for embedded ivi player
mobj = re . search ( r ' <embed[^>]+?src=([ " \' ])(?P<url>https?://(?:www \ .)?ivi \ .ru/video/player.+?) \ 1 ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Ivi ' )
2014-01-26 20:47:30 -08:00
# Look for embedded Huffington Post player
mobj = re . search (
2014-01-29 13:26:46 -08:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://embed \ .live \ .huffingtonpost \ .com/.+?) \ 1 ' , webpage )
2014-01-26 20:47:30 -08:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' HuffPost ' )
2014-02-23 16:15:51 -08: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 10:39:32 -07:00
return self . url_result ( compat_urllib_parse_unquote ( mobj . group ( ' url ' ) ) )
2014-02-23 16:15:51 -08:00
2014-03-11 08:51:36 -07:00
# Look for funnyordie embed
matches = re . findall ( r ' <iframe[^>]+?src= " (https?://(?:www \ .)?funnyordie \ .com/embed/[^ " ]+) " ' , webpage )
if matches :
2014-08-22 09:19:56 -07:00
return _playlist_from_matches (
matches , getter = unescapeHTML , ie = ' FunnyOrDie ' )
2014-03-11 08:51:36 -07:00
2015-01-02 06:46:17 -08: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 06:55:09 -08:00
return _playlist_from_matches ( matches , ie = ' BBCCoUk ' )
2015-01-02 06:46:17 -08:00
2014-03-16 12:00:31 -07:00
# Look for embedded RUTV player
rutv_url = RUTVIE . _extract_url ( webpage )
if rutv_url :
return self . url_result ( rutv_url , ' RUTV ' )
2015-06-12 03:22:46 -07:00
# Look for embedded TVC player
2015-06-12 05:15:30 -07:00
tvc_url = TVCIE . _extract_url ( webpage )
if tvc_url :
return self . url_result ( tvc_url , ' TVC ' )
2015-06-12 03:22:46 -07:00
2015-05-15 10:09:34 -07: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 14:39:14 -07:00
# Look for embedded PornHub player
2015-06-12 14:36:16 -07:00
pornhub_url = PornHubIE . _extract_url ( webpage )
if pornhub_url :
return self . url_result ( pornhub_url , ' PornHub ' )
2015-06-21 10:11:25 -07:00
# Look for embedded XHamster player
xhamster_urls = XHamsterEmbedIE . _extract_urls ( webpage )
if xhamster_urls :
return _playlist_from_matches ( xhamster_urls , ie = ' XHamsterEmbed ' )
2015-06-12 05:37:09 -07: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 02:20:44 -07:00
# Look for embedded TED player
mobj = re . search (
2015-01-05 09:16:47 -08:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://embed(?:-ssl)? \ .ted \ .com/.+?) \ 1 ' , webpage )
2014-03-22 02:20:44 -07:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' TED ' )
2014-04-04 07:23:09 -07: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 14:01:47 -07:00
# Look for embedded arte.tv player
mobj = re . search (
r ' <script [^>]*?src= " (?P<url>http://www \ .arte \ .tv/playerv2/embed[^ " ]+) " ' ,
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' ArteTVEmbed ' )
2015-07-18 09:56:00 -07: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 05:58:49 -07:00
# Look for embedded smotri.com player
smotri_url = SmotriIE . _extract_url ( webpage )
if smotri_url :
return self . url_result ( smotri_url , ' Smotri ' )
2015-07-09 11:25:36 -07:00
# Look for embedded Myvi.ru player
2015-07-09 11:27:44 -07:00
myvi_url = MyviIE . _extract_url ( webpage )
2015-07-09 11:25:36 -07:00
if myvi_url :
return self . url_result ( myvi_url )
2014-05-04 18:12:41 -07:00
# Look for embeded soundcloud player
mobj = re . search (
2014-10-29 12:27:58 -07:00
r ' <iframe \ s+(?:[a-zA-Z0-9_-]+= " [^ " ]+ " \ s+)*src= " (?P<url>https?://(?:w \ .)?soundcloud \ .com/player[^ " ]+) " ' ,
2014-05-04 18:12:41 -07:00
webpage )
if mobj is not None :
url = unescapeHTML ( mobj . group ( ' url ' ) )
return self . url_result ( url )
2014-06-09 14:06:25 -07:00
# Look for embedded vulture.com player
mobj = re . search (
r ' <iframe src= " (?P<url>https?://video \ .vulture \ .com/[^ " ]+) " ' ,
webpage )
if mobj is not None :
url = unescapeHTML ( mobj . group ( ' url ' ) )
return self . url_result ( url , ie = ' Vulture ' )
2014-06-22 12:38:04 -07:00
# Look for embedded mtvservices player
mobj = re . search (
r ' <iframe src= " (?P<url>https?://media \ .mtvnservices \ .com/embed/[^ " ]+) " ' ,
webpage )
if mobj is not None :
url = unescapeHTML ( mobj . group ( ' url ' ) )
return self . url_result ( url , ie = ' MTVServicesEmbedded ' )
2014-08-15 23:56:22 -07: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 06:20:49 -07:00
# Look for embedded sbs.com.au player
mobj = re . search (
2015-02-09 05:46:10 -08: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 06:20:49 -07:00
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' SBS ' )
2014-12-11 17:57:36 -08: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 06:19:28 -07:00
mobj = re . search (
2014-10-31 14:01:58 -07:00
r ' <iframe[^>]+?src=([ " \' ])(?P<url>https?://m(?:lb)? \ .mlb \ .com/shared/video/embed/embed \ .html \ ?.+?) \ 1 ' ,
2014-09-02 06:19:28 -07:00
webpage )
2015-05-03 11:20:07 -07:00
if not mobj :
mobj = re . search (
r ' data-video-link=[ " \' ](?P<url>http://m.mlb.com/video/[^ " \' ]+) ' ,
webpage )
2014-09-02 06:19:28 -07:00
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' MLB ' )
2014-10-13 05:59:35 -07:00
mobj = re . search (
r ' <iframe[^>]+?src=([ " \' ])(?P<url> %s ) \ 1 ' % CondeNastIE . EMBED_URL ,
webpage )
if mobj is not None :
return self . url_result ( self . _proto_relative_url ( mobj . group ( ' url ' ) , scheme = ' http: ' ) , ' CondeNast ' )
2014-11-13 07:12:51 -08:00
mobj = re . search (
r ' <iframe[^>]+src= " (?P<url>https?://new \ .livestream \ .com/[^ " ]+/player[^ " ]+) " ' ,
webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Livestream ' )
2015-02-21 11:39:26 -08: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 14:34:19 -08:00
# Look for Kaltura embeds
2015-07-02 08:39:46 -07:00
mobj = ( re . search ( r " (?s)kWidget \ .(?:thumb)?[Ee]mbed \ ( \ { .*? ' wid ' \ s*: \ s* ' _?(?P<partner_id>[^ ' ]+) ' ,.*? ' entry_id ' \ s*: \ s* ' (?P<id>[^ ' ]+) ' , " , webpage ) or
re . search ( r ' (?s)([ " \' ])(?:https?:)?//cdnapisec \ .kaltura \ .com/.*?(?:p|partner_id)/(?P<partner_id> \ d+).*? \ 1.*?entry_id \ s*: \ s*([ " \' ])(?P<id>[^ \ 2]+?) \ 2 ' , webpage ) )
2015-02-26 14:34:19 -08:00
if mobj is not None :
return self . url_result ( ' kaltura: %(partner_id)s : %(id)s ' % mobj . groupdict ( ) , ' Kaltura ' )
2015-03-07 08:22:57 -08:00
# Look for Eagle.Platform embeds
mobj = re . search (
r ' <iframe[^>]+src= " (?P<url>https?://.+? \ .media \ .eagleplatform \ .com/index/player \ ?.+?) " ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' EaglePlatform ' )
2015-03-07 08:34:44 -08: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 05:07:10 -07:00
# Look for Pladform embeds
mobj = re . search (
r ' <iframe[^>]+src= " (?P<url>https?://out \ .pladform \ .ru/player \ ?.+?) " ' , webpage )
if mobj is not None :
return self . url_result ( mobj . group ( ' url ' ) , ' Pladform ' )
2015-04-15 09:10:08 -07: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-28 18:57:37 -07: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 07:20:20 -07: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 12:36:09 -07: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 ' )
2015-06-29 00:01:30 -07:00
# Look for Google Drive embeds
google_drive_url = GoogleDriveEmbedIE . _extract_url ( webpage )
if google_drive_url :
return self . url_result ( google_drive_url , ' GoogleDrive ' )
2015-04-08 02:26:51 -07:00
# Look for UDN embeds
mobj = re . search (
r ' <iframe[^>]+src= " (?P<url> %s ) " ' % UDNEmbedIE . _VALID_URL , webpage )
if mobj is not None :
return self . url_result (
2015-04-08 06:39:34 -07:00
compat_urlparse . urljoin ( url , mobj . group ( ' url ' ) ) , ' UDNEmbed ' )
2015-04-08 02:26:51 -07:00
2015-04-20 12:18:38 -07:00
# Look for Senate ISVP iframe
senate_isvp_url = SenateISVPIE . _search_iframe_url ( webpage )
if senate_isvp_url :
2015-05-15 10:23:51 -07:00
return self . url_result ( senate_isvp_url , ' SenateISVP ' )
2015-04-20 12:18:38 -07:00
2015-06-21 06:30:34 -07: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 10:19:50 -07:00
# Look for OnionStudios embeds
onionstudios_url = OnionStudiosIE . _extract_url ( webpage )
if onionstudios_url :
return self . url_result ( onionstudios_url )
2015-06-27 05:26:14 -07:00
# Look for SnagFilms embeds
snagfilms_url = SnagFilmsEmbedIE . _extract_url ( webpage )
if snagfilms_url :
return self . url_result ( snagfilms_url )
2015-06-22 00:02:53 -07: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 ' )
2014-10-09 05:26:23 -07:00
def check_video ( vurl ) :
2015-02-09 01:42:25 -08:00
if YoutubeIE . suitable ( vurl ) :
return True
2014-10-09 05:26:23 -07: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 11:31:45 -07:00
# Start with something easy: JW Player in SWFObject
2014-10-09 05:26:23 -07:00
found = filter_video ( re . findall ( r ' flashvars: [ \' " ](?:.*&)?file=(http[^ \' " &]*) ' , webpage ) )
2014-04-29 17:23:51 -07:00
if not found :
2014-01-04 20:34:06 -08:00
# Look for gorilla-vid style embedding
2014-10-09 05:26:23 -07:00
found = filter_video ( re . findall ( r ''' (?sx)
2014-04-21 07:16:53 -07:00
( ? :
jw_plugins |
JWPlayerOptions |
jwplayer \s * \( \s * [ " ' ][^ ' " ] + [ " ' ] \ s* \ ) \ s* \ .setup
)
2015-02-09 01:42:25 -08:00
. * ?
[ ' " ]?file[ ' " ]? \ s*: \ s*[ " \' ](.*?)[ " \' ] ' ' ' , webpage ) )
2014-04-29 17:23:51 -07:00
if not found :
2013-06-23 11:31:45 -07:00
# Broaden the search a little bit
2014-10-09 05:26:23 -07:00
found = filter_video ( re . findall ( r ' [^A-Za-z0-9]?(?:file|source)=(http[^ \' " &]*) ' , webpage ) )
2014-04-29 17:23:51 -07:00
if not found :
# Broaden the findall a little bit: JWPlayer JS loader
2014-10-09 05:26:23 -07:00
found = filter_video ( re . findall (
r ' [^A-Za-z0-9]?file[ " \' ]?: \ s*[ " \' ](http(?![^ \' " ]+ \ .[0-9]+[ \' " ])[^ \' " ]+)[ " \' ] ' , webpage ) )
2014-08-23 20:31:32 -07:00
if not found :
# Flow player
2014-10-09 05:26:23 -07:00
found = filter_video ( re . findall ( r ''' (?xs)
2014-08-23 20:31:32 -07:00
flowplayer \( " [^ " ] + " , \ s*
\{ [ ^ } ] + ? \} \s * ,
2015-01-29 23:41:40 -08:00
\s * \{ [ ^ } ] + ? [ " ' ]?clip[ " ' ]? \ s*: \ s* \ { \ s*
2014-08-23 20:31:32 -07:00
[ " ' ]?url[ " ' ]? \ s*: \ s*[ " ' ] ( [ ^ " ' ]+)[ " ' ]
2014-10-09 05:26:23 -07:00
''' , webpage))
2015-01-23 03:00:25 -08:00
if not found :
# Cinerama player
found = re . findall (
r " cinerama \ .embedPlayer \ ( \ s* \ ' [^ ' ]+ \ ' , \ s* ' ([^ ' ]+) ' " , webpage )
2014-04-29 17:23:51 -07:00
if not found :
2013-06-23 11:31:45 -07:00
# Try to find twitter cards info
2014-10-09 05:26:23 -07:00
found = filter_video ( re . findall (
r ' <meta (?:property|name)= " twitter:player:stream " (?:content|value)= " (.+?) " ' , webpage ) )
2014-04-29 17:23:51 -07:00
if not found :
2013-06-23 11:31:45 -07: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-29 17:23:51 -07:00
m_video_type = re . findall ( r ' <meta.*?property= " og:video:type " .*?content= " video/(.*?) " ' , webpage )
2013-06-23 11:31:45 -07: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 05:26:23 -07:00
found = filter_video ( re . findall ( r ' <meta.*?property= " og:video " .*?content= " (.*?) " ' , webpage ) )
2014-04-29 17:23:51 -07:00
if not found :
2013-08-20 19:32:22 -07:00
# HTML5 video
2014-11-21 13:01:25 -08:00
found = re . findall ( r ' (?s)<video[^<]*(?:>.*?<source[^>]*)? \ s+src=[ " \' ](.*?)[ " \' ] ' , webpage )
2014-04-29 17:23:51 -07:00
if not found :
2015-03-17 11:05:40 -07:00
REDIRECT_REGEX = r ' [0-9] { ,2}; \ s*(?:URL|url)= \' ?([^ \' " ]+) '
2014-05-16 06:32:53 -07:00
found = re . search (
2014-02-26 22:21:59 -08:00
r ' (?i)<meta \ s+(?=(?:[a-z-]+= " [^ " ]+ " \ s+)*http-equiv= " refresh " ) '
2015-03-17 11:05:40 -07:00
r ' (?:[a-z-]+= " [^ " ]+ " \ s+)*?content= " %s ' % REDIRECT_REGEX ,
2014-02-26 22:21:59 -08:00
webpage )
2015-03-17 10:51:40 -07:00
if not found :
# Look also in Refresh HTTP header
refresh_header = head_response . headers . get ( ' Refresh ' )
if refresh_header :
2015-03-17 11:05:40 -07:00
found = re . search ( REDIRECT_REGEX , refresh_header )
2014-04-29 17:23:51 -07:00
if found :
2015-07-22 14:33:49 -07:00
new_url = compat_urlparse . urljoin ( url , unescapeHTML ( found . group ( 1 ) ) )
2014-02-26 22:21:59 -08:00
self . report_following_redirect ( new_url )
return {
' _type ' : ' url ' ,
' url ' : new_url ,
}
2014-04-29 17:23:51 -07:00
if not found :
2014-12-30 10:35:35 -08:00
raise UnsupportedError ( url )
2013-06-23 11:31:45 -07:00
2014-04-29 17:23:51 -07:00
entries = [ ]
for video_url in found :
video_url = compat_urlparse . urljoin ( url , video_url )
2015-07-17 10:39:32 -07:00
video_id = compat_urllib_parse_unquote ( os . path . basename ( video_url ) )
2013-06-23 11:31:45 -07:00
2014-04-29 17:23:51 -07: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 11:31:45 -07:00
2014-04-29 17:23:51 -07:00
# here's a fun little line of code for you:
video_id = os . path . splitext ( video_id ) [ 0 ]
2014-01-05 16:42:58 -08:00
2015-04-16 02:16:11 -07:00
if determine_ext ( video_url ) == ' smil ' :
entries . append ( {
' id ' : video_id ,
' formats ' : self . _extract_smil_formats ( video_url , video_id ) ,
' uploader ' : video_uploader ,
' title ' : video_title ,
' age_limit ' : age_limit ,
} )
else :
entries . append ( {
' id ' : video_id ,
' url ' : video_url ,
' uploader ' : video_uploader ,
' title ' : video_title ,
' age_limit ' : age_limit ,
} )
2014-04-29 17:23:51 -07:00
if len ( entries ) == 1 :
2014-05-01 02:28:37 -07:00
return entries [ 0 ]
2014-04-29 17:23:51 -07:00
else :
for num , e in enumerate ( entries , start = 1 ) :
2015-02-25 08:56:51 -08:00
# 'url' results don't have a title
if e . get ( ' title ' ) is not None :
e [ ' title ' ] = ' %s ( %d ) ' % ( e [ ' title ' ] , num )
2014-04-29 17:23:51 -07:00
return {
' _type ' : ' playlist ' ,
' entries ' : entries ,
}