]> jfr.im git - yt-dlp.git/blame - youtube_dl/extractor/globo.py
[globo] Update tests
[yt-dlp.git] / youtube_dl / extractor / globo.py
CommitLineData
f47754f0
S
1# coding: utf-8
2from __future__ import unicode_literals
3
4import random
5import math
6
7from .common import InfoExtractor
8c25f81b 8from ..compat import (
f47754f0
S
9 compat_str,
10 compat_chr,
11 compat_ord,
12)
8c25f81b
PH
13from ..utils import (
14 ExtractorError,
15 float_or_none,
fffccaaf 16 int_or_none,
8c25f81b 17)
f47754f0
S
18
19
20class GloboIE(InfoExtractor):
ad607563 21 _VALID_URL = '(?:globo:|https?://.+?\.globo\.com/(?:[^/]+/)*(?:v/(?:[^/]+/)?|videos/))(?P<id>\d{7,})'
f47754f0
S
22
23 _API_URL_TEMPLATE = 'http://api.globovideos.com/videos/%s/playlist'
d7d79106 24 _SECURITY_URL_TEMPLATE = 'http://security.video.globo.com/videos/%s/hash?player=flash&version=17.0.0.132&resource_id=%s'
f47754f0 25
f47754f0
S
26 _RESIGN_EXPIRATION = 86400
27
ad607563 28 _TESTS = [{
ad607563
S
29 'url': 'http://g1.globo.com/carros/autoesporte/videos/t/exclusivos-do-g1/v/mercedes-benz-gla-passa-por-teste-de-colisao-na-europa/3607726/',
30 'md5': 'b3ccc801f75cd04a914d51dadb83a78d',
31 'info_dict': {
32 'id': '3607726',
33 'ext': 'mp4',
34 'title': 'Mercedes-Benz GLA passa por teste de colisão na Europa',
35 'duration': 103.204,
36 'uploader': 'Globo.com',
37 'uploader_id': 265,
264cd00f 38 },
ad607563 39 }, {
264cd00f
S
40 'url': 'http://globoplay.globo.com/v/4581987/',
41 'md5': 'f36a1ecd6a50da1577eee6dd17f67eff',
ad607563 42 'info_dict': {
264cd00f 43 'id': '4581987',
ad607563 44 'ext': 'mp4',
264cd00f
S
45 'title': 'Acidentes de trânsito estão entre as maiores causas de queda de energia em SP',
46 'duration': 137.973,
47 'uploader': 'Rede Globo',
48 'uploader_id': 196,
49 },
50 }, {
51 'url': 'http://canalbrasil.globo.com/programas/sangue-latino/videos/3928201.html',
52 'only_matching': True,
53 }, {
54 'url': 'http://globosatplay.globo.com/globonews/v/4472924/',
55 'only_matching': True,
56 }, {
57 'url': 'http://globotv.globo.com/t/programa/v/clipe-sexo-e-as-negas-adeus/3836166/',
58 'only_matching': True,
59 }, {
60 'url': 'http://globotv.globo.com/canal-brasil/sangue-latino/t/todos-os-videos/v/ator-e-diretor-argentino-ricado-darin-fala-sobre-utopias-e-suas-perdas/3928201/',
61 'only_matching': True,
ad607563 62 }]
f47754f0 63
ad607563 64 class MD5:
f47754f0
S
65 HEX_FORMAT_LOWERCASE = 0
66 HEX_FORMAT_UPPERCASE = 1
67 BASE64_PAD_CHARACTER_DEFAULT_COMPLIANCE = ''
68 BASE64_PAD_CHARACTER_RFC_COMPLIANCE = '='
69 PADDING = '=0xFF01DD'
70 hexcase = 0
71 b64pad = ''
72
73 def __init__(self):
74 pass
75
76 class JSArray(list):
77 def __getitem__(self, y):
78 try:
79 return list.__getitem__(self, y)
80 except IndexError:
81 return 0
82
83 def __setitem__(self, i, y):
84 try:
85 return list.__setitem__(self, i, y)
86 except IndexError:
87 self.extend([0] * (i - len(self) + 1))
88 self[-1] = y
89
90 @classmethod
91 def hex_md5(cls, param1):
92 return cls.rstr2hex(cls.rstr_md5(cls.str2rstr_utf8(param1)))
93
94 @classmethod
95 def b64_md5(cls, param1, param2=None):
96 return cls.rstr2b64(cls.rstr_md5(cls.str2rstr_utf8(param1, param2)))
97
98 @classmethod
99 def any_md5(cls, param1, param2):
100 return cls.rstr2any(cls.rstr_md5(cls.str2rstr_utf8(param1)), param2)
101
102 @classmethod
103 def rstr_md5(cls, param1):
104 return cls.binl2rstr(cls.binl_md5(cls.rstr2binl(param1), len(param1) * 8))
105
106 @classmethod
107 def rstr2hex(cls, param1):
108 _loc_2 = '0123456789ABCDEF' if cls.hexcase else '0123456789abcdef'
109 _loc_3 = ''
110 for _loc_5 in range(0, len(param1)):
111 _loc_4 = compat_ord(param1[_loc_5])
112 _loc_3 += _loc_2[_loc_4 >> 4 & 15] + _loc_2[_loc_4 & 15]
113 return _loc_3
114
115 @classmethod
116 def rstr2b64(cls, param1):
117 _loc_2 = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_'
118 _loc_3 = ''
119 _loc_4 = len(param1)
120 for _loc_5 in range(0, _loc_4, 3):
121 _loc_6_1 = compat_ord(param1[_loc_5]) << 16
122 _loc_6_2 = compat_ord(param1[_loc_5 + 1]) << 8 if _loc_5 + 1 < _loc_4 else 0
123 _loc_6_3 = compat_ord(param1[_loc_5 + 2]) if _loc_5 + 2 < _loc_4 else 0
124 _loc_6 = _loc_6_1 | _loc_6_2 | _loc_6_3
125 for _loc_7 in range(0, 4):
126 if _loc_5 * 8 + _loc_7 * 6 > len(param1) * 8:
127 _loc_3 += cls.b64pad
128 else:
129 _loc_3 += _loc_2[_loc_6 >> 6 * (3 - _loc_7) & 63]
130 return _loc_3
131
132 @staticmethod
133 def rstr2any(param1, param2):
134 _loc_3 = len(param2)
135 _loc_4 = []
136 _loc_9 = [0] * ((len(param1) >> 2) + 1)
137 for _loc_5 in range(0, len(_loc_9)):
138 _loc_9[_loc_5] = compat_ord(param1[_loc_5 * 2]) << 8 | compat_ord(param1[_loc_5 * 2 + 1])
139
140 while len(_loc_9) > 0:
141 _loc_8 = []
142 _loc_7 = 0
143 for _loc_5 in range(0, len(_loc_9)):
144 _loc_7 = (_loc_7 << 16) + _loc_9[_loc_5]
145 _loc_6 = math.floor(_loc_7 / _loc_3)
146 _loc_7 -= _loc_6 * _loc_3
147 if len(_loc_8) > 0 or _loc_6 > 0:
148 _loc_8[len(_loc_8)] = _loc_6
149
150 _loc_4[len(_loc_4)] = _loc_7
151 _loc_9 = _loc_8
152
153 _loc_10 = ''
154 _loc_5 = len(_loc_4) - 1
155 while _loc_5 >= 0:
156 _loc_10 += param2[_loc_4[_loc_5]]
157 _loc_5 -= 1
158
159 return _loc_10
160
161 @classmethod
162 def str2rstr_utf8(cls, param1, param2=None):
163 _loc_3 = ''
164 _loc_4 = -1
165 if not param2:
166 param2 = cls.PADDING
167 param1 = param1 + param2[1:9]
168 while True:
169 _loc_4 += 1
170 if _loc_4 >= len(param1):
171 break
172 _loc_5 = compat_ord(param1[_loc_4])
173 _loc_6 = compat_ord(param1[_loc_4 + 1]) if _loc_4 + 1 < len(param1) else 0
174 if 55296 <= _loc_5 <= 56319 and 56320 <= _loc_6 <= 57343:
175 _loc_5 = 65536 + ((_loc_5 & 1023) << 10) + (_loc_6 & 1023)
176 _loc_4 += 1
177 if _loc_5 <= 127:
178 _loc_3 += compat_chr(_loc_5)
179 continue
180 if _loc_5 <= 2047:
181 _loc_3 += compat_chr(192 | _loc_5 >> 6 & 31) + compat_chr(128 | _loc_5 & 63)
182 continue
183 if _loc_5 <= 65535:
184 _loc_3 += compat_chr(224 | _loc_5 >> 12 & 15) + compat_chr(128 | _loc_5 >> 6 & 63) + compat_chr(
185 128 | _loc_5 & 63)
186 continue
187 if _loc_5 <= 2097151:
188 _loc_3 += compat_chr(240 | _loc_5 >> 18 & 7) + compat_chr(128 | _loc_5 >> 12 & 63) + compat_chr(
189 128 | _loc_5 >> 6 & 63) + compat_chr(128 | _loc_5 & 63)
190 return _loc_3
191
192 @staticmethod
193 def rstr2binl(param1):
194 _loc_2 = [0] * ((len(param1) >> 2) + 1)
195 for _loc_3 in range(0, len(_loc_2)):
196 _loc_2[_loc_3] = 0
197 for _loc_3 in range(0, len(param1) * 8, 8):
198 _loc_2[_loc_3 >> 5] |= (compat_ord(param1[_loc_3 // 8]) & 255) << _loc_3 % 32
199 return _loc_2
200
201 @staticmethod
202 def binl2rstr(param1):
203 _loc_2 = ''
204 for _loc_3 in range(0, len(param1) * 32, 8):
205 _loc_2 += compat_chr(param1[_loc_3 >> 5] >> _loc_3 % 32 & 255)
206 return _loc_2
207
208 @classmethod
209 def binl_md5(cls, param1, param2):
210 param1 = cls.JSArray(param1)
211 param1[param2 >> 5] |= 128 << param2 % 32
212 param1[(param2 + 64 >> 9 << 4) + 14] = param2
213 _loc_3 = 1732584193
214 _loc_4 = -271733879
215 _loc_5 = -1732584194
216 _loc_6 = 271733878
217 for _loc_7 in range(0, len(param1), 16):
218 _loc_8 = _loc_3
219 _loc_9 = _loc_4
220 _loc_10 = _loc_5
221 _loc_11 = _loc_6
222 _loc_3 = cls.md5_ff(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 0], 7, -680876936)
223 _loc_6 = cls.md5_ff(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 1], 12, -389564586)
224 _loc_5 = cls.md5_ff(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 2], 17, 606105819)
225 _loc_4 = cls.md5_ff(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 3], 22, -1044525330)
226 _loc_3 = cls.md5_ff(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 4], 7, -176418897)
227 _loc_6 = cls.md5_ff(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 5], 12, 1200080426)
228 _loc_5 = cls.md5_ff(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 6], 17, -1473231341)
229 _loc_4 = cls.md5_ff(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 7], 22, -45705983)
230 _loc_3 = cls.md5_ff(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 8], 7, 1770035416)
231 _loc_6 = cls.md5_ff(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 9], 12, -1958414417)
232 _loc_5 = cls.md5_ff(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 10], 17, -42063)
233 _loc_4 = cls.md5_ff(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 11], 22, -1990404162)
234 _loc_3 = cls.md5_ff(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 12], 7, 1804603682)
235 _loc_6 = cls.md5_ff(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 13], 12, -40341101)
236 _loc_5 = cls.md5_ff(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 14], 17, -1502002290)
237 _loc_4 = cls.md5_ff(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 15], 22, 1236535329)
238 _loc_3 = cls.md5_gg(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 1], 5, -165796510)
239 _loc_6 = cls.md5_gg(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 6], 9, -1069501632)
240 _loc_5 = cls.md5_gg(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 11], 14, 643717713)
241 _loc_4 = cls.md5_gg(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 0], 20, -373897302)
242 _loc_3 = cls.md5_gg(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 5], 5, -701558691)
243 _loc_6 = cls.md5_gg(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 10], 9, 38016083)
244 _loc_5 = cls.md5_gg(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 15], 14, -660478335)
245 _loc_4 = cls.md5_gg(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 4], 20, -405537848)
246 _loc_3 = cls.md5_gg(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 9], 5, 568446438)
247 _loc_6 = cls.md5_gg(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 14], 9, -1019803690)
248 _loc_5 = cls.md5_gg(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 3], 14, -187363961)
249 _loc_4 = cls.md5_gg(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 8], 20, 1163531501)
250 _loc_3 = cls.md5_gg(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 13], 5, -1444681467)
251 _loc_6 = cls.md5_gg(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 2], 9, -51403784)
252 _loc_5 = cls.md5_gg(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 7], 14, 1735328473)
253 _loc_4 = cls.md5_gg(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 12], 20, -1926607734)
254 _loc_3 = cls.md5_hh(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 5], 4, -378558)
255 _loc_6 = cls.md5_hh(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 8], 11, -2022574463)
256 _loc_5 = cls.md5_hh(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 11], 16, 1839030562)
257 _loc_4 = cls.md5_hh(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 14], 23, -35309556)
258 _loc_3 = cls.md5_hh(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 1], 4, -1530992060)
259 _loc_6 = cls.md5_hh(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 4], 11, 1272893353)
260 _loc_5 = cls.md5_hh(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 7], 16, -155497632)
261 _loc_4 = cls.md5_hh(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 10], 23, -1094730640)
262 _loc_3 = cls.md5_hh(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 13], 4, 681279174)
263 _loc_6 = cls.md5_hh(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 0], 11, -358537222)
264 _loc_5 = cls.md5_hh(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 3], 16, -722521979)
265 _loc_4 = cls.md5_hh(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 6], 23, 76029189)
266 _loc_3 = cls.md5_hh(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 9], 4, -640364487)
267 _loc_6 = cls.md5_hh(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 12], 11, -421815835)
268 _loc_5 = cls.md5_hh(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 15], 16, 530742520)
269 _loc_4 = cls.md5_hh(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 2], 23, -995338651)
270 _loc_3 = cls.md5_ii(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 0], 6, -198630844)
271 _loc_6 = cls.md5_ii(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 7], 10, 1126891415)
272 _loc_5 = cls.md5_ii(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 14], 15, -1416354905)
273 _loc_4 = cls.md5_ii(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 5], 21, -57434055)
274 _loc_3 = cls.md5_ii(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 12], 6, 1700485571)
275 _loc_6 = cls.md5_ii(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 3], 10, -1894986606)
276 _loc_5 = cls.md5_ii(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 10], 15, -1051523)
277 _loc_4 = cls.md5_ii(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 1], 21, -2054922799)
278 _loc_3 = cls.md5_ii(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 8], 6, 1873313359)
279 _loc_6 = cls.md5_ii(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 15], 10, -30611744)
280 _loc_5 = cls.md5_ii(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 6], 15, -1560198380)
281 _loc_4 = cls.md5_ii(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 13], 21, 1309151649)
282 _loc_3 = cls.md5_ii(_loc_3, _loc_4, _loc_5, _loc_6, param1[_loc_7 + 4], 6, -145523070)
283 _loc_6 = cls.md5_ii(_loc_6, _loc_3, _loc_4, _loc_5, param1[_loc_7 + 11], 10, -1120210379)
284 _loc_5 = cls.md5_ii(_loc_5, _loc_6, _loc_3, _loc_4, param1[_loc_7 + 2], 15, 718787259)
285 _loc_4 = cls.md5_ii(_loc_4, _loc_5, _loc_6, _loc_3, param1[_loc_7 + 9], 21, -343485551)
286 _loc_3 = cls.safe_add(_loc_3, _loc_8)
287 _loc_4 = cls.safe_add(_loc_4, _loc_9)
288 _loc_5 = cls.safe_add(_loc_5, _loc_10)
289 _loc_6 = cls.safe_add(_loc_6, _loc_11)
290 return [_loc_3, _loc_4, _loc_5, _loc_6]
291
292 @classmethod
293 def md5_cmn(cls, param1, param2, param3, param4, param5, param6):
294 return cls.safe_add(
295 cls.bit_rol(cls.safe_add(cls.safe_add(param2, param1), cls.safe_add(param4, param6)), param5), param3)
296
297 @classmethod
298 def md5_ff(cls, param1, param2, param3, param4, param5, param6, param7):
299 return cls.md5_cmn(param2 & param3 | ~param2 & param4, param1, param2, param5, param6, param7)
300
301 @classmethod
302 def md5_gg(cls, param1, param2, param3, param4, param5, param6, param7):
303 return cls.md5_cmn(param2 & param4 | param3 & ~param4, param1, param2, param5, param6, param7)
304
305 @classmethod
306 def md5_hh(cls, param1, param2, param3, param4, param5, param6, param7):
307 return cls.md5_cmn(param2 ^ param3 ^ param4, param1, param2, param5, param6, param7)
308
309 @classmethod
310 def md5_ii(cls, param1, param2, param3, param4, param5, param6, param7):
311 return cls.md5_cmn(param3 ^ (param2 | ~param4), param1, param2, param5, param6, param7)
312
313 @classmethod
314 def safe_add(cls, param1, param2):
315 _loc_3 = (param1 & 65535) + (param2 & 65535)
316 _loc_4 = (param1 >> 16) + (param2 >> 16) + (_loc_3 >> 16)
317 return cls.lshift(_loc_4, 16) | _loc_3 & 65535
318
319 @classmethod
320 def bit_rol(cls, param1, param2):
321 return cls.lshift(param1, param2) | (param1 & 0xFFFFFFFF) >> (32 - param2)
322
323 @staticmethod
324 def lshift(value, count):
325 r = (0xFFFFFFFF & value) << count
326 return -(~(r - 1) & 0xFFFFFFFF) if r > 0x7FFFFFFF else r
327
328 def _real_extract(self, url):
329 video_id = self._match_id(url)
330
f47754f0
S
331 video = self._download_json(
332 self._API_URL_TEMPLATE % video_id, video_id)['videos'][0]
333
334 title = video['title']
f47754f0
S
335
336 formats = []
f47754f0
S
337 for resource in video['resources']:
338 resource_id = resource.get('_id')
c3459d24 339 if not resource_id or resource_id.endswith('manifest'):
f47754f0
S
340 continue
341
342 security = self._download_json(
343 self._SECURITY_URL_TEMPLATE % (video_id, resource_id),
344 video_id, 'Downloading security hash for %s' % resource_id)
345
346 security_hash = security.get('hash')
347 if not security_hash:
348 message = security.get('message')
349 if message:
350 raise ExtractorError(
351 '%s returned error: %s' % (self.IE_NAME, message), expected=True)
352 continue
353
354 hash_code = security_hash[:2]
355 received_time = int(security_hash[2:12])
356 received_random = security_hash[12:22]
357 received_md5 = security_hash[22:]
358
359 sign_time = received_time + self._RESIGN_EXPIRATION
360 padding = '%010d' % random.randint(1, 10000000000)
361
362 signed_md5 = self.MD5.b64_md5(received_md5 + compat_str(sign_time) + padding)
363 signed_hash = hash_code + compat_str(received_time) + received_random + compat_str(sign_time) + padding + signed_md5
364
8c72beb2
S
365 resource_url = resource['url']
366 signed_url = '%s?h=%s&k=%s' % (resource_url, signed_hash, 'flash')
367 if resource_id.endswith('m3u8') or resource_url.endswith('.m3u8'):
e3778cce 368 m3u8_formats = self._extract_m3u8_formats(
5d235ca7
S
369 signed_url, resource_id, 'mp4', entry_protocol='m3u8_native',
370 m3u8_id='hls', fatal=False)
e3778cce
S
371 if m3u8_formats:
372 formats.extend(m3u8_formats)
8c72beb2
S
373 else:
374 formats.append({
375 'url': signed_url,
a4a6b7b8
S
376 'format_id': 'http-%s' % resource_id,
377 'height': int_or_none(resource.get('height')),
8c72beb2 378 })
f47754f0
S
379
380 self._sort_formats(formats)
381
fffccaaf 382 duration = float_or_none(video.get('duration'), 1000)
fffccaaf
S
383 uploader = video.get('channel')
384 uploader_id = video.get('channel_id')
385
f47754f0
S
386 return {
387 'id': video_id,
388 'title': title,
389 'duration': duration,
390 'uploader': uploader,
391 'uploader_id': uploader_id,
f47754f0 392 'formats': formats
5f6a1245 393 }
ad607563
S
394
395
396class GloboArticleIE(InfoExtractor):
397 _VALID_URL = 'https?://.+?\.globo\.com/(?:[^/]+/)*(?P<id>[^/]+)\.html'
398
399 _VIDEOID_REGEXES = [
400 r'\bdata-video-id=["\'](\d{7,})',
401 r'\bdata-player-videosids=["\'](\d{7,})',
402 r'\bvideosIDs\s*:\s*["\'](\d{7,})',
403 r'\bdata-id=["\'](\d{7,})',
404 r'<div[^>]+\bid=["\'](\d{7,})',
405 ]
406
407 _TEST = {
408 'url': 'http://g1.globo.com/jornal-nacional/noticia/2014/09/novidade-na-fiscalizacao-de-bagagem-pela-receita-provoca-discussoes.html',
409 'md5': '307fdeae4390ccfe6ba1aa198cf6e72b',
410 'info_dict': {
411 'id': '3652183',
412 'ext': 'mp4',
413 'title': 'Receita Federal explica como vai fiscalizar bagagens de quem retorna ao Brasil de avião',
414 'duration': 110.711,
415 'uploader': 'Rede Globo',
416 'uploader_id': 196,
ad607563
S
417 }
418 }
419
420 @classmethod
421 def suitable(cls, url):
422 return False if GloboIE.suitable(url) else super(GloboArticleIE, cls).suitable(url)
423
424 def _real_extract(self, url):
425 display_id = self._match_id(url)
426 webpage = self._download_webpage(url, display_id)
427 video_id = self._search_regex(self._VIDEOID_REGEXES, webpage, 'video id')
428 return self.url_result('globo:%s' % video_id, 'Globo')