| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2016-06-08 23:45:46 +08:00
										 |  |  | import itertools | 
					
						
							| 
									
										
										
										
											2015-12-31 13:05:46 +08:00
										 |  |  | import random | 
					
						
							| 
									
										
										
										
											2016-06-08 23:45:46 +08:00
										 |  |  | import re | 
					
						
							| 
									
										
										
										
											2015-12-31 13:05:46 +08:00
										 |  |  | import string | 
					
						
							|  |  |  | import time | 
					
						
							| 
									
										
										
										
											2013-06-23 22:01:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | from .common import InfoExtractor | 
					
						
							| 
									
										
										
										
											2015-06-15 23:28:59 +08:00
										 |  |  | from ..compat import ( | 
					
						
							|  |  |  |     compat_ord, | 
					
						
							| 
									
										
										
										
											2017-04-15 01:25:32 +07:00
										 |  |  |     compat_str, | 
					
						
							|  |  |  |     compat_urllib_parse_urlencode, | 
					
						
							| 
									
										
										
										
											2015-11-21 22:18:17 +06:00
										 |  |  | ) | 
					
						
							|  |  |  | from ..utils import ( | 
					
						
							|  |  |  |     ExtractorError, | 
					
						
							| 
									
										
										
										
											2016-06-08 23:45:46 +08:00
										 |  |  |     get_element_by_attribute, | 
					
						
							| 
									
										
										
										
											2017-04-15 01:25:32 +07:00
										 |  |  |     try_get, | 
					
						
							| 
									
										
										
										
											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-12-12 14:48:46 +08:00
										 |  |  |         # MD5 is unstable | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |         'url': 'http://v.youku.com/v_show/id_XMTc1ODE5Njcy.html', | 
					
						
							|  |  |  |         '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-12-12 14:57:14 +08:00
										 |  |  |         'skip': 'Available in China only', | 
					
						
							| 
									
										
										
										
											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, | 
					
						
							| 
									
										
										
										
											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', | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-04-23 02:51:17 +08:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         # /play/get.json contains streams with "channel_type":"tail" | 
					
						
							|  |  |  |         'url': 'http://v.youku.com/v_show/id_XOTUxMzg4NDMy.html', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': 'XOTUxMzg4NDMy', | 
					
						
							|  |  |  |             'title': '我的世界☆明月庄主☆车震猎杀☆杀人艺术Minecraft', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'playlist_count': 6, | 
					
						
							| 
									
										
										
										
											2015-06-15 23:36:28 +08:00
										 |  |  |     }] | 
					
						
							| 
									
										
										
										
											2013-06-27 20:25:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-12 11:26:15 +08:00
										 |  |  |     def construct_video_urls(self, data): | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |         # 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-12-12 11:26:15 +08:00
										 |  |  |             b'becaf9be', base64.b64decode(data['security']['encrypt_string'].encode('ascii')) | 
					
						
							| 
									
										
										
										
											2015-06-15 23:28:59 +08:00
										 |  |  |         ).decode('ascii').split('_') | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # get oip | 
					
						
							| 
									
										
										
										
											2015-12-12 11:26:15 +08:00
										 |  |  |         oip = data['security']['ip'] | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         fileid_dict = {} | 
					
						
							| 
									
										
										
										
											2015-12-12 11:26:15 +08:00
										 |  |  |         for stream in data['stream']: | 
					
						
							| 
									
										
										
										
											2016-04-23 02:51:17 +08:00
										 |  |  |             if stream.get('channel_type') == 'tail': | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2015-12-11 17:48:40 +08:00
										 |  |  |             format = stream.get('stream_type') | 
					
						
							| 
									
										
										
										
											2017-04-15 01:25:32 +07:00
										 |  |  |             fileid = try_get( | 
					
						
							|  |  |  |                 stream, lambda x: x['segs'][0]['fileid'], | 
					
						
							|  |  |  |                 compat_str) or stream['stream_fileid'] | 
					
						
							| 
									
										
										
										
											2015-12-11 17:48:40 +08:00
										 |  |  |             fileid_dict[format] = fileid | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def get_fileid(format, n): | 
					
						
							| 
									
										
										
										
											2015-12-11 20:08:14 +08:00
										 |  |  |             number = hex(int(str(n), 10))[2:].upper() | 
					
						
							|  |  |  |             if len(number) == 1: | 
					
						
							|  |  |  |                 number = '0' + number | 
					
						
							|  |  |  |             streamfileids = fileid_dict[format] | 
					
						
							|  |  |  |             fileid = streamfileids[0:8] + number + streamfileids[10:] | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |             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 = {} | 
					
						
							| 
									
										
										
										
											2015-12-12 11:26:15 +08:00
										 |  |  |         for stream in data['stream']: | 
					
						
							| 
									
										
										
										
											2016-04-23 02:51:17 +08:00
										 |  |  |             if stream.get('channel_type') == 'tail': | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2015-12-11 17:48:40 +08:00
										 |  |  |             format = stream.get('stream_type') | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |             video_urls = [] | 
					
						
							| 
									
										
										
										
											2015-12-11 17:48:40 +08:00
										 |  |  |             for dt in stream['segs']: | 
					
						
							| 
									
										
										
										
											2015-12-11 19:18:14 +08:00
										 |  |  |                 n = str(stream['segs'].index(dt)) | 
					
						
							| 
									
										
										
										
											2015-05-29 10:13:09 +08:00
										 |  |  |                 param = { | 
					
						
							| 
									
										
										
										
											2015-12-11 17:48:40 +08:00
										 |  |  |                     'K': dt['key'], | 
					
						
							| 
									
										
										
										
											2015-05-29 10:13:09 +08:00
										 |  |  |                     'hd': self.get_hd(format), | 
					
						
							|  |  |  |                     'myp': 0, | 
					
						
							|  |  |  |                     '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-12-12 14:41:53 +08:00
										 |  |  |                     '_00' + \ | 
					
						
							| 
									
										
										
										
											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) + '?' + \ | 
					
						
							| 
									
										
										
										
											2016-03-26 01:46:57 +06: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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-31 13:05:46 +08:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def get_ysuid(): | 
					
						
							|  |  |  |         return '%d%s' % (int(time.time()), ''.join([ | 
					
						
							|  |  |  |             random.choice(string.ascii_letters) for i in range(3)])) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |     def get_hd(self, fm): | 
					
						
							|  |  |  |         hd_id_dict = { | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |             '3gp': '0', | 
					
						
							| 
									
										
										
										
											2015-12-11 17:48:40 +08:00
										 |  |  |             '3gphd': '1', | 
					
						
							| 
									
										
										
										
											2015-12-12 15:24:58 +08:00
										 |  |  |             'flv': '0', | 
					
						
							| 
									
										
										
										
											2015-12-12 10:44:21 +08:00
										 |  |  |             'flvhd': '0', | 
					
						
							| 
									
										
										
										
											2015-12-12 15:24:58 +08:00
										 |  |  |             'mp4': '1', | 
					
						
							| 
									
										
										
										
											2015-12-12 10:44:21 +08:00
										 |  |  |             'mp4hd': '1', | 
					
						
							| 
									
										
										
										
											2015-12-12 15:24:58 +08:00
										 |  |  |             'mp4hd2': '1', | 
					
						
							| 
									
										
										
										
											2015-12-12 15:49:19 +08:00
										 |  |  |             'mp4hd3': '1', | 
					
						
							| 
									
										
										
										
											2015-12-12 15:24:58 +08:00
										 |  |  |             'hd2': '2', | 
					
						
							|  |  |  |             'hd3': '3', | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |         return hd_id_dict[fm] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def parse_ext_l(self, fm): | 
					
						
							|  |  |  |         ext_dict = { | 
					
						
							| 
									
										
										
										
											2015-12-12 15:24:58 +08:00
										 |  |  |             '3gp': 'flv', | 
					
						
							|  |  |  |             '3gphd': 'mp4', | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |             'flv': 'flv', | 
					
						
							| 
									
										
										
										
											2015-12-12 15:24:58 +08:00
										 |  |  |             'flvhd': 'flv', | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |             'mp4': 'mp4', | 
					
						
							| 
									
										
										
										
											2015-12-11 19:18:14 +08:00
										 |  |  |             'mp4hd': 'mp4', | 
					
						
							| 
									
										
										
										
											2015-12-12 10:44:21 +08:00
										 |  |  |             'mp4hd2': 'flv', | 
					
						
							|  |  |  |             'mp4hd3': 'flv', | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |             'hd2': 'flv', | 
					
						
							|  |  |  |             'hd3': 'flv', | 
					
						
							| 
									
										
										
										
											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', | 
					
						
							| 
									
										
										
										
											2015-12-12 15:24:58 +08:00
										 |  |  |             'flvhd': 'h4', | 
					
						
							| 
									
										
										
										
											2015-06-15 22:41:24 +08:00
										 |  |  |             'mp4': 'h3', | 
					
						
							| 
									
										
										
										
											2015-12-12 10:44:21 +08:00
										 |  |  |             'mp4hd': 'h3', | 
					
						
							| 
									
										
										
										
											2015-12-12 15:24:58 +08:00
										 |  |  |             'mp4hd2': 'h4', | 
					
						
							| 
									
										
										
										
											2015-12-12 10:44:21 +08:00
										 |  |  |             'mp4hd3': 'h4', | 
					
						
							| 
									
										
										
										
											2015-12-12 15:24:58 +08:00
										 |  |  |             '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-12-31 13:05:46 +08:00
										 |  |  |         self._set_cookie('youku.com', '__ysuid', self.get_ysuid()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 00:06:23 +08:00
										 |  |  |         def retrieve_data(req_url, note): | 
					
						
							| 
									
										
										
										
											2015-12-10 21:42:12 +08:00
										 |  |  |             headers = { | 
					
						
							| 
									
										
										
										
											2015-12-12 14:41:53 +08:00
										 |  |  |                 'Referer': req_url, | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2016-07-03 23:23:48 +08:00
										 |  |  |             headers.update(self.geo_verification_headers()) | 
					
						
							| 
									
										
										
										
											2015-12-12 14:41:53 +08:00
										 |  |  |             self._set_cookie('youku.com', 'xreferrer', 'http://www.youku.com') | 
					
						
							| 
									
										
										
										
											2013-06-23 22:01:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-03 23:23:48 +08:00
										 |  |  |             raw_data = self._download_json(req_url, video_id, note=note, headers=headers) | 
					
						
							| 
									
										
										
										
											2015-12-10 21:42:12 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-11 17:48:40 +08:00
										 |  |  |             return raw_data['data'] | 
					
						
							| 
									
										
										
										
											2015-12-10 21:42:12 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-14 14:25:04 +06:00
										 |  |  |         video_password = self._downloader.params.get('videopassword') | 
					
						
							| 
									
										
										
										
											2015-09-01 00:19:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 00:06:23 +08:00
										 |  |  |         # request basic data | 
					
						
							| 
									
										
										
										
											2016-02-14 15:37:17 +06:00
										 |  |  |         basic_data_url = 'http://play.youku.com/play/get.json?vid=%s&ct=12' % video_id | 
					
						
							| 
									
										
										
										
											2015-09-01 00:19:58 +08:00
										 |  |  |         if video_password: | 
					
						
							| 
									
										
										
										
											2015-12-12 11:21:44 +08:00
										 |  |  |             basic_data_url += '&pwd=%s' % video_password | 
					
						
							| 
									
										
										
										
											2015-09-01 11:07:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-12 15:19:14 +08:00
										 |  |  |         data = retrieve_data(basic_data_url, 'Downloading JSON metadata') | 
					
						
							| 
									
										
										
										
											2014-09-25 09:58:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-12 11:26:15 +08:00
										 |  |  |         error = data.get('error') | 
					
						
							| 
									
										
										
										
											2015-12-12 11:21:44 +08:00
										 |  |  |         if error: | 
					
						
							|  |  |  |             error_note = error.get('note') | 
					
						
							|  |  |  |             if error_note is not None and '因版权原因无法观看此视频' in error_note: | 
					
						
							| 
									
										
										
										
											2015-06-15 23:54:55 +08:00
										 |  |  |                 raise ExtractorError( | 
					
						
							|  |  |  |                     'Youku said: Sorry, this video is available in China only', expected=True) | 
					
						
							| 
									
										
										
										
											2016-02-10 20:05:17 +06:00
										 |  |  |             elif error_note and '该视频被设为私密' in error_note: | 
					
						
							|  |  |  |                 raise ExtractorError( | 
					
						
							|  |  |  |                     'Youku said: Sorry, this video is private', expected=True) | 
					
						
							| 
									
										
										
										
											2015-06-15 23:54:55 +08:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2015-12-12 11:21:44 +08:00
										 |  |  |                 msg = 'Youku server reported error %i' % error.get('code') | 
					
						
							| 
									
										
										
										
											2015-12-17 12:51:50 +01:00
										 |  |  |                 if error_note is not None: | 
					
						
							| 
									
										
										
										
											2015-12-12 11:21:44 +08:00
										 |  |  |                     msg += ': ' + error_note | 
					
						
							| 
									
										
										
										
											2015-06-15 23:54:55 +08:00
										 |  |  |                 raise ExtractorError(msg) | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-12 14:41:53 +08:00
										 |  |  |         # get video title | 
					
						
							| 
									
										
										
										
											2015-12-12 11:26:15 +08:00
										 |  |  |         title = data['video']['title'] | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # generate video_urls_dict | 
					
						
							| 
									
										
										
										
											2015-12-12 11:26:15 +08:00
										 |  |  |         video_urls_dict = self.construct_video_urls(data) | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # 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 | 
					
						
							| 
									
										
										
										
											2015-12-12 11:26:15 +08:00
										 |  |  |         } for i in range(max(len(v.get('segs')) for v in data['stream']))] | 
					
						
							|  |  |  |         for stream in data['stream']: | 
					
						
							| 
									
										
										
										
											2016-04-23 02:51:17 +08:00
										 |  |  |             if stream.get('channel_type') == 'tail': | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2015-12-11 17:48:40 +08:00
										 |  |  |             fm = stream.get('stream_type') | 
					
						
							| 
									
										
										
										
											2015-05-28 17:00:09 +08:00
										 |  |  |             video_urls = video_urls_dict[fm] | 
					
						
							| 
									
										
										
										
											2015-12-11 17:48:40 +08:00
										 |  |  |             for video_url, seg, entry in zip(video_urls, stream['segs'], entries): | 
					
						
							| 
									
										
										
										
											2015-06-16 14:41:52 +02:00
										 |  |  |                 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']), | 
					
						
							| 
									
										
										
										
											2016-05-29 13:54:05 +08:00
										 |  |  |                     'width': stream.get('width'), | 
					
						
							|  |  |  |                     'height': stream.get('height'), | 
					
						
							| 
									
										
										
										
											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, | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-06-08 23:45:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class YoukuShowIE(InfoExtractor): | 
					
						
							|  |  |  |     _VALID_URL = r'https?://(?:www\.)?youku\.com/show_page/id_(?P<id>[0-9a-z]+)\.html' | 
					
						
							|  |  |  |     IE_NAME = 'youku:show' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _TEST = { | 
					
						
							|  |  |  |         'url': 'http://www.youku.com/show_page/id_zc7c670be07ff11e48b3f.html', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': 'zc7c670be07ff11e48b3f', | 
					
						
							|  |  |  |             'title': '花千骨 未删减版', | 
					
						
							|  |  |  |             'description': 'md5:578d4f2145ae3f9128d9d4d863312910', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'playlist_count': 50, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _PAGE_SIZE = 40 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _find_videos_in_page(self, webpage): | 
					
						
							|  |  |  |         videos = re.findall( | 
					
						
							|  |  |  |             r'<li><a[^>]+href="(?P<url>https?://v\.youku\.com/[^"]+)"[^>]+title="(?P<title>[^"]+)"', webpage) | 
					
						
							|  |  |  |         return [ | 
					
						
							|  |  |  |             self.url_result(video_url, YoukuIE.ie_key(), title) | 
					
						
							|  |  |  |             for video_url, title in videos] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _real_extract(self, url): | 
					
						
							|  |  |  |         show_id = self._match_id(url) | 
					
						
							|  |  |  |         webpage = self._download_webpage(url, show_id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         entries = self._find_videos_in_page(webpage) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         playlist_title = self._html_search_regex( | 
					
						
							|  |  |  |             r'<span[^>]+class="name">([^<]+)</span>', webpage, 'playlist title', fatal=False) | 
					
						
							|  |  |  |         detail_div = get_element_by_attribute('class', 'detail', webpage) or '' | 
					
						
							|  |  |  |         playlist_description = self._html_search_regex( | 
					
						
							|  |  |  |             r'<span[^>]+style="display:none"[^>]*>([^<]+)</span>', | 
					
						
							|  |  |  |             detail_div, 'playlist description', fatal=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for idx in itertools.count(1): | 
					
						
							|  |  |  |             episodes_page = self._download_webpage( | 
					
						
							|  |  |  |                 'http://www.youku.com/show_episode/id_%s.html' % show_id, | 
					
						
							|  |  |  |                 show_id, query={'divid': 'reload_%d' % (idx * self._PAGE_SIZE + 1)}, | 
					
						
							|  |  |  |                 note='Downloading episodes page %d' % idx) | 
					
						
							|  |  |  |             new_entries = self._find_videos_in_page(episodes_page) | 
					
						
							|  |  |  |             entries.extend(new_entries) | 
					
						
							|  |  |  |             if len(new_entries) < self._PAGE_SIZE: | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return self.playlist_result(entries, show_id, playlist_title, playlist_description) |