from __future__ import unicode_literals
import base64
-import json
from .common import InfoExtractor
from ..compat import (
'''
_TESTS = [{
+ # MD5 is unstable
'url': 'http://v.youku.com/v_show/id_XMTc1ODE5Njcy.html',
- 'md5': '5f3af4192eabacc4501508d54a8cabd7',
'info_dict': {
'id': 'XMTc1ODE5Njcy_part1',
'title': '★Smile﹗♡ Git Fresh -Booty Music舞蹈.',
'title': '武媚娘传奇 85',
},
'playlist_count': 11,
+ 'skip': 'Available in China only',
}, {
'url': 'http://v.youku.com/v_show/id_XMTI1OTczNDM5Mg==.html',
'info_dict': {
'title': '花千骨 04',
},
'playlist_count': 13,
- 'skip': 'Available in China only',
}, {
'url': 'http://v.youku.com/v_show/id_XNjA1NzA2Njgw.html',
'note': 'Video protected with password',
},
}]
- def construct_video_urls(self, data1, data2):
+ def construct_video_urls(self, data):
# get sid, token
def yk_t(s1, s2):
ls = list(range(256))
return bytes(s)
sid, token = yk_t(
- b'becaf9be', base64.b64decode(data2['security']['encrypt_string'].encode('ascii'))
+ b'becaf9be', base64.b64decode(data['security']['encrypt_string'].encode('ascii'))
).decode('ascii').split('_')
# get oip
- oip = data1['security']['ip']
-
- # get fileid
- string_ls = list(
- 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/\:._-1234567890')
+ oip = data['security']['ip']
fileid_dict = {}
- for stream in data1['stream']:
+ for stream in data['stream']:
format = stream.get('stream_type')
fileid = stream['stream_fileid']
fileid_dict[format] = fileid
# generate video_urls
video_urls_dict = {}
- for stream in data1['stream']:
+ for stream in data['stream']:
format = stream.get('stream_type')
video_urls = []
for dt in stream['segs']:
- #n = str(int(dt['size']))
n = str(stream['segs'].index(dt))
param = {
'K': dt['key'],
'hd': self.get_hd(format),
'myp': 0,
- #'ts': dt['total_milliseconds_video'],
'ypp': 0,
'ctype': 12,
'ev': 1,
video_url = \
'http://k.youku.com/player/getFlvPath/' + \
'sid/' + sid + \
- '_00'+ \
+ '_00' + \
'/st/' + self.parse_ext_l(format) + \
'/fileid/' + get_fileid(format, n) + '?' + \
compat_urllib_parse.urlencode(param)
def get_hd(self, fm):
hd_id_dict = {
- 'flv': '0',
- 'mp4': '1',
- 'hd2': '2',
- 'hd3': '3',
'3gp': '0',
'3gphd': '1',
+ 'flv': '0',
'flvhd': '0',
+ 'mp4': '1',
'mp4hd': '1',
- 'mp4hd2': '1'
+ 'mp4hd2': '1',
+ 'hd2': '2',
+ 'hd3': '3',
}
return hd_id_dict[fm]
def parse_ext_l(self, fm):
ext_dict = {
+ '3gp': 'flv',
+ '3gphd': 'mp4',
'flv': 'flv',
+ 'flvhd': 'flv',
'mp4': 'mp4',
'mp4hd': 'mp4',
'mp4hd2': 'flv',
'mp4hd3': 'flv',
'hd2': 'flv',
'hd3': 'flv',
- '3gp': 'flv',
- '3gphd': 'mp4',
- 'flvhd': 'flv'
}
return ext_dict[fm]
_dict = {
'3gp': 'h6',
'3gphd': 'h5',
- 'flvhd': 'h4',
'flv': 'h4',
+ 'flvhd': 'h4',
'mp4': 'h3',
- 'hd2': 'h2',
- 'hd3': 'h1',
'mp4hd': 'h3',
+ 'mp4hd2': 'h4',
'mp4hd3': 'h4',
- 'mp4hd2': 'h4'
+ 'hd2': 'h2',
+ 'hd3': 'h1',
}
return _dict[fm]
video_id = self._match_id(url)
def retrieve_data(req_url, note):
-
-
headers = {
- 'Referer': req_url,
- }
- self._set_cookie('youku.com','xreferrer','http://www.youku.com')
- req = sanitized_Request(req_url,headers=headers)
+ 'Referer': req_url,
+ }
+ self._set_cookie('youku.com', 'xreferrer', 'http://www.youku.com')
+ req = sanitized_Request(req_url, headers=headers)
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)
- js = json.dumps(raw_data)
return raw_data['data']
-
video_password = self._downloader.params.get('videopassword', None)
# request basic data
basic_data_url = "http://play.youku.com/play/get.json?vid=%s&ct=12" % video_id
if video_password:
- basic_data_url += '?password=%s' % video_password
+ basic_data_url += '&pwd=%s' % video_password
- data1 = retrieve_data(
- basic_data_url,
- 'Downloading JSON metadata 1')
- data2 = retrieve_data(
- "http://play.youku.com/play/get.json?vid=%s&ct=12" % video_id,
- 'Downloading JSON metadata 2')
+ data = retrieve_data(basic_data_url, 'Downloading JSON metadata')
- error_code = data1.get('error_code')
- if error_code:
- error = data1.get('error')
- if error is not None and '因版权原因无法观看此视频' in error:
+ error = data.get('error')
+ if error:
+ error_note = error.get('note')
+ if error_note is not None and '因版权原因无法观看此视频' in error_note:
raise ExtractorError(
'Youku said: Sorry, this video is available in China only', expected=True)
else:
- msg = 'Youku server reported error %i' % error_code
+ msg = 'Youku server reported error %i' % error.get('code')
if error is not None:
- msg += ': ' + error
+ msg += ': ' + error_note
raise ExtractorError(msg)
- #get video title
- title = data1['video']['title']
-
+ # get video title
+ title = data['video']['title']
# generate video_urls_dict
- video_urls_dict = self.construct_video_urls(data1, data2)
+ video_urls_dict = self.construct_video_urls(data)
# construct info
entries = [{
'formats': [],
# some formats are not available for all parts, we have to detect
# which one has all
- } for i in range(max(len(v.get('segs')) for v in data1['stream']))]
- for stream in data1['stream']:
+ } for i in range(max(len(v.get('segs')) for v in data['stream']))]
+ for stream in data['stream']:
fm = stream.get('stream_type')
video_urls = video_urls_dict[fm]
for video_url, seg, entry in zip(video_urls, stream['segs'], entries):