You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

456 lines
22 KiB

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