2013-08-29 13:51:38 -04:00
# coding: utf-8
2014-01-06 11:15:27 -05:00
from __future__ import unicode_literals
2013-08-29 13:51:38 -04:00
2014-01-06 11:15:27 -05:00
import re
2014-08-09 14:21:16 -04:00
import calendar
import datetime
2013-08-29 13:16:07 -04:00
from . common import InfoExtractor
2016-10-14 12:43:09 -04:00
from . . compat import compat_str
2013-08-29 13:16:07 -04:00
from . . utils import (
2014-01-06 11:15:27 -05:00
HEADRequest ,
unified_strdate ,
2015-03-07 06:31:03 -05:00
strip_jsonp ,
int_or_none ,
float_or_none ,
determine_ext ,
remove_end ,
2016-10-14 12:43:09 -04:00
unescapeHTML ,
2013-08-29 13:16:07 -04:00
)
2014-01-06 11:15:27 -05:00
2014-08-09 14:21:16 -04:00
class ORFTVthekIE ( InfoExtractor ) :
IE_NAME = ' orf:tvthek '
IE_DESC = ' ORF TVthek '
2016-10-14 12:43:09 -04:00
_VALID_URL = r ' https?://tvthek \ .orf \ .at/(?:[^/]+/)+(?P<id> \ d+) '
2014-01-06 11:15:27 -05:00
2014-12-16 10:45:28 -05:00
_TESTS = [ {
2014-12-13 06:41:31 -05:00
' url ' : ' http://tvthek.orf.at/program/Aufgetischt/2745173/Aufgetischt-Mit-der-Steirischen-Tafelrunde/8891389 ' ,
' playlist ' : [ {
' md5 ' : ' 2942210346ed779588f428a92db88712 ' ,
' info_dict ' : {
' id ' : ' 8896777 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Aufgetischt: Mit der Steirischen Tafelrunde ' ,
' description ' : ' md5:c1272f0245537812d4e36419c207b67d ' ,
' duration ' : 2668 ,
' upload_date ' : ' 20141208 ' ,
} ,
} ] ,
2014-12-16 10:45:28 -05:00
' skip ' : ' Blocked outside of Austria / Germany ' ,
} , {
' url ' : ' http://tvthek.orf.at/topic/Im-Wandel-der-Zeit/8002126/Best-of-Ingrid-Thurnher/7982256 ' ,
2016-07-07 17:39:39 -04:00
' info_dict ' : {
' id ' : ' 7982259 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Best of Ingrid Thurnher ' ,
' upload_date ' : ' 20140527 ' ,
' description ' : ' Viele Jahre war Ingrid Thurnher das " Gesicht " der ZIB 2. Vor ihrem Wechsel zur ZIB 2 im Jahr 1995 moderierte sie unter anderem " Land und Leute " , " Österreich-Bild " und " Niederösterreich heute " . ' ,
} ,
' params ' : {
' skip_download ' : True , # rtsp downloads
} ,
2014-12-16 10:45:28 -05:00
' _skip ' : ' Blocked outside of Austria / Germany ' ,
2016-10-14 12:43:09 -04:00
} , {
' url ' : ' http://tvthek.orf.at/topic/Fluechtlingskrise/10463081/Heimat-Fremde-Heimat/13879132/Senioren-betreuen-Migrantenkinder/13879141 ' ,
' skip_download ' : True ,
} , {
' url ' : ' http://tvthek.orf.at/profile/Universum/35429 ' ,
' skip_download ' : True ,
2014-12-16 10:45:28 -05:00
} ]
2013-08-29 13:16:07 -04:00
def _real_extract ( self , url ) :
2014-12-13 06:41:31 -05:00
playlist_id = self . _match_id ( url )
2013-08-29 13:16:07 -04:00
webpage = self . _download_webpage ( url , playlist_id )
2016-10-14 12:43:09 -04:00
data_jsb = self . _parse_json (
self . _search_regex (
r ' <div[^>]+class=([ " \' ]).*?VideoPlaylist.*? \ 1[^>]+data-jsb=([ " \' ])(?P<json>.+?) \ 2 ' ,
webpage , ' playlist ' , group = ' json ' ) ,
playlist_id , transform_source = unescapeHTML ) [ ' playlist ' ] [ ' videos ' ]
2014-01-06 11:15:27 -05:00
def quality_to_int ( s ) :
m = re . search ( ' ([0-9]+) ' , s )
if m is None :
return - 1
return int ( m . group ( 1 ) )
entries = [ ]
2016-10-14 12:43:09 -04:00
for sd in data_jsb :
video_id , title = sd . get ( ' id ' ) , sd . get ( ' title ' )
if not video_id or not title :
continue
video_id = compat_str ( video_id )
2014-01-06 11:15:27 -05:00
formats = [ {
' preference ' : - 10 if fd [ ' delivery ' ] == ' hls ' else None ,
' format_id ' : ' %s - %s - %s ' % (
fd [ ' delivery ' ] , fd [ ' quality ' ] , fd [ ' quality_string ' ] ) ,
' url ' : fd [ ' src ' ] ,
' protocol ' : fd [ ' protocol ' ] ,
' quality ' : quality_to_int ( fd [ ' quality ' ] ) ,
2016-10-14 12:43:09 -04:00
} for fd in sd [ ' sources ' ] ]
2014-01-06 11:15:27 -05:00
# Check for geoblocking.
# There is a property is_geoprotection, but that's always false
geo_str = sd . get ( ' geoprotection_string ' )
if geo_str :
try :
http_url = next (
f [ ' url ' ]
for f in formats
if re . match ( r ' ^https?://.* \ .mp4$ ' , f [ ' url ' ] ) )
except StopIteration :
pass
else :
req = HEADRequest ( http_url )
2014-01-06 23:51:46 -05:00
self . _request_webpage (
2014-01-06 11:15:27 -05:00
req , video_id ,
note = ' Testing for geoblocking ' ,
errnote = ( (
' This video seems to be blocked outside of %s . '
' You may want to try the streaming-* formats. ' )
% geo_str ) ,
fatal = False )
2016-02-16 11:23:38 -05:00
self . _check_formats ( formats , video_id )
2014-01-06 11:15:27 -05:00
self . _sort_formats ( formats )
2016-10-19 06:34:15 -04:00
subtitles = { }
for sub in sd . get ( ' subtitles ' , [ ] ) :
sub_src = sub . get ( ' src ' )
if not sub_src :
continue
subtitles . setdefault ( sub . get ( ' lang ' , ' de-AT ' ) , [ ] ) . append ( {
' url ' : sub_src ,
} )
2016-10-14 12:43:09 -04:00
upload_date = unified_strdate ( sd . get ( ' created_date ' ) )
2014-01-06 11:15:27 -05:00
entries . append ( {
2013-08-29 13:16:07 -04:00
' _type ' : ' video ' ,
2014-01-06 11:15:27 -05:00
' id ' : video_id ,
2016-10-14 12:43:09 -04:00
' title ' : title ,
2014-01-06 11:15:27 -05:00
' formats ' : formats ,
2016-10-19 06:34:15 -04:00
' subtitles ' : subtitles ,
2014-01-06 11:15:27 -05:00
' description ' : sd . get ( ' description ' ) ,
2016-10-14 12:43:09 -04:00
' duration ' : int_or_none ( sd . get ( ' duration_in_seconds ' ) ) ,
2014-01-06 11:15:27 -05:00
' upload_date ' : upload_date ,
' thumbnail ' : sd . get ( ' image_full_url ' ) ,
} )
return {
' _type ' : ' playlist ' ,
' entries ' : entries ,
' id ' : playlist_id ,
}
2014-08-09 14:21:16 -04:00
class ORFOE1IE ( InfoExtractor ) :
IE_NAME = ' orf:oe1 '
IE_DESC = ' Radio Österreich 1 '
2016-07-26 12:14:04 -04:00
_VALID_URL = r ' https?://oe1 \ .orf \ .at/(?:programm/|konsole \ ?.*? \ btrack_id=)(?P<id>[0-9]+) '
2015-01-10 08:27:27 -05:00
# Audios on ORF radio are only available for 7 days, so we can't add tests.
2016-07-26 12:14:04 -04:00
_TESTS = [ {
2015-01-10 08:27:27 -05:00
' url ' : ' http://oe1.orf.at/konsole?show=on_demand#?track_id=394211 ' ,
' only_matching ' : True ,
2016-07-26 12:14:04 -04:00
} , {
' url ' : ' http://oe1.orf.at/konsole?show=ondemand&track_id=443608&load_day=/programm/konsole/tag/20160726 ' ,
' only_matching ' : True ,
} ]
2014-08-09 14:21:16 -04:00
def _real_extract ( self , url ) :
2014-12-13 06:41:31 -05:00
show_id = self . _match_id ( url )
2014-08-09 14:21:16 -04:00
data = self . _download_json (
' http://oe1.orf.at/programm/ %s /konsole ' % show_id ,
show_id
)
timestamp = datetime . datetime . strptime ( ' %s %s ' % (
data [ ' item ' ] [ ' day_label ' ] ,
data [ ' item ' ] [ ' time ' ]
) , ' %d . % m. % Y % H: % M ' )
unix_timestamp = calendar . timegm ( timestamp . utctimetuple ( ) )
return {
' id ' : show_id ,
' title ' : data [ ' item ' ] [ ' title ' ] ,
' url ' : data [ ' item ' ] [ ' url_stream ' ] ,
' ext ' : ' mp3 ' ,
' description ' : data [ ' item ' ] . get ( ' info ' ) ,
' timestamp ' : unix_timestamp
}
class ORFFM4IE ( InfoExtractor ) :
2015-01-10 08:23:54 -05:00
IE_NAME = ' orf:fm4 '
2014-08-09 14:21:16 -04:00
IE_DESC = ' radio FM4 '
2016-03-21 11:36:32 -04:00
_VALID_URL = r ' https?://fm4 \ .orf \ .at/(?:7tage/?#|player/)(?P<date>[0-9]+)/(?P<show> \ w+) '
2014-08-09 14:21:16 -04:00
2016-01-14 11:11:33 -05:00
_TEST = {
' url ' : ' http://fm4.orf.at/player/20160110/IS/ ' ,
' md5 ' : ' 01e736e8f1cef7e13246e880a59ad298 ' ,
' info_dict ' : {
' id ' : ' 2016-01-10_2100_tl_54_7DaysSun13_11244 ' ,
' ext ' : ' mp3 ' ,
' title ' : ' Im Sumpf ' ,
' description ' : ' md5:384c543f866c4e422a55f66a62d669cd ' ,
' duration ' : 7173 ,
' timestamp ' : 1452456073 ,
' upload_date ' : ' 20160110 ' ,
} ,
2016-04-29 02:24:07 -04:00
' skip ' : ' Live streams on FM4 got deleted soon ' ,
2016-01-14 11:11:33 -05:00
}
2014-08-09 14:21:16 -04:00
def _real_extract ( self , url ) :
mobj = re . match ( self . _VALID_URL , url )
show_date = mobj . group ( ' date ' )
show_id = mobj . group ( ' show ' )
data = self . _download_json (
' http://audioapi.orf.at/fm4/json/2.0/broadcasts/ %s /4 %s ' % ( show_date , show_id ) ,
show_id
)
def extract_entry_dict ( info , title , subtitle ) :
return {
' id ' : info [ ' loopStreamId ' ] . replace ( ' .mp3 ' , ' ' ) ,
' url ' : ' http://loopstream01.apa.at/?channel=fm4&id= %s ' % info [ ' loopStreamId ' ] ,
' title ' : title ,
' description ' : subtitle ,
' duration ' : ( info [ ' end ' ] - info [ ' start ' ] ) / 1000 ,
' timestamp ' : info [ ' start ' ] / 1000 ,
' ext ' : ' mp3 '
}
entries = [ extract_entry_dict ( t , data [ ' title ' ] , data [ ' subtitle ' ] ) for t in data [ ' streams ' ] ]
return {
' _type ' : ' playlist ' ,
' id ' : show_id ,
' title ' : data [ ' title ' ] ,
' description ' : data [ ' subtitle ' ] ,
' entries ' : entries
2014-11-23 14:41:03 -05:00
}
2015-03-07 06:31:03 -05:00
class ORFIPTVIE ( InfoExtractor ) :
IE_NAME = ' orf:iptv '
IE_DESC = ' iptv.ORF.at '
2016-03-21 11:36:32 -04:00
_VALID_URL = r ' https?://iptv \ .orf \ .at/(?:#/)?stories/(?P<id> \ d+) '
2015-03-07 06:31:03 -05:00
_TEST = {
2015-04-25 11:06:27 -04:00
' url ' : ' http://iptv.orf.at/stories/2275236/ ' ,
' md5 ' : ' c8b22af4718a4b4af58342529453e3e5 ' ,
2015-03-07 06:31:03 -05:00
' info_dict ' : {
2015-04-25 11:06:27 -04:00
' id ' : ' 350612 ' ,
2015-03-07 06:31:03 -05:00
' ext ' : ' flv ' ,
2015-04-25 11:06:27 -04:00
' title ' : ' Weitere Evakuierungen um Vulkan Calbuco ' ,
' description ' : ' md5:d689c959bdbcf04efeddedbf2299d633 ' ,
' duration ' : 68.197 ,
2015-03-07 06:31:03 -05:00
' thumbnail ' : ' re:^https?://.* \ .jpg$ ' ,
2015-04-25 11:06:27 -04:00
' upload_date ' : ' 20150425 ' ,
2015-03-07 06:31:03 -05:00
} ,
}
def _real_extract ( self , url ) :
story_id = self . _match_id ( url )
webpage = self . _download_webpage (
' http://iptv.orf.at/stories/ %s ' % story_id , story_id )
video_id = self . _search_regex (
r ' data-video(?:id)?= " ( \ d+) " ' , webpage , ' video id ' )
data = self . _download_json (
' http://bits.orf.at/filehandler/static-api/json/current/data.json?file= %s ' % video_id ,
video_id ) [ 0 ]
duration = float_or_none ( data [ ' duration ' ] , 1000 )
video = data [ ' sources ' ] [ ' default ' ]
load_balancer_url = video [ ' loadBalancerUrl ' ]
abr = int_or_none ( video . get ( ' audioBitrate ' ) )
vbr = int_or_none ( video . get ( ' bitrate ' ) )
fps = int_or_none ( video . get ( ' videoFps ' ) )
width = int_or_none ( video . get ( ' videoWidth ' ) )
height = int_or_none ( video . get ( ' videoHeight ' ) )
thumbnail = video . get ( ' preview ' )
rendition = self . _download_json (
load_balancer_url , video_id , transform_source = strip_jsonp )
f = {
' abr ' : abr ,
' vbr ' : vbr ,
' fps ' : fps ,
' width ' : width ,
' height ' : height ,
}
formats = [ ]
for format_id , format_url in rendition [ ' redirect ' ] . items ( ) :
if format_id == ' rtmp ' :
ff = f . copy ( )
ff . update ( {
' url ' : format_url ,
' format_id ' : format_id ,
} )
formats . append ( ff )
elif determine_ext ( format_url ) == ' f4m ' :
formats . extend ( self . _extract_f4m_formats (
format_url , video_id , f4m_id = format_id ) )
elif determine_ext ( format_url ) == ' m3u8 ' :
formats . extend ( self . _extract_m3u8_formats (
format_url , video_id , ' mp4 ' , m3u8_id = format_id ) )
else :
continue
self . _sort_formats ( formats )
title = remove_end ( self . _og_search_title ( webpage ) , ' - iptv.ORF.at ' )
description = self . _og_search_description ( webpage )
upload_date = unified_strdate ( self . _html_search_meta (
' dc.date ' , webpage , ' upload date ' ) )
return {
' id ' : video_id ,
' title ' : title ,
' description ' : description ,
' duration ' : duration ,
' thumbnail ' : thumbnail ,
' upload_date ' : upload_date ,
' formats ' : formats ,
}