| 
									
										
										
										
											2013-06-27 20:32:02 +02:00
										 |  |  | # coding: utf-8 | 
					
						
							| 
									
										
										
										
											2014-09-25 09:58:09 +02:00
										 |  |  | from __future__ import unicode_literals | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | import base64 | 
					
						
							| 
									
										
										
										
											2013-06-23 22:01:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | from .common import InfoExtractor | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | from ..utils import ExtractorError | 
					
						
							| 
									
										
										
										
											2013-06-23 22:01:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-15 23:28:59 +08:00
										 |  |  | from ..compat import ( | 
					
						
							|  |  |  |     compat_urllib_parse, | 
					
						
							|  |  |  |     compat_ord, | 
					
						
							| 
									
										
										
										
											2015-06-16 00:06:23 +08:00
										 |  |  |     compat_urllib_request, | 
					
						
							| 
									
										
										
										
											2015-06-15 23:28:59 +08:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2015-05-29 10:13:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-23 22:01:02 +02:00
										 |  |  | class YoukuIE(InfoExtractor): | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |     IE_NAME = 'youku' | 
					
						
							| 
									
										
										
										
											2015-07-11 00:58:13 +08:00
										 |  |  |     IE_DESC = '优酷' | 
					
						
							| 
									
										
										
										
											2014-09-25 09:58:09 +02:00
										 |  |  |     _VALID_URL = r'''(?x)
 | 
					
						
							|  |  |  |         (?: | 
					
						
							|  |  |  |             http://(?:v|player)\.youku\.com/(?:v_show/id_|player\.php/sid/)| | 
					
						
							|  |  |  |             youku:) | 
					
						
							|  |  |  |         (?P<id>[A-Za-z0-9]+)(?:\.html|/v\.swf|) | 
					
						
							|  |  |  |     '''
 | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-15 23:36:28 +08:00
										 |  |  |     _TESTS = [{ | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |         'url': 'http://v.youku.com/v_show/id_XMTc1ODE5Njcy.html', | 
					
						
							|  |  |  |         'md5': '5f3af4192eabacc4501508d54a8cabd7', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							| 
									
										
										
										
											2015-06-15 23:46:07 +08:00
										 |  |  |             'id': 'XMTc1ODE5Njcy_part1', | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |             'title': '★Smile﹗♡ Git Fresh -Booty Music舞蹈.', | 
					
						
							|  |  |  |             'ext': 'flv' | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-06-15 23:36:28 +08:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'http://player.youku.com/player.php/sid/XNDgyMDQ2NTQw/v.swf', | 
					
						
							|  |  |  |         'only_matching': True, | 
					
						
							| 
									
										
										
										
											2015-06-15 23:46:07 +08:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'http://v.youku.com/v_show/id_XODgxNjg1Mzk2_ev_1.html', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': 'XODgxNjg1Mzk2', | 
					
						
							|  |  |  |             'title': '武媚娘传奇 85', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'playlist_count': 11, | 
					
						
							| 
									
										
										
										
											2015-06-16 00:06:23 +08:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'http://v.youku.com/v_show/id_XMTI1OTczNDM5Mg==.html', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': 'XMTI1OTczNDM5Mg', | 
					
						
							|  |  |  |             'title': '花千骨 04', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'playlist_count': 13, | 
					
						
							|  |  |  |         'skip': 'Available in China only', | 
					
						
							| 
									
										
										
										
											2015-09-01 00:19:58 +08:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'http://v.youku.com/v_show/id_XNjA1NzA2Njgw.html', | 
					
						
							|  |  |  |         'note': 'Video protected with password', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': 'XNjA1NzA2Njgw', | 
					
						
							| 
									
										
										
										
											2015-09-01 22:26:17 +06:00
										 |  |  |             'title': '邢義田复旦讲座之想象中的胡人—从“左衽孔子”说起', | 
					
						
							| 
									
										
										
										
											2015-09-01 00:19:58 +08:00
										 |  |  |         }, | 
					
						
							| 
									
										
										
										
											2015-09-01 22:28:03 +06:00
										 |  |  |         'playlist_count': 19, | 
					
						
							| 
									
										
										
										
											2015-09-01 00:19:58 +08:00
										 |  |  |         'params': { | 
					
						
							|  |  |  |             'videopassword': '100600', | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2015-06-15 23:36:28 +08:00
										 |  |  |     }] | 
					
						
							| 
									
										
										
										
											2013-06-27 20:25:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |     def construct_video_urls(self, data1, data2): | 
					
						
							|  |  |  |         # get sid, token | 
					
						
							|  |  |  |         def yk_t(s1, s2): | 
					
						
							|  |  |  |             ls = list(range(256)) | 
					
						
							|  |  |  |             t = 0 | 
					
						
							|  |  |  |             for i in range(256): | 
					
						
							| 
									
										
										
										
											2015-06-15 23:28:59 +08:00
										 |  |  |                 t = (t + ls[i] + compat_ord(s1[i % len(s1)])) % 256 | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |                 ls[i], ls[t] = ls[t], ls[i] | 
					
						
							| 
									
										
										
										
											2015-06-15 23:28:59 +08:00
										 |  |  |             s = bytearray() | 
					
						
							| 
									
										
										
										
											2015-05-28 21:04:58 +08:00
										 |  |  |             x, y = 0, 0 | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |             for i in range(len(s2)): | 
					
						
							|  |  |  |                 y = (y + 1) % 256 | 
					
						
							|  |  |  |                 x = (x + ls[y]) % 256 | 
					
						
							|  |  |  |                 ls[x], ls[y] = ls[y], ls[x] | 
					
						
							| 
									
										
										
										
											2015-06-15 23:28:59 +08:00
										 |  |  |                 s.append(compat_ord(s2[i]) ^ ls[(ls[x] + ls[y]) % 256]) | 
					
						
							|  |  |  |             return bytes(s) | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         sid, token = yk_t( | 
					
						
							| 
									
										
										
										
											2015-06-15 23:28:59 +08:00
										 |  |  |             b'becaf9be', base64.b64decode(data2['ep'].encode('ascii')) | 
					
						
							|  |  |  |         ).decode('ascii').split('_') | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # get oip | 
					
						
							|  |  |  |         oip = data2['ip'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # get fileid | 
					
						
							|  |  |  |         string_ls = list( | 
					
						
							|  |  |  |             'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/\:._-1234567890') | 
					
						
							|  |  |  |         shuffled_string_ls = [] | 
					
						
							|  |  |  |         seed = data1['seed'] | 
					
						
							|  |  |  |         N = len(string_ls) | 
					
						
							|  |  |  |         for ii in range(N): | 
					
						
							|  |  |  |             seed = (seed * 0xd3 + 0x754f) % 0x10000 | 
					
						
							|  |  |  |             idx = seed * len(string_ls) // 0x10000 | 
					
						
							|  |  |  |             shuffled_string_ls.append(string_ls[idx]) | 
					
						
							|  |  |  |             del string_ls[idx] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fileid_dict = {} | 
					
						
							|  |  |  |         for format in data1['streamtypes']: | 
					
						
							|  |  |  |             streamfileid = [ | 
					
						
							|  |  |  |                 int(i) for i in data1['streamfileids'][format].strip('*').split('*')] | 
					
						
							|  |  |  |             fileid = ''.join( | 
					
						
							|  |  |  |                 [shuffled_string_ls[i] for i in streamfileid]) | 
					
						
							|  |  |  |             fileid_dict[format] = fileid[:8] + '%s' + fileid[10:] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def get_fileid(format, n): | 
					
						
							|  |  |  |             fileid = fileid_dict[format] % hex(int(n))[2:].upper().zfill(2) | 
					
						
							|  |  |  |             return fileid | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # get ep | 
					
						
							|  |  |  |         def generate_ep(format, n): | 
					
						
							|  |  |  |             fileid = get_fileid(format, n) | 
					
						
							|  |  |  |             ep_t = yk_t( | 
					
						
							| 
									
										
										
										
											2015-06-15 23:28:59 +08:00
										 |  |  |                 b'bf7e5f01', | 
					
						
							|  |  |  |                 ('%s_%s_%s' % (sid, fileid, token)).encode('ascii') | 
					
						
							| 
									
										
										
										
											2015-05-28 21:04:58 +08:00
										 |  |  |             ) | 
					
						
							| 
									
										
										
										
											2015-06-15 23:28:59 +08:00
										 |  |  |             ep = base64.b64encode(ep_t).decode('ascii') | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |             return ep | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # generate video_urls | 
					
						
							|  |  |  |         video_urls_dict = {} | 
					
						
							|  |  |  |         for format in data1['streamtypes']: | 
					
						
							|  |  |  |             video_urls = [] | 
					
						
							|  |  |  |             for dt in data1['segs'][format]: | 
					
						
							|  |  |  |                 n = str(int(dt['no'])) | 
					
						
							| 
									
										
										
										
											2015-05-29 10:13:09 +08:00
										 |  |  |                 param = { | 
					
						
							|  |  |  |                     'K': dt['k'], | 
					
						
							|  |  |  |                     'hd': self.get_hd(format), | 
					
						
							|  |  |  |                     'myp': 0, | 
					
						
							|  |  |  |                     'ts': dt['seconds'], | 
					
						
							|  |  |  |                     'ypp': 0, | 
					
						
							|  |  |  |                     'ctype': 12, | 
					
						
							|  |  |  |                     'ev': 1, | 
					
						
							|  |  |  |                     'token': token, | 
					
						
							|  |  |  |                     'oip': oip, | 
					
						
							|  |  |  |                     'ep': generate_ep(format, n) | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |                 video_url = \ | 
					
						
							|  |  |  |                     'http://k.youku.com/player/getFlvPath/' + \ | 
					
						
							|  |  |  |                     'sid/' + sid + \ | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |                     '_' + str(int(n) + 1).zfill(2) + \ | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |                     '/st/' + self.parse_ext_l(format) + \ | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |                     '/fileid/' + get_fileid(format, n) + '?' + \ | 
					
						
							| 
									
										
										
										
											2015-05-29 10:13:09 +08:00
										 |  |  |                     compat_urllib_parse.urlencode(param) | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |                 video_urls.append(video_url) | 
					
						
							|  |  |  |             video_urls_dict[format] = video_urls | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return video_urls_dict | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_hd(self, fm): | 
					
						
							|  |  |  |         hd_id_dict = { | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |             'flv': '0', | 
					
						
							|  |  |  |             'mp4': '1', | 
					
						
							|  |  |  |             'hd2': '2', | 
					
						
							|  |  |  |             'hd3': '3', | 
					
						
							|  |  |  |             '3gp': '0', | 
					
						
							|  |  |  |             '3gphd': '1' | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |         return hd_id_dict[fm] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def parse_ext_l(self, fm): | 
					
						
							|  |  |  |         ext_dict = { | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |             'flv': 'flv', | 
					
						
							|  |  |  |             'mp4': 'mp4', | 
					
						
							|  |  |  |             'hd2': 'flv', | 
					
						
							|  |  |  |             'hd3': 'flv', | 
					
						
							|  |  |  |             '3gp': 'flv', | 
					
						
							|  |  |  |             '3gphd': 'mp4' | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |         return ext_dict[fm] | 
					
						
							| 
									
										
										
										
											2013-06-23 22:01:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-30 10:03:32 +08:00
										 |  |  |     def get_format_name(self, fm): | 
					
						
							|  |  |  |         _dict = { | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |             '3gp': 'h6', | 
					
						
							|  |  |  |             '3gphd': 'h5', | 
					
						
							|  |  |  |             'flv': 'h4', | 
					
						
							|  |  |  |             'mp4': 'h3', | 
					
						
							|  |  |  |             'hd2': 'h2', | 
					
						
							|  |  |  |             'hd3': 'h1' | 
					
						
							| 
									
										
										
										
											2015-05-30 10:03:32 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |         return _dict[fm] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-23 22:01:02 +02:00
										 |  |  |     def _real_extract(self, url): | 
					
						
							| 
									
										
										
										
											2015-06-15 23:31:30 +08:00
										 |  |  |         video_id = self._match_id(url) | 
					
						
							| 
									
										
										
										
											2013-06-23 22:01:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 00:06:23 +08:00
										 |  |  |         def retrieve_data(req_url, note): | 
					
						
							|  |  |  |             req = compat_urllib_request.Request(req_url) | 
					
						
							| 
									
										
										
										
											2013-06-23 22:01:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 00:06:23 +08:00
										 |  |  |             cn_verification_proxy = self._downloader.params.get('cn_verification_proxy') | 
					
						
							|  |  |  |             if cn_verification_proxy: | 
					
						
							|  |  |  |                 req.add_header('Ytdl-request-proxy', cn_verification_proxy) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             raw_data = self._download_json(req, video_id, note=note) | 
					
						
							|  |  |  |             return raw_data['data'][0] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-01 00:19:58 +08:00
										 |  |  |         video_password = self._downloader.params.get('videopassword', None) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 00:06:23 +08:00
										 |  |  |         # request basic data | 
					
						
							| 
									
										
										
										
											2015-09-01 11:07:47 +08:00
										 |  |  |         basic_data_url = 'http://v.youku.com/player/getPlayList/VideoIDS/%s' % video_id | 
					
						
							| 
									
										
										
										
											2015-09-01 00:19:58 +08:00
										 |  |  |         if video_password: | 
					
						
							| 
									
										
										
										
											2015-09-01 22:26:17 +06:00
										 |  |  |             basic_data_url += '?password=%s' % video_password | 
					
						
							| 
									
										
										
										
											2015-09-01 11:07:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         data1 = retrieve_data( | 
					
						
							|  |  |  |             basic_data_url, | 
					
						
							|  |  |  |             'Downloading JSON metadata 1') | 
					
						
							| 
									
										
										
										
											2015-06-16 00:06:23 +08:00
										 |  |  |         data2 = retrieve_data( | 
					
						
							|  |  |  |             'http://v.youku.com/player/getPlayList/VideoIDS/%s/Pf/4/ctype/12/ev/1' % video_id, | 
					
						
							|  |  |  |             'Downloading JSON metadata 2') | 
					
						
							| 
									
										
										
										
											2014-09-25 09:58:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |         error_code = data1.get('error_code') | 
					
						
							| 
									
										
										
										
											2014-09-25 09:58:09 +02:00
										 |  |  |         if error_code: | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |             error = data1.get('error') | 
					
						
							| 
									
										
										
										
											2015-06-15 23:54:55 +08:00
										 |  |  |             if error is not None and '因版权原因无法观看此视频' in error: | 
					
						
							|  |  |  |                 raise ExtractorError( | 
					
						
							|  |  |  |                     'Youku said: Sorry, this video is available in China only', expected=True) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 msg = 'Youku server reported error %i' % error_code | 
					
						
							|  |  |  |                 if error is not None: | 
					
						
							|  |  |  |                     msg += ': ' + error | 
					
						
							|  |  |  |                 raise ExtractorError(msg) | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         title = data1['title'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # generate video_urls_dict | 
					
						
							|  |  |  |         video_urls_dict = self.construct_video_urls(data1, data2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # construct info | 
					
						
							| 
									
										
										
										
											2015-06-16 14:41:52 +02:00
										 |  |  |         entries = [{ | 
					
						
							|  |  |  |             'id': '%s_part%d' % (video_id, i + 1), | 
					
						
							|  |  |  |             'title': title, | 
					
						
							|  |  |  |             'formats': [], | 
					
						
							|  |  |  |             # some formats are not available for all parts, we have to detect | 
					
						
							|  |  |  |             # which one has all | 
					
						
							|  |  |  |         } for i in range(max(len(v) for v in data1['segs'].values()))] | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |         for fm in data1['streamtypes']: | 
					
						
							|  |  |  |             video_urls = video_urls_dict[fm] | 
					
						
							| 
									
										
										
										
											2015-06-16 14:41:52 +02:00
										 |  |  |             for video_url, seg, entry in zip(video_urls, data1['segs'][fm], entries): | 
					
						
							|  |  |  |                 entry['formats'].append({ | 
					
						
							|  |  |  |                     'url': video_url, | 
					
						
							| 
									
										
										
										
											2015-06-16 00:15:09 +08:00
										 |  |  |                     'format_id': self.get_format_name(fm), | 
					
						
							|  |  |  |                     'ext': self.parse_ext_l(fm), | 
					
						
							| 
									
										
										
										
											2015-06-16 14:41:52 +02:00
										 |  |  |                     'filesize': int(seg['size']), | 
					
						
							| 
									
										
										
										
											2015-06-16 00:15:09 +08:00
										 |  |  |                 }) | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-15 23:46:07 +08:00
										 |  |  |         return { | 
					
						
							|  |  |  |             '_type': 'multi_video', | 
					
						
							|  |  |  |             'id': video_id, | 
					
						
							|  |  |  |             'title': title, | 
					
						
							|  |  |  |             'entries': entries, | 
					
						
							|  |  |  |         } |