]> jfr.im git - yt-dlp.git/blame - test/test_YoutubeDL.py
Completely change project name to yt-dlp (#85)
[yt-dlp.git] / test / test_YoutubeDL.py
CommitLineData
e028d0d1 1#!/usr/bin/env python
db13c16e 2# coding: utf-8
e028d0d1 3
89087418
PH
4from __future__ import unicode_literals
5
5d254f77
PH
6# Allow direct execution
7import os
e028d0d1
JMF
8import sys
9import unittest
5d254f77 10sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
e028d0d1 11
0217c783
PH
12import copy
13
c57f7757 14from test.helper import FakeYDL, assertRegexpMatches
7a5c1cfe
P
15from yt_dlp import YoutubeDL
16from yt_dlp.compat import compat_str, compat_urllib_error
17from yt_dlp.extractor import YoutubeIE
18from yt_dlp.extractor.common import InfoExtractor
19from yt_dlp.postprocessor.common import PostProcessor
20from yt_dlp.utils import ExtractorError, match_filter_func
e028d0d1 21
8508557e
JMF
22TEST_URL = 'http://localhost/sample.mp4'
23
e028d0d1
JMF
24
25class YDL(FakeYDL):
f4d96df0
PH
26 def __init__(self, *args, **kwargs):
27 super(YDL, self).__init__(*args, **kwargs)
e028d0d1 28 self.downloaded_info_dicts = []
f4d96df0 29 self.msgs = []
5d254f77 30
e028d0d1
JMF
31 def process_info(self, info_dict):
32 self.downloaded_info_dicts.append(info_dict)
33
f4d96df0
PH
34 def to_screen(self, msg):
35 self.msgs.append(msg)
36
5d254f77 37
3537b93d
PH
38def _make_result(formats, **kwargs):
39 res = {
40 'formats': formats,
41 'id': 'testid',
42 'title': 'testttitle',
43 'extractor': 'testex',
0396806f 44 'extractor_key': 'TestEx',
732044af 45 'webpage_url': 'http://example.com/watch?v=shenanigans',
3537b93d
PH
46 }
47 res.update(**kwargs)
48 return res
49
50
e028d0d1
JMF
51class TestFormatSelection(unittest.TestCase):
52 def test_prefer_free_formats(self):
53 # Same resolution => download webm
54 ydl = YDL()
55 ydl.params['prefer_free_formats'] = True
5d254f77 56 formats = [
8508557e
JMF
57 {'ext': 'webm', 'height': 460, 'url': TEST_URL},
58 {'ext': 'mp4', 'height': 460, 'url': TEST_URL},
5d254f77 59 ]
3537b93d 60 info_dict = _make_result(formats)
3d4a70b8
PH
61 yie = YoutubeIE(ydl)
62 yie._sort_formats(info_dict['formats'])
e028d0d1
JMF
63 ydl.process_ie_result(info_dict)
64 downloaded = ydl.downloaded_info_dicts[0]
89087418 65 self.assertEqual(downloaded['ext'], 'webm')
e028d0d1
JMF
66
67 # Different resolution => download best quality (mp4)
68 ydl = YDL()
69 ydl.params['prefer_free_formats'] = True
5d254f77 70 formats = [
8508557e
JMF
71 {'ext': 'webm', 'height': 720, 'url': TEST_URL},
72 {'ext': 'mp4', 'height': 1080, 'url': TEST_URL},
5d254f77 73 ]
89087418 74 info_dict['formats'] = formats
3d4a70b8
PH
75 yie = YoutubeIE(ydl)
76 yie._sort_formats(info_dict['formats'])
e028d0d1
JMF
77 ydl.process_ie_result(info_dict)
78 downloaded = ydl.downloaded_info_dicts[0]
89087418 79 self.assertEqual(downloaded['ext'], 'mp4')
e028d0d1 80
5d0c5371 81 # No prefer_free_formats => prefer mp4 and webm
e028d0d1
JMF
82 ydl = YDL()
83 ydl.params['prefer_free_formats'] = False
5d254f77 84 formats = [
8508557e
JMF
85 {'ext': 'webm', 'height': 720, 'url': TEST_URL},
86 {'ext': 'mp4', 'height': 720, 'url': TEST_URL},
87 {'ext': 'flv', 'height': 720, 'url': TEST_URL},
5d254f77 88 ]
89087418 89 info_dict['formats'] = formats
3d4a70b8
PH
90 yie = YoutubeIE(ydl)
91 yie._sort_formats(info_dict['formats'])
92 ydl.process_ie_result(info_dict)
93 downloaded = ydl.downloaded_info_dicts[0]
89087418 94 self.assertEqual(downloaded['ext'], 'mp4')
3d4a70b8
PH
95
96 ydl = YDL()
97 ydl.params['prefer_free_formats'] = False
98 formats = [
8508557e
JMF
99 {'ext': 'flv', 'height': 720, 'url': TEST_URL},
100 {'ext': 'webm', 'height': 720, 'url': TEST_URL},
3d4a70b8 101 ]
89087418 102 info_dict['formats'] = formats
3d4a70b8
PH
103 yie = YoutubeIE(ydl)
104 yie._sort_formats(info_dict['formats'])
e028d0d1
JMF
105 ydl.process_ie_result(info_dict)
106 downloaded = ydl.downloaded_info_dicts[0]
5d0c5371 107 self.assertEqual(downloaded['ext'], 'webm')
e028d0d1 108
a9c58ad9
JMF
109 def test_format_selection(self):
110 formats = [
8508557e 111 {'format_id': '35', 'ext': 'mp4', 'preference': 1, 'url': TEST_URL},
232541df 112 {'format_id': 'example-with-dashes', 'ext': 'webm', 'preference': 1, 'url': TEST_URL},
8508557e
JMF
113 {'format_id': '45', 'ext': 'webm', 'preference': 2, 'url': TEST_URL},
114 {'format_id': '47', 'ext': 'webm', 'preference': 3, 'url': TEST_URL},
115 {'format_id': '2', 'ext': 'flv', 'preference': 4, 'url': TEST_URL},
a9c58ad9 116 ]
3537b93d 117 info_dict = _make_result(formats)
a9c58ad9 118
89087418 119 ydl = YDL({'format': '20/47'})
8e3e0322 120 ydl.process_ie_result(info_dict.copy())
a9c58ad9 121 downloaded = ydl.downloaded_info_dicts[0]
89087418 122 self.assertEqual(downloaded['format_id'], '47')
a9c58ad9 123
89087418 124 ydl = YDL({'format': '20/71/worst'})
8e3e0322 125 ydl.process_ie_result(info_dict.copy())
a9c58ad9 126 downloaded = ydl.downloaded_info_dicts[0]
89087418 127 self.assertEqual(downloaded['format_id'], '35')
a9c58ad9
JMF
128
129 ydl = YDL()
8e3e0322 130 ydl.process_ie_result(info_dict.copy())
a9c58ad9 131 downloaded = ydl.downloaded_info_dicts[0]
89087418 132 self.assertEqual(downloaded['format_id'], '2')
a9c58ad9 133
89087418 134 ydl = YDL({'format': 'webm/mp4'})
8e3e0322 135 ydl.process_ie_result(info_dict.copy())
49e86983 136 downloaded = ydl.downloaded_info_dicts[0]
89087418 137 self.assertEqual(downloaded['format_id'], '47')
49e86983 138
89087418 139 ydl = YDL({'format': '3gp/40/mp4'})
8e3e0322 140 ydl.process_ie_result(info_dict.copy())
49e86983 141 downloaded = ydl.downloaded_info_dicts[0]
89087418 142 self.assertEqual(downloaded['format_id'], '35')
49e86983 143
232541df
JMF
144 ydl = YDL({'format': 'example-with-dashes'})
145 ydl.process_ie_result(info_dict.copy())
146 downloaded = ydl.downloaded_info_dicts[0]
147 self.assertEqual(downloaded['format_id'], 'example-with-dashes')
148
ba7678f9
PH
149 def test_format_selection_audio(self):
150 formats = [
8508557e
JMF
151 {'format_id': 'audio-low', 'ext': 'webm', 'preference': 1, 'vcodec': 'none', 'url': TEST_URL},
152 {'format_id': 'audio-mid', 'ext': 'webm', 'preference': 2, 'vcodec': 'none', 'url': TEST_URL},
153 {'format_id': 'audio-high', 'ext': 'flv', 'preference': 3, 'vcodec': 'none', 'url': TEST_URL},
154 {'format_id': 'vid', 'ext': 'mp4', 'preference': 4, 'url': TEST_URL},
ba7678f9 155 ]
3537b93d 156 info_dict = _make_result(formats)
ba7678f9 157
89087418 158 ydl = YDL({'format': 'bestaudio'})
ba7678f9
PH
159 ydl.process_ie_result(info_dict.copy())
160 downloaded = ydl.downloaded_info_dicts[0]
89087418 161 self.assertEqual(downloaded['format_id'], 'audio-high')
ba7678f9 162
89087418 163 ydl = YDL({'format': 'worstaudio'})
ba7678f9
PH
164 ydl.process_ie_result(info_dict.copy())
165 downloaded = ydl.downloaded_info_dicts[0]
89087418 166 self.assertEqual(downloaded['format_id'], 'audio-low')
ba7678f9
PH
167
168 formats = [
8508557e
JMF
169 {'format_id': 'vid-low', 'ext': 'mp4', 'preference': 1, 'url': TEST_URL},
170 {'format_id': 'vid-high', 'ext': 'mp4', 'preference': 2, 'url': TEST_URL},
ba7678f9 171 ]
3537b93d 172 info_dict = _make_result(formats)
ba7678f9 173
89087418 174 ydl = YDL({'format': 'bestaudio/worstaudio/best'})
ba7678f9
PH
175 ydl.process_ie_result(info_dict.copy())
176 downloaded = ydl.downloaded_info_dicts[0]
89087418 177 self.assertEqual(downloaded['format_id'], 'vid-high')
ba7678f9 178
0217c783
PH
179 def test_format_selection_audio_exts(self):
180 formats = [
181 {'format_id': 'mp3-64', 'ext': 'mp3', 'abr': 64, 'url': 'http://_', 'vcodec': 'none'},
182 {'format_id': 'ogg-64', 'ext': 'ogg', 'abr': 64, 'url': 'http://_', 'vcodec': 'none'},
183 {'format_id': 'aac-64', 'ext': 'aac', 'abr': 64, 'url': 'http://_', 'vcodec': 'none'},
184 {'format_id': 'mp3-32', 'ext': 'mp3', 'abr': 32, 'url': 'http://_', 'vcodec': 'none'},
185 {'format_id': 'aac-32', 'ext': 'aac', 'abr': 32, 'url': 'http://_', 'vcodec': 'none'},
186 ]
187
188 info_dict = _make_result(formats)
189 ydl = YDL({'format': 'best'})
190 ie = YoutubeIE(ydl)
191 ie._sort_formats(info_dict['formats'])
192 ydl.process_ie_result(copy.deepcopy(info_dict))
193 downloaded = ydl.downloaded_info_dicts[0]
194 self.assertEqual(downloaded['format_id'], 'aac-64')
195
196 ydl = YDL({'format': 'mp3'})
197 ie = YoutubeIE(ydl)
198 ie._sort_formats(info_dict['formats'])
199 ydl.process_ie_result(copy.deepcopy(info_dict))
200 downloaded = ydl.downloaded_info_dicts[0]
201 self.assertEqual(downloaded['format_id'], 'mp3-64')
202
203 ydl = YDL({'prefer_free_formats': True})
204 ie = YoutubeIE(ydl)
205 ie._sort_formats(info_dict['formats'])
206 ydl.process_ie_result(copy.deepcopy(info_dict))
207 downloaded = ydl.downloaded_info_dicts[0]
208 self.assertEqual(downloaded['format_id'], 'ogg-64')
209
bc6d5978
JMF
210 def test_format_selection_video(self):
211 formats = [
8508557e
JMF
212 {'format_id': 'dash-video-low', 'ext': 'mp4', 'preference': 1, 'acodec': 'none', 'url': TEST_URL},
213 {'format_id': 'dash-video-high', 'ext': 'mp4', 'preference': 2, 'acodec': 'none', 'url': TEST_URL},
214 {'format_id': 'vid', 'ext': 'mp4', 'preference': 3, 'url': TEST_URL},
bc6d5978 215 ]
3537b93d 216 info_dict = _make_result(formats)
bc6d5978
JMF
217
218 ydl = YDL({'format': 'bestvideo'})
219 ydl.process_ie_result(info_dict.copy())
220 downloaded = ydl.downloaded_info_dicts[0]
221 self.assertEqual(downloaded['format_id'], 'dash-video-high')
222
223 ydl = YDL({'format': 'worstvideo'})
224 ydl.process_ie_result(info_dict.copy())
225 downloaded = ydl.downloaded_info_dicts[0]
226 self.assertEqual(downloaded['format_id'], 'dash-video-low')
227
4c3b16d5
S
228 ydl = YDL({'format': 'bestvideo[format_id^=dash][format_id$=low]'})
229 ydl.process_ie_result(info_dict.copy())
230 downloaded = ydl.downloaded_info_dicts[0]
231 self.assertEqual(downloaded['format_id'], 'dash-video-low')
232
b913348d 233 formats = [
234 {'format_id': 'vid-vcodec-dot', 'ext': 'mp4', 'preference': 1, 'vcodec': 'avc1.123456', 'acodec': 'none', 'url': TEST_URL},
235 ]
236 info_dict = _make_result(formats)
237
238 ydl = YDL({'format': 'bestvideo[vcodec=avc1.123456]'})
239 ydl.process_ie_result(info_dict.copy())
240 downloaded = ydl.downloaded_info_dicts[0]
241 self.assertEqual(downloaded['format_id'], 'vid-vcodec-dot')
242
2cc779f4
S
243 def test_format_selection_string_ops(self):
244 formats = [
245 {'format_id': 'abc-cba', 'ext': 'mp4', 'url': TEST_URL},
e118a879 246 {'format_id': 'zxc-cxz', 'ext': 'webm', 'url': TEST_URL},
2cc779f4
S
247 ]
248 info_dict = _make_result(formats)
249
250 # equals (=)
251 ydl = YDL({'format': '[format_id=abc-cba]'})
252 ydl.process_ie_result(info_dict.copy())
253 downloaded = ydl.downloaded_info_dicts[0]
254 self.assertEqual(downloaded['format_id'], 'abc-cba')
255
256 # does not equal (!=)
257 ydl = YDL({'format': '[format_id!=abc-cba]'})
e118a879
S
258 ydl.process_ie_result(info_dict.copy())
259 downloaded = ydl.downloaded_info_dicts[0]
260 self.assertEqual(downloaded['format_id'], 'zxc-cxz')
261
262 ydl = YDL({'format': '[format_id!=abc-cba][format_id!=zxc-cxz]'})
2cc779f4
S
263 self.assertRaises(ExtractorError, ydl.process_ie_result, info_dict.copy())
264
265 # starts with (^=)
266 ydl = YDL({'format': '[format_id^=abc]'})
267 ydl.process_ie_result(info_dict.copy())
268 downloaded = ydl.downloaded_info_dicts[0]
269 self.assertEqual(downloaded['format_id'], 'abc-cba')
270
271 # does not start with (!^=)
e118a879
S
272 ydl = YDL({'format': '[format_id!^=abc]'})
273 ydl.process_ie_result(info_dict.copy())
274 downloaded = ydl.downloaded_info_dicts[0]
275 self.assertEqual(downloaded['format_id'], 'zxc-cxz')
276
277 ydl = YDL({'format': '[format_id!^=abc][format_id!^=zxc]'})
2cc779f4
S
278 self.assertRaises(ExtractorError, ydl.process_ie_result, info_dict.copy())
279
280 # ends with ($=)
281 ydl = YDL({'format': '[format_id$=cba]'})
282 ydl.process_ie_result(info_dict.copy())
283 downloaded = ydl.downloaded_info_dicts[0]
284 self.assertEqual(downloaded['format_id'], 'abc-cba')
285
286 # does not end with (!$=)
e118a879
S
287 ydl = YDL({'format': '[format_id!$=cba]'})
288 ydl.process_ie_result(info_dict.copy())
289 downloaded = ydl.downloaded_info_dicts[0]
290 self.assertEqual(downloaded['format_id'], 'zxc-cxz')
291
292 ydl = YDL({'format': '[format_id!$=cba][format_id!$=cxz]'})
2cc779f4
S
293 self.assertRaises(ExtractorError, ydl.process_ie_result, info_dict.copy())
294
295 # contains (*=)
e118a879 296 ydl = YDL({'format': '[format_id*=bc-cb]'})
2cc779f4
S
297 ydl.process_ie_result(info_dict.copy())
298 downloaded = ydl.downloaded_info_dicts[0]
299 self.assertEqual(downloaded['format_id'], 'abc-cba')
300
301 # does not contain (!*=)
e118a879
S
302 ydl = YDL({'format': '[format_id!*=bc-cb]'})
303 ydl.process_ie_result(info_dict.copy())
304 downloaded = ydl.downloaded_info_dicts[0]
305 self.assertEqual(downloaded['format_id'], 'zxc-cxz')
306
307 ydl = YDL({'format': '[format_id!*=abc][format_id!*=zxc]'})
308 self.assertRaises(ExtractorError, ydl.process_ie_result, info_dict.copy())
309
2cc779f4
S
310 ydl = YDL({'format': '[format_id!*=-]'})
311 self.assertRaises(ExtractorError, ydl.process_ie_result, info_dict.copy())
312
3d4a70b8 313 def test_youtube_format_selection(self):
5d0c5371 314 return
315 # disabled for now - this needs some changes
316
3d4a70b8 317 order = [
86bf2905 318 '38', '37', '46', '22', '45', '35', '44', '18', '34', '43', '6', '5', '17', '36', '13',
3d4a70b8
PH
319 # Apple HTTP Live Streaming
320 '96', '95', '94', '93', '92', '132', '151',
321 # 3D
322 '85', '84', '102', '83', '101', '82', '100',
323 # Dash video
c11125f9 324 '137', '248', '136', '247', '135', '246',
3d4a70b8
PH
325 '245', '244', '134', '243', '133', '242', '160',
326 # Dash audio
a053c349 327 '141', '172', '140', '171', '139',
3d4a70b8
PH
328 ]
329
67134eab
JMF
330 def format_info(f_id):
331 info = YoutubeIE._formats[f_id].copy()
1df41411 332
91cb6b50 333 # XXX: In real cases InfoExtractor._parse_mpd_formats() fills up 'acodec'
1df41411
YCH
334 # and 'vcodec', while in tests such information is incomplete since
335 # commit a6c2c24479e5f4827ceb06f64d855329c0a6f593
336 # test_YoutubeDL.test_youtube_format_selection is broken without
337 # this fix
338 if 'acodec' in info and 'vcodec' not in info:
339 info['vcodec'] = 'none'
340 elif 'vcodec' in info and 'acodec' not in info:
341 info['acodec'] = 'none'
342
67134eab
JMF
343 info['format_id'] = f_id
344 info['url'] = 'url:' + f_id
345 return info
346 formats_order = [format_info(f_id) for f_id in order]
347
348 info_dict = _make_result(list(formats_order), extractor='youtube')
349 ydl = YDL({'format': 'bestvideo+bestaudio'})
350 yie = YoutubeIE(ydl)
351 yie._sort_formats(info_dict['formats'])
352 ydl.process_ie_result(info_dict)
353 downloaded = ydl.downloaded_info_dicts[0]
5d0c5371 354 self.assertEqual(downloaded['format_id'], '248+172')
67134eab 355 self.assertEqual(downloaded['ext'], 'mp4')
3d4a70b8 356
cf2ac6df
JMF
357 info_dict = _make_result(list(formats_order), extractor='youtube')
358 ydl = YDL({'format': 'bestvideo[height>=999999]+bestaudio/best'})
359 yie = YoutubeIE(ydl)
360 yie._sort_formats(info_dict['formats'])
361 ydl.process_ie_result(info_dict)
362 downloaded = ydl.downloaded_info_dicts[0]
363 self.assertEqual(downloaded['format_id'], '38')
364
f5f4a27a
JMF
365 info_dict = _make_result(list(formats_order), extractor='youtube')
366 ydl = YDL({'format': 'bestvideo/best,bestaudio'})
367 yie = YoutubeIE(ydl)
368 yie._sort_formats(info_dict['formats'])
369 ydl.process_ie_result(info_dict)
370 downloaded_ids = [info['format_id'] for info in ydl.downloaded_info_dicts]
371 self.assertEqual(downloaded_ids, ['137', '141'])
372
0130afb7
JMF
373 info_dict = _make_result(list(formats_order), extractor='youtube')
374 ydl = YDL({'format': '(bestvideo[ext=mp4],bestvideo[ext=webm])+bestaudio'})
375 yie = YoutubeIE(ydl)
376 yie._sort_formats(info_dict['formats'])
377 ydl.process_ie_result(info_dict)
378 downloaded_ids = [info['format_id'] for info in ydl.downloaded_info_dicts]
379 self.assertEqual(downloaded_ids, ['137+141', '248+141'])
380
381 info_dict = _make_result(list(formats_order), extractor='youtube')
382 ydl = YDL({'format': '(bestvideo[ext=mp4],bestvideo[ext=webm])[height<=720]+bestaudio'})
383 yie = YoutubeIE(ydl)
384 yie._sort_formats(info_dict['formats'])
385 ydl.process_ie_result(info_dict)
386 downloaded_ids = [info['format_id'] for info in ydl.downloaded_info_dicts]
387 self.assertEqual(downloaded_ids, ['136+141', '247+141'])
388
389 info_dict = _make_result(list(formats_order), extractor='youtube')
390 ydl = YDL({'format': '(bestvideo[ext=none]/bestvideo[ext=webm])+bestaudio'})
391 yie = YoutubeIE(ydl)
392 yie._sort_formats(info_dict['formats'])
393 ydl.process_ie_result(info_dict)
394 downloaded_ids = [info['format_id'] for info in ydl.downloaded_info_dicts]
395 self.assertEqual(downloaded_ids, ['248+141'])
396
67134eab 397 for f1, f2 in zip(formats_order, formats_order[1:]):
3537b93d 398 info_dict = _make_result([f1, f2], extractor='youtube')
8dd54188 399 ydl = YDL({'format': 'best/bestvideo'})
3d4a70b8
PH
400 yie = YoutubeIE(ydl)
401 yie._sort_formats(info_dict['formats'])
402 ydl.process_ie_result(info_dict)
403 downloaded = ydl.downloaded_info_dicts[0]
67134eab 404 self.assertEqual(downloaded['format_id'], f1['format_id'])
3d4a70b8 405
3537b93d 406 info_dict = _make_result([f2, f1], extractor='youtube')
8dd54188 407 ydl = YDL({'format': 'best/bestvideo'})
3d4a70b8
PH
408 yie = YoutubeIE(ydl)
409 yie._sort_formats(info_dict['formats'])
410 ydl.process_ie_result(info_dict)
411 downloaded = ydl.downloaded_info_dicts[0]
67134eab 412 self.assertEqual(downloaded['format_id'], f1['format_id'])
3d4a70b8 413
317f7ab6
S
414 def test_audio_only_extractor_format_selection(self):
415 # For extractors with incomplete formats (all formats are audio-only or
416 # video-only) best and worst should fallback to corresponding best/worst
417 # video-only or audio-only formats (as per
067aa17e 418 # https://github.com/ytdl-org/youtube-dl/pull/5556)
317f7ab6
S
419 formats = [
420 {'format_id': 'low', 'ext': 'mp3', 'preference': 1, 'vcodec': 'none', 'url': TEST_URL},
421 {'format_id': 'high', 'ext': 'mp3', 'preference': 2, 'vcodec': 'none', 'url': TEST_URL},
422 ]
423 info_dict = _make_result(formats)
424
425 ydl = YDL({'format': 'best'})
426 ydl.process_ie_result(info_dict.copy())
427 downloaded = ydl.downloaded_info_dicts[0]
428 self.assertEqual(downloaded['format_id'], 'high')
429
430 ydl = YDL({'format': 'worst'})
431 ydl.process_ie_result(info_dict.copy())
432 downloaded = ydl.downloaded_info_dicts[0]
433 self.assertEqual(downloaded['format_id'], 'low')
434
435 def test_format_not_available(self):
436 formats = [
437 {'format_id': 'regular', 'ext': 'mp4', 'height': 360, 'url': TEST_URL},
438 {'format_id': 'video', 'ext': 'mp4', 'height': 720, 'acodec': 'none', 'url': TEST_URL},
439 ]
440 info_dict = _make_result(formats)
441
442 # This must fail since complete video-audio format does not match filter
443 # and extractor does not provide incomplete only formats (i.e. only
444 # video-only or audio-only).
445 ydl = YDL({'format': 'best[height>360]'})
446 self.assertRaises(ExtractorError, ydl.process_ie_result, info_dict.copy())
447
8cda78ef 448 def test_format_selection_issue_10083(self):
067aa17e 449 # See https://github.com/ytdl-org/youtube-dl/issues/10083
8cda78ef
S
450 formats = [
451 {'format_id': 'regular', 'height': 360, 'url': TEST_URL},
452 {'format_id': 'video', 'height': 720, 'acodec': 'none', 'url': TEST_URL},
453 {'format_id': 'audio', 'vcodec': 'none', 'url': TEST_URL},
454 ]
455 info_dict = _make_result(formats)
456
457 ydl = YDL({'format': 'best[height>360]/bestvideo[height>360]+bestaudio'})
458 ydl.process_ie_result(info_dict.copy())
459 self.assertEqual(ydl.downloaded_info_dicts[0]['format_id'], 'video+audio')
460
0a31a350
JMF
461 def test_invalid_format_specs(self):
462 def assert_syntax_error(format_spec):
463 ydl = YDL({'format': format_spec})
464 info_dict = _make_result([{'format_id': 'foo', 'url': TEST_URL}])
465 self.assertRaises(SyntaxError, ydl.process_ie_result, info_dict)
466
467 assert_syntax_error('bestvideo,,best')
468 assert_syntax_error('+bestaudio')
469 assert_syntax_error('bestvideo+')
d96d604e 470 assert_syntax_error('/')
0a31a350 471
083c9df9
PH
472 def test_format_filtering(self):
473 formats = [
474 {'format_id': 'A', 'filesize': 500, 'width': 1000},
475 {'format_id': 'B', 'filesize': 1000, 'width': 500},
476 {'format_id': 'C', 'filesize': 1000, 'width': 400},
477 {'format_id': 'D', 'filesize': 2000, 'width': 600},
478 {'format_id': 'E', 'filesize': 3000},
479 {'format_id': 'F'},
480 {'format_id': 'G', 'filesize': 1000000},
481 ]
482 for f in formats:
483 f['url'] = 'http://_/'
484 f['ext'] = 'unknown'
485 info_dict = _make_result(formats)
486
487 ydl = YDL({'format': 'best[filesize<3000]'})
488 ydl.process_ie_result(info_dict)
489 downloaded = ydl.downloaded_info_dicts[0]
490 self.assertEqual(downloaded['format_id'], 'D')
491
492 ydl = YDL({'format': 'best[filesize<=3000]'})
493 ydl.process_ie_result(info_dict)
494 downloaded = ydl.downloaded_info_dicts[0]
495 self.assertEqual(downloaded['format_id'], 'E')
496
497 ydl = YDL({'format': 'best[filesize <= ? 3000]'})
498 ydl.process_ie_result(info_dict)
499 downloaded = ydl.downloaded_info_dicts[0]
500 self.assertEqual(downloaded['format_id'], 'F')
501
502 ydl = YDL({'format': 'best [filesize = 1000] [width>450]'})
503 ydl.process_ie_result(info_dict)
504 downloaded = ydl.downloaded_info_dicts[0]
505 self.assertEqual(downloaded['format_id'], 'B')
506
507 ydl = YDL({'format': 'best [filesize = 1000] [width!=450]'})
508 ydl.process_ie_result(info_dict)
509 downloaded = ydl.downloaded_info_dicts[0]
510 self.assertEqual(downloaded['format_id'], 'C')
511
512 ydl = YDL({'format': '[filesize>?1]'})
513 ydl.process_ie_result(info_dict)
514 downloaded = ydl.downloaded_info_dicts[0]
515 self.assertEqual(downloaded['format_id'], 'G')
516
517 ydl = YDL({'format': '[filesize<1M]'})
518 ydl.process_ie_result(info_dict)
519 downloaded = ydl.downloaded_info_dicts[0]
520 self.assertEqual(downloaded['format_id'], 'E')
521
522 ydl = YDL({'format': '[filesize<1MiB]'})
523 ydl.process_ie_result(info_dict)
524 downloaded = ydl.downloaded_info_dicts[0]
525 self.assertEqual(downloaded['format_id'], 'G')
526
5acfa126
JMF
527 ydl = YDL({'format': 'all[width>=400][width<=600]'})
528 ydl.process_ie_result(info_dict)
529 downloaded_ids = [info['format_id'] for info in ydl.downloaded_info_dicts]
530 self.assertEqual(downloaded_ids, ['B', 'C', 'D'])
531
bb8e5536
JMF
532 ydl = YDL({'format': 'best[height<40]'})
533 try:
534 ydl.process_ie_result(info_dict)
535 except ExtractorError:
536 pass
537 self.assertEqual(ydl.downloaded_info_dicts, [])
538
0017d9ad
S
539 def test_default_format_spec(self):
540 ydl = YDL({'simulate': True})
5d0c5371 541 self.assertEqual(ydl._default_format_spec({}), 'bestvideo*+bestaudio/best')
0017d9ad 542
d08dcd2d
S
543 ydl = YDL({})
544 self.assertEqual(ydl._default_format_spec({'is_live': True}), 'best/bestvideo+bestaudio')
af0f7428 545
d08dcd2d 546 ydl = YDL({'simulate': True})
5d0c5371 547 self.assertEqual(ydl._default_format_spec({'is_live': True}), 'bestvideo*+bestaudio/best')
af0f7428 548
0017d9ad 549 ydl = YDL({'outtmpl': '-'})
af0f7428 550 self.assertEqual(ydl._default_format_spec({}), 'best/bestvideo+bestaudio')
0017d9ad
S
551
552 ydl = YDL({})
5d0c5371 553 self.assertEqual(ydl._default_format_spec({}, download=False), 'bestvideo*+bestaudio/best')
af0f7428 554 self.assertEqual(ydl._default_format_spec({'is_live': True}), 'best/bestvideo+bestaudio')
0017d9ad 555
f20bf146
JMF
556
557class TestYoutubeDL(unittest.TestCase):
ab84349b
JMF
558 def test_subtitles(self):
559 def s_formats(lang, autocaption=False):
560 return [{
561 'ext': ext,
562 'url': 'http://localhost/video.%s.%s' % (lang, ext),
563 '_auto': autocaption,
564 } for ext in ['vtt', 'srt', 'ass']]
565 subtitles = dict((l, s_formats(l)) for l in ['en', 'fr', 'es'])
566 auto_captions = dict((l, s_formats(l, True)) for l in ['it', 'pt', 'es'])
567 info_dict = {
568 'id': 'test',
569 'title': 'Test',
570 'url': 'http://localhost/video.mp4',
571 'subtitles': subtitles,
572 'automatic_captions': auto_captions,
573 'extractor': 'TEST',
732044af 574 'webpage_url': 'http://example.com/watch?v=shenanigans',
ab84349b
JMF
575 }
576
577 def get_info(params={}):
578 params.setdefault('simulate', True)
579 ydl = YDL(params)
580 ydl.report_warning = lambda *args, **kargs: None
581 return ydl.process_video_result(info_dict, download=False)
582
583 result = get_info()
584 self.assertFalse(result.get('requested_subtitles'))
585 self.assertEqual(result['subtitles'], subtitles)
586 self.assertEqual(result['automatic_captions'], auto_captions)
587
588 result = get_info({'writesubtitles': True})
589 subs = result['requested_subtitles']
590 self.assertTrue(subs)
591 self.assertEqual(set(subs.keys()), set(['en']))
592 self.assertTrue(subs['en'].get('data') is None)
593 self.assertEqual(subs['en']['ext'], 'ass')
594
595 result = get_info({'writesubtitles': True, 'subtitlesformat': 'foo/srt'})
596 subs = result['requested_subtitles']
597 self.assertEqual(subs['en']['ext'], 'srt')
598
599 result = get_info({'writesubtitles': True, 'subtitleslangs': ['es', 'fr', 'it']})
600 subs = result['requested_subtitles']
601 self.assertTrue(subs)
602 self.assertEqual(set(subs.keys()), set(['es', 'fr']))
603
604 result = get_info({'writesubtitles': True, 'writeautomaticsub': True, 'subtitleslangs': ['es', 'pt']})
605 subs = result['requested_subtitles']
606 self.assertTrue(subs)
607 self.assertEqual(set(subs.keys()), set(['es', 'pt']))
608 self.assertFalse(subs['es']['_auto'])
609 self.assertTrue(subs['pt']['_auto'])
610
98c70d6f
JMF
611 result = get_info({'writeautomaticsub': True, 'subtitleslangs': ['es', 'pt']})
612 subs = result['requested_subtitles']
613 self.assertTrue(subs)
614 self.assertEqual(set(subs.keys()), set(['es', 'pt']))
615 self.assertTrue(subs['es']['_auto'])
616 self.assertTrue(subs['pt']['_auto'])
617
b6c45014
JMF
618 def test_add_extra_info(self):
619 test_dict = {
620 'extractor': 'Foo',
621 }
622 extra_info = {
623 'extractor': 'Bar',
624 'playlist': 'funny videos',
625 }
626 YDL.add_extra_info(test_dict, extra_info)
627 self.assertEqual(test_dict['extractor'], 'Foo')
628 self.assertEqual(test_dict['playlist'], 'funny videos')
629
26e63931
JMF
630 def test_prepare_filename(self):
631 info = {
89087418
PH
632 'id': '1234',
633 'ext': 'mp4',
634 'width': None,
d0d9ade4 635 'height': 1080,
15da37c7
S
636 'title1': '$PATH',
637 'title2': '%PATH%',
26e63931 638 }
5f6a1245 639
a820dc72
RA
640 def fname(templ, na_placeholder='NA'):
641 params = {'outtmpl': templ}
642 if na_placeholder != 'NA':
643 params['outtmpl_na_placeholder'] = na_placeholder
644 ydl = YoutubeDL(params)
26e63931 645 return ydl.prepare_filename(info)
89087418
PH
646 self.assertEqual(fname('%(id)s.%(ext)s'), '1234.mp4')
647 self.assertEqual(fname('%(id)s-%(width)s.%(ext)s'), '1234-NA.mp4')
a820dc72
RA
648 NA_TEST_OUTTMPL = '%(uploader_date)s-%(width)d-%(id)s.%(ext)s'
649 # Replace missing fields with 'NA' by default
650 self.assertEqual(fname(NA_TEST_OUTTMPL), 'NA-NA-1234.mp4')
651 # Or by provided placeholder
652 self.assertEqual(fname(NA_TEST_OUTTMPL, na_placeholder='none'), 'none-none-1234.mp4')
653 self.assertEqual(fname(NA_TEST_OUTTMPL, na_placeholder=''), '--1234.mp4')
d0d9ade4
S
654 self.assertEqual(fname('%(height)d.%(ext)s'), '1080.mp4')
655 self.assertEqual(fname('%(height)6d.%(ext)s'), ' 1080.mp4')
656 self.assertEqual(fname('%(height)-6d.%(ext)s'), '1080 .mp4')
657 self.assertEqual(fname('%(height)06d.%(ext)s'), '001080.mp4')
658 self.assertEqual(fname('%(height) 06d.%(ext)s'), ' 01080.mp4')
659 self.assertEqual(fname('%(height) 06d.%(ext)s'), ' 01080.mp4')
660 self.assertEqual(fname('%(height)0 6d.%(ext)s'), ' 01080.mp4')
661 self.assertEqual(fname('%(height)0 6d.%(ext)s'), ' 01080.mp4')
662 self.assertEqual(fname('%(height) 0 6d.%(ext)s'), ' 01080.mp4')
15da37c7
S
663 self.assertEqual(fname('%%'), '%')
664 self.assertEqual(fname('%%%%'), '%%')
d0d9ade4
S
665 self.assertEqual(fname('%%(height)06d.%(ext)s'), '%(height)06d.mp4')
666 self.assertEqual(fname('%(width)06d.%(ext)s'), 'NA.mp4')
667 self.assertEqual(fname('%(width)06d.%%(ext)s'), 'NA.%(ext)s')
668 self.assertEqual(fname('%%(width)06d.%(ext)s'), '%(width)06d.mp4')
15da37c7
S
669 self.assertEqual(fname('Hello %(title1)s'), 'Hello $PATH')
670 self.assertEqual(fname('Hello %(title2)s'), 'Hello %PATH%')
26e63931 671
c57f7757
PH
672 def test_format_note(self):
673 ydl = YoutubeDL()
674 self.assertEqual(ydl._format_note({}), '')
675 assertRegexpMatches(self, ydl._format_note({
676 'vbr': 10,
398dea32 677 }), r'^\s*10k$')
5d583bdf
S
678 assertRegexpMatches(self, ydl._format_note({
679 'fps': 30,
398dea32 680 }), r'^30fps$')
5d583bdf 681
2b4ecde2
JMF
682 def test_postprocessors(self):
683 filename = 'post-processor-testfile.mp4'
684 audiofile = filename + '.mp3'
685
686 class SimplePP(PostProcessor):
687 def run(self, info):
2b4ecde2
JMF
688 with open(audiofile, 'wt') as f:
689 f.write('EXAMPLE')
592e97e8 690 return [info['filepath']], info
2b4ecde2 691
592e97e8 692 def run_pp(params, PP):
2b4ecde2
JMF
693 with open(filename, 'wt') as f:
694 f.write('EXAMPLE')
695 ydl = YoutubeDL(params)
592e97e8 696 ydl.add_post_processor(PP())
2b4ecde2
JMF
697 ydl.post_process(filename, {'filepath': filename})
698
592e97e8 699 run_pp({'keepvideo': True}, SimplePP)
2b4ecde2
JMF
700 self.assertTrue(os.path.exists(filename), '%s doesn\'t exist' % filename)
701 self.assertTrue(os.path.exists(audiofile), '%s doesn\'t exist' % audiofile)
702 os.unlink(filename)
703 os.unlink(audiofile)
704
592e97e8 705 run_pp({'keepvideo': False}, SimplePP)
2b4ecde2
JMF
706 self.assertFalse(os.path.exists(filename), '%s exists' % filename)
707 self.assertTrue(os.path.exists(audiofile), '%s doesn\'t exist' % audiofile)
708 os.unlink(audiofile)
709
592e97e8
JMF
710 class ModifierPP(PostProcessor):
711 def run(self, info):
712 with open(info['filepath'], 'wt') as f:
713 f.write('MODIFIED')
714 return [], info
715
716 run_pp({'keepvideo': False}, ModifierPP)
717 self.assertTrue(os.path.exists(filename), '%s doesn\'t exist' % filename)
718 os.unlink(filename)
719
531980d8
JMF
720 def test_match_filter(self):
721 class FilterYDL(YDL):
722 def __init__(self, *args, **kwargs):
723 super(FilterYDL, self).__init__(*args, **kwargs)
724 self.params['simulate'] = True
725
726 def process_info(self, info_dict):
727 super(YDL, self).process_info(info_dict)
728
729 def _match_entry(self, info_dict, incomplete):
730 res = super(FilterYDL, self)._match_entry(info_dict, incomplete)
731 if res is None:
732 self.downloaded_info_dicts.append(info_dict)
733 return res
734
735 first = {
736 'id': '1',
737 'url': TEST_URL,
738 'title': 'one',
739 'extractor': 'TEST',
740 'duration': 30,
741 'filesize': 10 * 1024,
e5a088dc 742 'playlist_id': '42',
db13c16e
S
743 'uploader': "變態妍字幕版 太妍 тест",
744 'creator': "тест ' 123 ' тест--",
732044af 745 'webpage_url': 'http://example.com/watch?v=shenanigans',
531980d8
JMF
746 }
747 second = {
748 'id': '2',
749 'url': TEST_URL,
750 'title': 'two',
751 'extractor': 'TEST',
752 'duration': 10,
753 'description': 'foo',
754 'filesize': 5 * 1024,
e5a088dc 755 'playlist_id': '43',
db13c16e 756 'uploader': "тест 123",
732044af 757 'webpage_url': 'http://example.com/watch?v=SHENANIGANS',
531980d8
JMF
758 }
759 videos = [first, second]
760
761 def get_videos(filter_=None):
762 ydl = FilterYDL({'match_filter': filter_})
763 for v in videos:
764 ydl.process_ie_result(v, download=True)
765 return [v['id'] for v in ydl.downloaded_info_dicts]
766
767 res = get_videos()
768 self.assertEqual(res, ['1', '2'])
769
770 def f(v):
771 if v['id'] == '1':
772 return None
773 else:
774 return 'Video id is not 1'
775 res = get_videos(f)
776 self.assertEqual(res, ['1'])
777
778 f = match_filter_func('duration < 30')
779 res = get_videos(f)
780 self.assertEqual(res, ['2'])
781
782 f = match_filter_func('description = foo')
783 res = get_videos(f)
784 self.assertEqual(res, ['2'])
785
786 f = match_filter_func('description =? foo')
787 res = get_videos(f)
788 self.assertEqual(res, ['1', '2'])
789
790 f = match_filter_func('filesize > 5KiB')
791 res = get_videos(f)
792 self.assertEqual(res, ['1'])
793
e5a088dc
S
794 f = match_filter_func('playlist_id = 42')
795 res = get_videos(f)
796 self.assertEqual(res, ['1'])
797
db13c16e
S
798 f = match_filter_func('uploader = "變態妍字幕版 太妍 тест"')
799 res = get_videos(f)
800 self.assertEqual(res, ['1'])
801
802 f = match_filter_func('uploader != "變態妍字幕版 太妍 тест"')
803 res = get_videos(f)
804 self.assertEqual(res, ['2'])
805
806 f = match_filter_func('creator = "тест \' 123 \' тест--"')
807 res = get_videos(f)
808 self.assertEqual(res, ['1'])
809
810 f = match_filter_func("creator = 'тест \\' 123 \\' тест--'")
811 res = get_videos(f)
812 self.assertEqual(res, ['1'])
813
814 f = match_filter_func(r"creator = 'тест \' 123 \' тест--' & duration > 30")
815 res = get_videos(f)
816 self.assertEqual(res, [])
817
e9eaf3fb
JMF
818 def test_playlist_items_selection(self):
819 entries = [{
820 'id': compat_str(i),
821 'title': compat_str(i),
822 'url': TEST_URL,
823 } for i in range(1, 5)]
824 playlist = {
825 '_type': 'playlist',
826 'id': 'test',
827 'entries': entries,
828 'extractor': 'test:playlist',
829 'extractor_key': 'test:playlist',
830 'webpage_url': 'http://example.com',
831 }
832
4e9e1e24 833 def get_downloaded_info_dicts(params):
e9eaf3fb 834 ydl = YDL(params)
e0abaab2
S
835 # make a deep copy because the dictionary and nested entries
836 # can be modified
837 ydl.process_ie_result(copy.deepcopy(playlist))
4e9e1e24
S
838 return ydl.downloaded_info_dicts
839
840 def get_ids(params):
841 return [int(v['id']) for v in get_downloaded_info_dicts(params)]
e9eaf3fb
JMF
842
843 result = get_ids({})
844 self.assertEqual(result, [1, 2, 3, 4])
845
846 result = get_ids({'playlistend': 10})
847 self.assertEqual(result, [1, 2, 3, 4])
848
849 result = get_ids({'playlistend': 2})
850 self.assertEqual(result, [1, 2])
851
852 result = get_ids({'playliststart': 10})
853 self.assertEqual(result, [])
854
855 result = get_ids({'playliststart': 2})
856 self.assertEqual(result, [2, 3, 4])
857
858 result = get_ids({'playlist_items': '2-4'})
859 self.assertEqual(result, [2, 3, 4])
860
861 result = get_ids({'playlist_items': '2,4'})
862 self.assertEqual(result, [2, 4])
863
864 result = get_ids({'playlist_items': '10'})
865 self.assertEqual(result, [])
866
86a15ed6
S
867 result = get_ids({'playlist_items': '3-10'})
868 self.assertEqual(result, [3, 4])
869
cd6fc19e
S
870 result = get_ids({'playlist_items': '2-4,3-4,3'})
871 self.assertEqual(result, [2, 3, 4])
872
4e9e1e24
S
873 # Tests for https://github.com/ytdl-org/youtube-dl/issues/10591
874 # @{
875 result = get_downloaded_info_dicts({'playlist_items': '2-4,3-4,3'})
876 self.assertEqual(result[0]['playlist_index'], 2)
877 self.assertEqual(result[1]['playlist_index'], 3)
878
879 result = get_downloaded_info_dicts({'playlist_items': '2-4,3-4,3'})
880 self.assertEqual(result[0]['playlist_index'], 2)
881 self.assertEqual(result[1]['playlist_index'], 3)
882 self.assertEqual(result[2]['playlist_index'], 4)
883
884 result = get_downloaded_info_dicts({'playlist_items': '4,2'})
885 self.assertEqual(result[0]['playlist_index'], 4)
886 self.assertEqual(result[1]['playlist_index'], 2)
887 # @}
888
e37afbe0 889 def test_urlopen_no_file_protocol(self):
067aa17e 890 # see https://github.com/ytdl-org/youtube-dl/issues/8227
e37afbe0
JMF
891 ydl = YDL()
892 self.assertRaises(compat_urllib_error.URLError, ydl.urlopen, 'file:///etc/passwd')
893
b286f201
YCH
894 def test_do_not_override_ie_key_in_url_transparent(self):
895 ydl = YDL()
896
897 class Foo1IE(InfoExtractor):
898 _VALID_URL = r'foo1:'
899
900 def _real_extract(self, url):
901 return {
902 '_type': 'url_transparent',
903 'url': 'foo2:',
904 'ie_key': 'Foo2',
0396806f
S
905 'title': 'foo1 title',
906 'id': 'foo1_id',
b286f201
YCH
907 }
908
909 class Foo2IE(InfoExtractor):
910 _VALID_URL = r'foo2:'
911
912 def _real_extract(self, url):
913 return {
914 '_type': 'url',
915 'url': 'foo3:',
916 'ie_key': 'Foo3',
917 }
918
919 class Foo3IE(InfoExtractor):
920 _VALID_URL = r'foo3:'
921
922 def _real_extract(self, url):
51350db5 923 return _make_result([{'url': TEST_URL}], title='foo3 title')
b286f201
YCH
924
925 ydl.add_info_extractor(Foo1IE(ydl))
926 ydl.add_info_extractor(Foo2IE(ydl))
927 ydl.add_info_extractor(Foo3IE(ydl))
928 ydl.extract_info('foo1:')
929 downloaded = ydl.downloaded_info_dicts[0]
930 self.assertEqual(downloaded['url'], TEST_URL)
51350db5 931 self.assertEqual(downloaded['title'], 'foo1 title')
0396806f
S
932 self.assertEqual(downloaded['id'], 'testid')
933 self.assertEqual(downloaded['extractor'], 'testex')
934 self.assertEqual(downloaded['extractor_key'], 'TestEx')
b286f201 935
a0566bbf 936 # Test case for https://github.com/ytdl-org/youtube-dl/issues/27064
937 def test_ignoreerrors_for_playlist_with_url_transparent_iterable_entries(self):
938
939 class _YDL(YDL):
940 def __init__(self, *args, **kwargs):
941 super(_YDL, self).__init__(*args, **kwargs)
942
943 def trouble(self, s, tb=None):
944 pass
945
946 ydl = _YDL({
947 'format': 'extra',
948 'ignoreerrors': True,
949 })
950
951 class VideoIE(InfoExtractor):
952 _VALID_URL = r'video:(?P<id>\d+)'
953
954 def _real_extract(self, url):
955 video_id = self._match_id(url)
956 formats = [{
957 'format_id': 'default',
958 'url': 'url:',
959 }]
960 if video_id == '0':
961 raise ExtractorError('foo')
962 if video_id == '2':
963 formats.append({
964 'format_id': 'extra',
965 'url': TEST_URL,
966 })
967 return {
968 'id': video_id,
969 'title': 'Video %s' % video_id,
970 'formats': formats,
971 }
972
973 class PlaylistIE(InfoExtractor):
974 _VALID_URL = r'playlist:'
975
976 def _entries(self):
977 for n in range(3):
978 video_id = compat_str(n)
979 yield {
980 '_type': 'url_transparent',
981 'ie_key': VideoIE.ie_key(),
982 'id': video_id,
983 'url': 'video:%s' % video_id,
984 'title': 'Video Transparent %s' % video_id,
985 }
986
987 def _real_extract(self, url):
988 return self.playlist_result(self._entries())
989
990 ydl.add_info_extractor(VideoIE(ydl))
991 ydl.add_info_extractor(PlaylistIE(ydl))
992 info = ydl.extract_info('playlist:')
993 entries = info['entries']
994 self.assertEqual(len(entries), 3)
995 self.assertTrue(entries[0] is None)
996 self.assertTrue(entries[1] is None)
997 self.assertEqual(len(ydl.downloaded_info_dicts), 1)
998 downloaded = ydl.downloaded_info_dicts[0]
999 self.assertEqual(entries[2], downloaded)
1000 self.assertEqual(downloaded['url'], TEST_URL)
1001 self.assertEqual(downloaded['title'], 'Video Transparent 2')
1002 self.assertEqual(downloaded['id'], '2')
1003 self.assertEqual(downloaded['extractor'], 'Video')
1004 self.assertEqual(downloaded['extractor_key'], 'Video')
1005
2b4ecde2 1006
e028d0d1
JMF
1007if __name__ == '__main__':
1008 unittest.main()