2015-02-20 07:13:41 -05:00
# coding: utf-8
from __future__ import unicode_literals
from . common import InfoExtractor
2016-09-15 13:54:34 -04:00
from . . compat import compat_str
2015-02-20 07:13:41 -05:00
from . . utils import (
ExtractorError ,
2016-09-15 13:54:34 -04:00
int_or_none ,
2015-02-20 07:13:41 -05:00
parse_iso8601 ,
2016-09-15 13:54:34 -04:00
try_get ,
update_url_query ,
2015-02-20 07:13:41 -05:00
)
class TV4IE ( InfoExtractor ) :
IE_DESC = ' tv4.se and tv4play.se '
_VALID_URL = r ''' (?x)https?://(?:www \ .)?
( ? :
tv4 \. se / ( ? : [ ^ / ] + ) / klipp / ( ? : . * ) - |
tv4play \. se /
( ? :
( ? : program | barn ) / ( ? : [ ^ \? ] + ) \? video_id = |
iframe / video / |
film / |
sport / |
)
) ( ? P < id > [ 0 - 9 ] + ) '''
_TESTS = [
{
' url ' : ' http://www.tv4.se/kalla-fakta/klipp/kalla-fakta-5-english-subtitles-2491650 ' ,
' md5 ' : ' 909d6454b87b10a25aa04c4bdd416a9b ' ,
' info_dict ' : {
' id ' : ' 2491650 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Kalla Fakta 5 (english subtitles) ' ,
2017-01-02 07:08:07 -05:00
' thumbnail ' : r ' re:^https?://.* \ .jpg$ ' ,
2015-02-20 07:13:41 -05:00
' timestamp ' : int ,
' upload_date ' : ' 20131125 ' ,
} ,
} ,
{
' url ' : ' http://www.tv4play.se/iframe/video/3054113 ' ,
' md5 ' : ' 77f851c55139ffe0ebd41b6a5552489b ' ,
' info_dict ' : {
' id ' : ' 3054113 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Så här jobbar ficktjuvarna - se avslöjande bilder ' ,
2017-01-02 07:08:07 -05:00
' thumbnail ' : r ' re:^https?://.* \ .jpg$ ' ,
2015-02-20 07:13:41 -05:00
' description ' : ' Unika bilder avslöjar hur turisternas fickor vittjas mitt på Stockholms central. Två experter på ficktjuvarna avslöjar knepen du ska se upp för. ' ,
' timestamp ' : int ,
' upload_date ' : ' 20150130 ' ,
} ,
} ,
{
' url ' : ' http://www.tv4play.se/sport/3060959 ' ,
' only_matching ' : True ,
} ,
{
' url ' : ' http://www.tv4play.se/film/2378136 ' ,
' only_matching ' : True ,
} ,
{
' url ' : ' http://www.tv4play.se/barn/looney-tunes?video_id=3062412 ' ,
' only_matching ' : True ,
} ,
]
def _real_extract ( self , url ) :
video_id = self . _match_id ( url )
info = self . _download_json (
2016-09-15 13:54:34 -04:00
' http://www.tv4play.se/player/assets/ %s .json ' % video_id ,
video_id , ' Downloading video info JSON ' )
2015-02-20 07:13:41 -05:00
2016-01-10 10:17:47 -05:00
# If is_geo_restricted is true, it doesn't necessarily mean we can't download it
2016-09-15 13:54:34 -04:00
if info . get ( ' is_geo_restricted ' ) :
2015-02-20 07:13:41 -05:00
self . report_warning ( ' This content might not be available in your country due to licensing restrictions. ' )
2016-09-15 13:54:34 -04:00
if info . get ( ' requires_subscription ' ) :
2015-02-20 07:13:41 -05:00
raise ExtractorError ( ' This content requires subscription. ' , expected = True )
2016-09-15 13:54:34 -04:00
title = info [ ' title ' ]
2015-02-20 07:13:41 -05:00
formats = [ ]
2016-09-15 13:54:34 -04:00
# http formats are linked with unresolvable host
for kind in ( ' hls ' , ' ' ) :
data = self . _download_json (
' https://prima.tv4play.se/api/web/asset/ %s /play.json ' % video_id ,
video_id , ' Downloading sources JSON ' , query = {
' protocol ' : kind ,
' videoFormat ' : ' MP4+WEBVTTS+WEBVTT ' ,
} )
item = try_get ( data , lambda x : x [ ' playback ' ] [ ' items ' ] [ ' item ' ] , dict )
manifest_url = item . get ( ' url ' )
if not isinstance ( manifest_url , compat_str ) :
continue
if kind == ' hls ' :
formats . extend ( self . _extract_m3u8_formats (
manifest_url , video_id , ' mp4 ' , entry_protocol = ' m3u8_native ' ,
m3u8_id = kind , fatal = False ) )
else :
formats . extend ( self . _extract_f4m_formats (
update_url_query ( manifest_url , { ' hdcore ' : ' 3.8.0 ' } ) ,
video_id , f4m_id = ' hds ' , fatal = False ) )
2015-02-20 07:13:41 -05:00
self . _sort_formats ( formats )
return {
' id ' : video_id ,
2016-09-15 13:54:34 -04:00
' title ' : title ,
2015-02-20 07:13:41 -05:00
' formats ' : formats ,
' description ' : info . get ( ' description ' ) ,
' timestamp ' : parse_iso8601 ( info . get ( ' broadcast_date_time ' ) ) ,
2016-09-15 13:54:34 -04:00
' duration ' : int_or_none ( info . get ( ' duration ' ) ) ,
2015-02-20 07:13:41 -05:00
' thumbnail ' : info . get ( ' image ' ) ,
2016-09-15 13:54:34 -04:00
' is_live ' : info . get ( ' is_live ' ) is True ,
2015-02-20 07:13:41 -05:00
}