Compare commits
	
		
			255 Commits
		
	
	
		
			2013.10.18
			...
			2013.11.19
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 34b3afc7be | ||
|  | 00373a4c5d | ||
|  | cb7dfeeac4 | ||
|  | efd6c574a2 | ||
|  | 4113e6ab56 | ||
|  | 9a942a4671 | ||
|  | 9906d397a0 | ||
|  | ae8f787141 | ||
|  | a81b4d5c8f | ||
|  | 887c6acdf2 | ||
|  | 83aa529330 | ||
|  | 96b31b6533 | ||
|  | fccd377198 | ||
|  | 73c566695f | ||
|  | 63b7b7224a | ||
|  | ce80c8b8ee | ||
|  | 749febf4d1 | ||
|  | bdde425cbe | ||
|  | 746f491f82 | ||
|  | 1672647ade | ||
|  | 90b6bbc38c | ||
|  | ce02ed60f2 | ||
|  | 1e5b9a95fd | ||
|  | 1d699755e0 | ||
|  | ddf49c6344 | ||
|  | d1c252048b | ||
|  | eab2724138 | ||
|  | 21ea3e06c9 | ||
|  | 52d703d3d1 | ||
|  | ce152341a1 | ||
|  | f058e34011 | ||
|  | b5349e8721 | ||
|  | 7150858d49 | ||
|  | 91c7271aab | ||
|  | aa13b2dffd | ||
|  | fc2ef392be | ||
|  | 463a908705 | ||
|  | d24ffe1cfa | ||
|  | 78fb87b283 | ||
|  | ab2d524780 | ||
|  | 85d61685f1 | ||
|  | b9643eed7c | ||
|  | feee2ecfa9 | ||
|  | a25a5cfeec | ||
|  | 0e145dd541 | ||
|  | 9f9be844fc | ||
|  | e3b9ab5e18 | ||
|  | c66d2baa9c | ||
|  | ca715127a2 | ||
|  | ea7a7af1d4 | ||
|  | 80b9bbce86 | ||
|  | d37936386f | ||
|  | c3a3028f9f | ||
|  | 6c5ad80cdc | ||
|  | b5bdc2699a | ||
|  | 384b98cd8f | ||
|  | eb9b5bffef | ||
|  | 8b8cbd8f6d | ||
|  | 72b18c5d34 | ||
|  | eb0a839866 | ||
|  | 1777d5a952 | ||
|  | d4b7da84c3 | ||
|  | 801dbbdffd | ||
|  | 0ed05a1d2d | ||
|  | 1008bebade | ||
|  | ae84f879d7 | ||
|  | be6dfd1b49 | ||
|  | 231516b6c9 | ||
|  | fb53d58dcf | ||
|  | 2a9e9b210b | ||
|  | 897d6cc43a | ||
|  | f470c6c812 | ||
|  | 566d4e0425 | ||
|  | 81be02d2f9 | ||
|  | c2b6a482d5 | ||
|  | 12c167c881 | ||
|  | 20aafee7fa | ||
|  | be07375b66 | ||
|  | dd5bcdc4c9 | ||
|  | 6161d17579 | ||
|  | 4ac5306ae7 | ||
|  | b1a80ec1a9 | ||
|  | 672fe94dcb | ||
|  | 51040b72ed | ||
|  | 4f045eef8f | ||
|  | 5d7b253ea0 | ||
|  | b0759f0c19 | ||
|  | 065472936a | ||
|  | fc4a0c2aec | ||
|  | eeb165e674 | ||
|  | 9ee2b5f6f2 | ||
|  | da54be877a | ||
|  | 50a886b7ab | ||
|  | 76e67c2cb6 | ||
|  | 5137ebac0b | ||
|  | a8eeb0597b | ||
|  | 4ed3e51080 | ||
|  | 7f34001d57 | ||
|  | 2dcf7d8f99 | ||
|  | 19b0668251 | ||
|  | e7e6b54d8a | ||
|  | 2a1a8ffe41 | ||
|  | 08fb86c49b | ||
|  | 3633d77c0f | ||
|  | 165e179764 | ||
|  | 12ebdd1506 | ||
|  | 1baf9a5938 | ||
|  | a56f9de156 | ||
|  | fa5d47af4b | ||
|  | d607038753 | ||
|  | 9ac6a01aaf | ||
|  | be97abc247 | ||
|  | 9103bbc5cd | ||
|  | b6c45014ae | ||
|  | a3dd924871 | ||
|  | 137bbb3e37 | ||
|  | 86ad94bb2e | ||
|  | 3e56add7c9 | ||
|  | f52f01b5d2 | ||
|  | 98d7efb537 | ||
|  | cf51923545 | ||
|  | 38fcd4597a | ||
|  | 165e3bb67a | ||
|  | 38db46794f | ||
|  | a9a3876d55 | ||
|  | 1f343eaabb | ||
|  | 72a5b4f702 | ||
|  | 0a43ddf320 | ||
|  | 31366066bd | ||
|  | aa2484e390 | ||
|  | 8eddf3e91d | ||
|  | 60d142aa8d | ||
|  | 66cf3ac342 | ||
|  | ab4e151347 | ||
|  | ac2547f5ff | ||
|  | 5f1ea943ab | ||
|  | 0ef7ad5cd4 | ||
|  | 9f1109a564 | ||
|  | 33b1d9595d | ||
|  | 7193498811 | ||
|  | 72321ead7b | ||
|  | b5d0d817bc | ||
|  | 94badb2599 | ||
|  | b9a836515f | ||
|  | 21c924f406 | ||
|  | e54fd4b23b | ||
|  | 57dd9a8f2f | ||
|  | 912cbf5d4e | ||
|  | 43d7895ea0 | ||
|  | f7ff55aa78 | ||
|  | 795f28f871 | ||
|  | f6cc16f5d8 | ||
|  | 321a01f971 | ||
|  | 646e17a53d | ||
|  | dd508b7c4f | ||
|  | 2563bcc85c | ||
|  | 702665c085 | ||
|  | dcc2a706ef | ||
|  | 2bc67c35ac | ||
|  | 77ae65877e | ||
|  | 32a35e4418 | ||
|  | 369a759acc | ||
|  | 79b3f61228 | ||
|  | 216d71d001 | ||
|  | 78a3a9f89e | ||
|  | a7685f3bf4 | ||
|  | f088ea5486 | ||
|  | 1003d108d5 | ||
|  | 8abeeb9449 | ||
|  | c1002e96e9 | ||
|  | 77d0a82fef | ||
|  | ebc14f251c | ||
|  | d41e6efc85 | ||
|  | 8ffa13e03e | ||
|  | db477d3a37 | ||
|  | 750e9833b8 | ||
|  | 82f0ac657c | ||
|  | eb6a2277a2 | ||
|  | f8778fb0fa | ||
|  | e2f9de207c | ||
|  | a93cc0d943 | ||
|  | 7d8c2e07f2 | ||
|  | efb4c36b18 | ||
|  | 29526d0d2b | ||
|  | 198e370f23 | ||
|  | c19f7764a5 | ||
|  | bc63d9d329 | ||
|  | aa929c37d5 | ||
|  | af4d506eb3 | ||
|  | 5da0549581 | ||
|  | 749a4fd2fd | ||
|  | 6f71ef580c | ||
|  | 67874aeffa | ||
|  | 3e6a330d38 | ||
|  | aee5e18c8f | ||
|  | 5b11143d05 | ||
|  | 7b2212e954 | ||
|  | 71865091ab | ||
|  | 125cfd78e8 | ||
|  | 8cb57d9b91 | ||
|  | 14e10b2b6e | ||
|  | 6e76104d66 | ||
|  | 1d45a23b74 | ||
|  | 7df286540f | ||
|  | 5d0c97541a | ||
|  | 49a25557b0 | ||
|  | b5936c0059 | ||
|  | 600cc1a4f0 | ||
|  | ea32fbacc8 | ||
|  | 00fe14fc75 | ||
|  | fcc28edb2f | ||
|  | fac6be2dd5 | ||
|  | 1cf64ee468 | ||
|  | cdec0190c4 | ||
|  | 2450bcb28b | ||
|  | 3126050c0f | ||
|  | 93b22c7828 | ||
|  | 0a89b2852e | ||
|  | 55b3e45bba | ||
|  | 365bcf6d97 | ||
|  | 71907db3ba | ||
|  | 6803655ced | ||
|  | df1c39ec5c | ||
|  | 80f55a9511 | ||
|  | 7853cc5ae1 | ||
|  | 586a91b67f | ||
|  | b028e96144 | ||
|  | ce68b5907c | ||
|  | fe7e0c9825 | ||
|  | 12893efe01 | ||
|  | a6387bfd3c | ||
|  | f6a54188c2 | ||
|  | cbbd9a9c69 | ||
|  | 685a9cd2f1 | ||
|  | 182a107877 | ||
|  | 8c51aa6506 | ||
|  | 3fd39e37f2 | ||
|  | 49e86983e7 | ||
|  | a9c58ad945 | ||
|  | f8b45beacc | ||
|  | 9d92015d43 | ||
|  | 50a6150ed9 | ||
|  | b0505eb611 | ||
|  | 284acd57d6 | ||
|  | 8ed6b34477 | ||
|  | f6f1fc9286 | ||
|  | 8e590a117f | ||
|  | d5594202aa | ||
|  | b186d949cf | ||
|  | 3d2986063c | ||
|  | 41fd7c7e60 | ||
|  | fdefe96bf2 | ||
|  | 16f36a6fc9 | ||
|  | f44415360e | ||
|  | cce722b79c | 
							
								
								
									
										17
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								README.md
									
									
									
									
									
								
							| @@ -21,6 +21,8 @@ which means you can modify it, redistribute it or use it however you like. | ||||
|                                sudo if needed) | ||||
|     -i, --ignore-errors        continue on download errors, for example to to | ||||
|                                skip unavailable videos in a playlist | ||||
|     --abort-on-error           Abort downloading of further videos (in the | ||||
|                                playlist or the command line) if an error occurs | ||||
|     --dump-user-agent          display the current browser identification | ||||
|     --user-agent UA            specify a custom user agent | ||||
|     --referer REF              specify a custom referer, use if the video access | ||||
| @@ -30,7 +32,7 @@ which means you can modify it, redistribute it or use it however you like. | ||||
|     --extractor-descriptions   Output descriptions of all supported extractors | ||||
|     --proxy URL                Use the specified HTTP/HTTPS proxy | ||||
|     --no-check-certificate     Suppress HTTPS certificate validation. | ||||
|     --cache-dir None           Location in the filesystem where youtube-dl can | ||||
|     --cache-dir DIR            Location in the filesystem where youtube-dl can | ||||
|                                store downloaded information permanently. By | ||||
|                                default $XDG_CACHE_HOME/youtube-dl or ~/.cache | ||||
|                                /youtube-dl . | ||||
| @@ -76,7 +78,10 @@ which means you can modify it, redistribute it or use it however you like. | ||||
|                                %(uploader_id)s for the uploader nickname if | ||||
|                                different, %(autonumber)s to get an automatically | ||||
|                                incremented number, %(ext)s for the filename | ||||
|                                extension, %(upload_date)s for the upload date | ||||
|                                extension, %(format)s for the format description | ||||
|                                (like "22 - 1280x720" or "HD"),%(format_id)s for | ||||
|                                the unique id of the format (like Youtube's | ||||
|                                itags: "137"),%(upload_date)s for the upload date | ||||
|                                (YYYYMMDD), %(extractor)s for the provider | ||||
|                                (youtube, metacafe, etc), %(id)s for the video id | ||||
|                                , %(playlist)s for the playlist the video is in, | ||||
| @@ -87,12 +92,14 @@ which means you can modify it, redistribute it or use it however you like. | ||||
|                                ownloads/%(uploader)s/%(title)s-%(id)s.%(ext)s' . | ||||
|     --autonumber-size NUMBER   Specifies the number of digits in %(autonumber)s | ||||
|                                when it is present in output filename template or | ||||
|                                --autonumber option is given | ||||
|                                --auto-number option is given | ||||
|     --restrict-filenames       Restrict filenames to only ASCII characters, and | ||||
|                                avoid "&" and spaces in filenames | ||||
|     -a, --batch-file FILE      file containing URLs to download ('-' for stdin) | ||||
|     -w, --no-overwrites        do not overwrite files | ||||
|     -c, --continue             resume partially downloaded files | ||||
|     -c, --continue             force resume of partially downloaded files. By | ||||
|                                default, youtube-dl will resume downloads if | ||||
|                                possible. | ||||
|     --no-continue              do not resume partially downloaded files (restart | ||||
|                                from beginning) | ||||
|     --cookies FILE             file to read cookies from and dump cookie jar in | ||||
| @@ -122,6 +129,8 @@ which means you can modify it, redistribute it or use it however you like. | ||||
|     -v, --verbose              print various debugging information | ||||
|     --dump-intermediate-pages  print downloaded pages to debug problems(very | ||||
|                                verbose) | ||||
|     --write-pages              Write downloaded pages to files in the current | ||||
|                                directory | ||||
|  | ||||
| ## Video Format Options: | ||||
|     -f, --format FORMAT        video format code, specifiy the order of | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| __youtube-dl() | ||||
| __youtube_dl() | ||||
| { | ||||
|     local cur prev opts | ||||
|     COMPREPLY=() | ||||
| @@ -15,4 +15,4 @@ __youtube-dl() | ||||
|     fi | ||||
| } | ||||
|  | ||||
| complete -F __youtube-dl youtube-dl | ||||
| complete -F __youtube_dl youtube-dl | ||||
|   | ||||
							
								
								
									
										39
									
								
								devscripts/check-porn.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								devscripts/check-porn.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| """ | ||||
| This script employs a VERY basic heuristic ('porn' in webpage.lower()) to check | ||||
| if we are not 'age_limit' tagging some porn site | ||||
| """ | ||||
|  | ||||
| # Allow direct execution | ||||
| import os | ||||
| import sys | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from test.helper import get_testcases | ||||
| from youtube_dl.utils import compat_urllib_request | ||||
|  | ||||
| for test in get_testcases(): | ||||
|     try: | ||||
|         webpage = compat_urllib_request.urlopen(test['url'], timeout=10).read() | ||||
|     except: | ||||
|         print('\nFail: {0}'.format(test['name'])) | ||||
|         continue | ||||
|  | ||||
|     webpage = webpage.decode('utf8', 'replace') | ||||
|  | ||||
|     if 'porn' in webpage.lower() and ('info_dict' not in test | ||||
|                                       or 'age_limit' not in test['info_dict'] | ||||
|                                       or test['info_dict']['age_limit'] != 18): | ||||
|         print('\nPotential missing age_limit check: {0}'.format(test['name'])) | ||||
|  | ||||
|     elif 'porn' not in webpage.lower() and ('info_dict' in test and | ||||
|                                             'age_limit' in test['info_dict'] and | ||||
|                                             test['info_dict']['age_limit'] == 18): | ||||
|         print('\nPotential false negative: {0}'.format(test['name'])) | ||||
|  | ||||
|     else: | ||||
|         sys.stdout.write('.') | ||||
|     sys.stdout.flush() | ||||
|  | ||||
| print() | ||||
| @@ -88,10 +88,6 @@ ROOT=$(pwd) | ||||
|     "$ROOT/devscripts/gh-pages/update-sites.py" | ||||
|     git add *.html *.html.in update | ||||
|     git commit -m "release $version" | ||||
|     git show HEAD | ||||
|     read -p "Is it good, can I push? (y/n) " -n 1 | ||||
|     if [[ ! $REPLY =~ ^[Yy]$ ]]; then exit 1; fi | ||||
|     echo | ||||
|     git push "$ROOT" gh-pages | ||||
|     git push "$ORIGIN_URL" gh-pages | ||||
| ) | ||||
|   | ||||
							
								
								
									
										8
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								setup.py
									
									
									
									
									
								
							| @@ -8,8 +8,10 @@ import sys | ||||
|  | ||||
| try: | ||||
|     from setuptools import setup | ||||
|     setuptools_available = True | ||||
| except ImportError: | ||||
|     from distutils.core import setup | ||||
|     setuptools_available = False | ||||
|  | ||||
| try: | ||||
|     # This will create an exe that needs Microsoft Visual C++ 2008 | ||||
| @@ -43,13 +45,16 @@ if len(sys.argv) >= 2 and sys.argv[1] == 'py2exe': | ||||
|     params = py2exe_params | ||||
| else: | ||||
|     params = { | ||||
|         'scripts': ['bin/youtube-dl'], | ||||
|         'data_files': [  # Installing system-wide would require sudo... | ||||
|             ('etc/bash_completion.d', ['youtube-dl.bash-completion']), | ||||
|             ('share/doc/youtube_dl', ['README.txt']), | ||||
|             ('share/man/man1/', ['youtube-dl.1']) | ||||
|         ] | ||||
|     } | ||||
|     if setuptools_available: | ||||
|         params['entry_points'] = {'console_scripts': ['youtube-dl = youtube_dl:main']} | ||||
|     else: | ||||
|         params['scripts'] = ['bin/youtube-dl'] | ||||
|  | ||||
| # Get the version from youtube_dl/version.py without importing the package | ||||
| exec(compile(open('youtube_dl/version.py').read(), | ||||
| @@ -63,6 +68,7 @@ setup( | ||||
|     ' YouTube.com and other video sites.', | ||||
|     url='https://github.com/rg3/youtube-dl', | ||||
|     author='Ricardo Garcia', | ||||
|     author_email='ytdl@yt-dl.org', | ||||
|     maintainer='Philipp Hagemeister', | ||||
|     maintainer_email='phihag@phihag.de', | ||||
|     packages=['youtube_dl', 'youtube_dl.extractor'], | ||||
|   | ||||
| @@ -5,9 +5,11 @@ import json | ||||
| import os.path | ||||
| import re | ||||
| import types | ||||
| import sys | ||||
|  | ||||
| import youtube_dl.extractor | ||||
| from youtube_dl import YoutubeDL | ||||
| from youtube_dl.utils import preferredencoding | ||||
|  | ||||
|  | ||||
| def global_setup(): | ||||
| @@ -33,6 +35,21 @@ def try_rm(filename): | ||||
|             raise | ||||
|  | ||||
|  | ||||
| def report_warning(message): | ||||
|     ''' | ||||
|     Print the message to stderr, it will be prefixed with 'WARNING:' | ||||
|     If stderr is a tty file the 'WARNING:' will be colored | ||||
|     ''' | ||||
|     if sys.stderr.isatty() and os.name != 'nt': | ||||
|         _msg_header = u'\033[0;33mWARNING:\033[0m' | ||||
|     else: | ||||
|         _msg_header = u'WARNING:' | ||||
|     output = u'%s %s\n' % (_msg_header, message) | ||||
|     if 'b' in getattr(sys.stderr, 'mode', '') or sys.version_info[0] < 3: | ||||
|         output = output.encode(preferredencoding()) | ||||
|     sys.stderr.write(output) | ||||
|  | ||||
|  | ||||
| class FakeYDL(YoutubeDL): | ||||
|     def __init__(self, override=None): | ||||
|         # Different instances of the downloader can't share the same dictionary | ||||
|   | ||||
| @@ -62,10 +62,10 @@ class TestFormatSelection(unittest.TestCase): | ||||
|  | ||||
|     def test_format_limit(self): | ||||
|         formats = [ | ||||
|             {u'format_id': u'meh'}, | ||||
|             {u'format_id': u'good'}, | ||||
|             {u'format_id': u'great'}, | ||||
|             {u'format_id': u'excellent'}, | ||||
|             {u'format_id': u'meh', u'url': u'http://example.com/meh'}, | ||||
|             {u'format_id': u'good', u'url': u'http://example.com/good'}, | ||||
|             {u'format_id': u'great', u'url': u'http://example.com/great'}, | ||||
|             {u'format_id': u'excellent', u'url': u'http://example.com/exc'}, | ||||
|         ] | ||||
|         info_dict = { | ||||
|             u'formats': formats, u'extractor': u'test', 'id': 'testvid'} | ||||
| @@ -94,6 +94,52 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded[u'format_id'], u'excellent') | ||||
|  | ||||
|     def test_format_selection(self): | ||||
|         formats = [ | ||||
|             {u'format_id': u'35', u'ext': u'mp4'}, | ||||
|             {u'format_id': u'45', u'ext': u'webm'}, | ||||
|             {u'format_id': u'47', u'ext': u'webm'}, | ||||
|             {u'format_id': u'2', u'ext': u'flv'}, | ||||
|         ] | ||||
|         info_dict = {u'formats': formats, u'extractor': u'test'} | ||||
|  | ||||
|         ydl = YDL({'format': u'20/47'}) | ||||
|         ydl.process_ie_result(info_dict) | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded['format_id'], u'47') | ||||
|  | ||||
|         ydl = YDL({'format': u'20/71/worst'}) | ||||
|         ydl.process_ie_result(info_dict) | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded['format_id'], u'35') | ||||
|  | ||||
|         ydl = YDL() | ||||
|         ydl.process_ie_result(info_dict) | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded['format_id'], u'2') | ||||
|  | ||||
|         ydl = YDL({'format': u'webm/mp4'}) | ||||
|         ydl.process_ie_result(info_dict) | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded['format_id'], u'47') | ||||
|  | ||||
|         ydl = YDL({'format': u'3gp/40/mp4'}) | ||||
|         ydl.process_ie_result(info_dict) | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded['format_id'], u'35') | ||||
|  | ||||
|     def test_add_extra_info(self): | ||||
|         test_dict = { | ||||
|             'extractor': 'Foo', | ||||
|         } | ||||
|         extra_info = { | ||||
|             'extractor': 'Bar', | ||||
|             'playlist': 'funny videos', | ||||
|         } | ||||
|         YDL.add_extra_info(test_dict, extra_info) | ||||
|         self.assertEqual(test_dict['extractor'], 'Foo') | ||||
|         self.assertEqual(test_dict['playlist'], 'funny videos') | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
|   | ||||
| @@ -1,70 +0,0 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| # Allow direct execution | ||||
| import os | ||||
| import sys | ||||
| import unittest | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from test.helper import FakeYDL, global_setup, md5 | ||||
| global_setup() | ||||
|  | ||||
|  | ||||
| from youtube_dl.extractor import DailymotionIE | ||||
|  | ||||
| class TestDailymotionSubtitles(unittest.TestCase): | ||||
|     def setUp(self): | ||||
|         self.DL = FakeYDL() | ||||
|         self.url = 'http://www.dailymotion.com/video/xczg00' | ||||
|     def getInfoDict(self): | ||||
|         IE = DailymotionIE(self.DL) | ||||
|         info_dict = IE.extract(self.url) | ||||
|         return info_dict | ||||
|     def getSubtitles(self): | ||||
|         info_dict = self.getInfoDict() | ||||
|         return info_dict[0]['subtitles'] | ||||
|     def test_no_writesubtitles(self): | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(subtitles, None) | ||||
|     def test_subtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '976553874490cba125086bbfea3ff76f') | ||||
|     def test_subtitles_lang(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitleslangs'] = ['fr'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['fr']), '594564ec7d588942e384e920e5341792') | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles.keys()), 5) | ||||
|     def test_list_subtitles(self): | ||||
|         self.DL.expect_warning(u'Automatic Captions not supported by this server') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|     def test_automatic_captions(self): | ||||
|         self.DL.expect_warning(u'Automatic Captions not supported by this server') | ||||
|         self.DL.params['writeautomaticsub'] = True | ||||
|         self.DL.params['subtitleslang'] = ['en'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertTrue(len(subtitles.keys()) == 0) | ||||
|     def test_nosubtitles(self): | ||||
|         self.DL.expect_warning(u'video doesn\'t have subtitles') | ||||
|         self.url = 'http://www.dailymotion.com/video/x12u166_le-zapping-tele-star-du-08-aout-2013_tv' | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles), 0) | ||||
|     def test_multiple_langs(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         langs = ['es', 'fr', 'de'] | ||||
|         self.DL.params['subtitleslangs'] = langs | ||||
|         subtitles = self.getSubtitles() | ||||
|         for lang in langs: | ||||
|             self.assertTrue(subtitles.get(lang) is not None, u'Subtitles for \'%s\' not extracted' % lang) | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
| @@ -6,7 +6,14 @@ import sys | ||||
| import unittest | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from test.helper import get_params, get_testcases, global_setup, try_rm, md5 | ||||
| from test.helper import ( | ||||
|     get_params, | ||||
|     get_testcases, | ||||
|     global_setup, | ||||
|     try_rm, | ||||
|     md5, | ||||
|     report_warning | ||||
| ) | ||||
| global_setup() | ||||
|  | ||||
|  | ||||
| @@ -19,10 +26,12 @@ import youtube_dl.YoutubeDL | ||||
| from youtube_dl.utils import ( | ||||
|     compat_str, | ||||
|     compat_urllib_error, | ||||
|     compat_HTTPError, | ||||
|     DownloadError, | ||||
|     ExtractorError, | ||||
|     UnavailableVideoError, | ||||
| ) | ||||
| from youtube_dl.extractor import get_info_extractor | ||||
|  | ||||
| RETRIES = 3 | ||||
|  | ||||
| @@ -55,17 +64,25 @@ def generator(test_case): | ||||
|  | ||||
|     def test_template(self): | ||||
|         ie = youtube_dl.extractor.get_info_extractor(test_case['name']) | ||||
|         other_ies = [get_info_extractor(ie_key) for ie_key in test_case.get('add_ie', [])] | ||||
|         def print_skipping(reason): | ||||
|             print('Skipping %s: %s' % (test_case['name'], reason)) | ||||
|         if not ie._WORKING: | ||||
|         if not ie.working(): | ||||
|             print_skipping('IE marked as not _WORKING') | ||||
|             return | ||||
|         if 'playlist' not in test_case and not test_case['file']: | ||||
|             print_skipping('No output file specified') | ||||
|             return | ||||
|         if 'playlist' not in test_case: | ||||
|             info_dict = test_case.get('info_dict', {}) | ||||
|             if not test_case.get('file') and not (info_dict.get('id') and info_dict.get('ext')): | ||||
|                 print_skipping('The output file cannot be know, the "file" ' | ||||
|                     'key is missing or the info_dict is incomplete') | ||||
|                 return | ||||
|         if 'skip' in test_case: | ||||
|             print_skipping(test_case['skip']) | ||||
|             return | ||||
|         for other_ie in other_ies: | ||||
|             if not other_ie.working(): | ||||
|                 print_skipping(u'test depends on %sIE, marked as not WORKING' % other_ie.ie_key()) | ||||
|                 return | ||||
|  | ||||
|         params = get_params(test_case.get('params', {})) | ||||
|  | ||||
| @@ -77,35 +94,47 @@ def generator(test_case): | ||||
|                 finished_hook_called.add(status['filename']) | ||||
|         ydl.fd.add_progress_hook(_hook) | ||||
|  | ||||
|         def get_tc_filename(tc): | ||||
|             return tc.get('file') or ydl.prepare_filename(tc.get('info_dict', {})) | ||||
|  | ||||
|         test_cases = test_case.get('playlist', [test_case]) | ||||
|         for tc in test_cases: | ||||
|             try_rm(tc['file']) | ||||
|             try_rm(tc['file'] + '.part') | ||||
|             try_rm(tc['file'] + '.info.json') | ||||
|         def try_rm_tcs_files(): | ||||
|             for tc in test_cases: | ||||
|                 tc_filename = get_tc_filename(tc) | ||||
|                 try_rm(tc_filename) | ||||
|                 try_rm(tc_filename + '.part') | ||||
|                 try_rm(tc_filename + '.info.json') | ||||
|         try_rm_tcs_files() | ||||
|         try: | ||||
|             for retry in range(1, RETRIES + 1): | ||||
|             try_num = 1 | ||||
|             while True: | ||||
|                 try: | ||||
|                     ydl.download([test_case['url']]) | ||||
|                 except (DownloadError, ExtractorError) as err: | ||||
|                     if retry == RETRIES: raise | ||||
|  | ||||
|                     # Check if the exception is not a network related one | ||||
|                     if not err.exc_info[0] in (compat_urllib_error.URLError, socket.timeout, UnavailableVideoError): | ||||
|                     if not err.exc_info[0] in (compat_urllib_error.URLError, socket.timeout, UnavailableVideoError) or (err.exc_info[0] == compat_HTTPError and err.exc_info[1].code == 503): | ||||
|                         raise | ||||
|  | ||||
|                     print('Retrying: {0} failed tries\n\n##########\n\n'.format(retry)) | ||||
|                     if try_num == RETRIES: | ||||
|                         report_warning(u'Failed due to network errors, skipping...') | ||||
|                         return | ||||
|  | ||||
|                     print('Retrying: {0} failed tries\n\n##########\n\n'.format(try_num)) | ||||
|  | ||||
|                     try_num += 1 | ||||
|                 else: | ||||
|                     break | ||||
|  | ||||
|             for tc in test_cases: | ||||
|                 tc_filename = get_tc_filename(tc) | ||||
|                 if not test_case.get('params', {}).get('skip_download', False): | ||||
|                     self.assertTrue(os.path.exists(tc['file']), msg='Missing file ' + tc['file']) | ||||
|                     self.assertTrue(tc['file'] in finished_hook_called) | ||||
|                 self.assertTrue(os.path.exists(tc['file'] + '.info.json')) | ||||
|                     self.assertTrue(os.path.exists(tc_filename), msg='Missing file ' + tc_filename) | ||||
|                     self.assertTrue(tc_filename in finished_hook_called) | ||||
|                 self.assertTrue(os.path.exists(tc_filename + '.info.json')) | ||||
|                 if 'md5' in tc: | ||||
|                     md5_for_file = _file_md5(tc['file']) | ||||
|                     md5_for_file = _file_md5(tc_filename) | ||||
|                     self.assertEqual(md5_for_file, tc['md5']) | ||||
|                 with io.open(tc['file'] + '.info.json', encoding='utf-8') as infof: | ||||
|                 with io.open(tc_filename + '.info.json', encoding='utf-8') as infof: | ||||
|                     info_dict = json.load(infof) | ||||
|                 for (info_field, expected) in tc.get('info_dict', {}).items(): | ||||
|                     if isinstance(expected, compat_str) and expected.startswith('md5:'): | ||||
| @@ -125,11 +154,11 @@ def generator(test_case): | ||||
|                 # Check for the presence of mandatory fields | ||||
|                 for key in ('id', 'url', 'title', 'ext'): | ||||
|                     self.assertTrue(key in info_dict.keys() and info_dict[key]) | ||||
|                 # Check for mandatory fields that are automatically set by YoutubeDL | ||||
|                 for key in ['webpage_url', 'extractor', 'extractor_key']: | ||||
|                     self.assertTrue(info_dict.get(key), u'Missing field: %s' % key) | ||||
|         finally: | ||||
|             for tc in test_cases: | ||||
|                 try_rm(tc['file']) | ||||
|                 try_rm(tc['file'] + '.part') | ||||
|                 try_rm(tc['file'] + '.info.json') | ||||
|             try_rm_tcs_files() | ||||
|  | ||||
|     return test_template | ||||
|  | ||||
|   | ||||
| @@ -17,9 +17,11 @@ from youtube_dl.extractor import ( | ||||
|     DailymotionUserIE, | ||||
|     VimeoChannelIE, | ||||
|     UstreamChannelIE, | ||||
|     SoundcloudSetIE, | ||||
|     SoundcloudUserIE, | ||||
|     LivestreamIE, | ||||
|     NHLVideocenterIE, | ||||
|     BambuserChannelIE, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -60,6 +62,14 @@ class TestPlaylists(unittest.TestCase): | ||||
|         self.assertEqual(result['id'], u'5124905') | ||||
|         self.assertTrue(len(result['entries']) >= 11) | ||||
|  | ||||
|     def test_soundcloud_set(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = SoundcloudSetIE(dl) | ||||
|         result = ie.extract('https://soundcloud.com/the-concept-band/sets/the-royal-concept-ep') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['title'], u'The Royal Concept EP') | ||||
|         self.assertTrue(len(result['entries']) >= 6) | ||||
|  | ||||
|     def test_soundcloud_user(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = SoundcloudUserIE(dl) | ||||
| @@ -85,5 +95,13 @@ class TestPlaylists(unittest.TestCase): | ||||
|         self.assertEqual(result['title'], u'Highlights') | ||||
|         self.assertEqual(len(result['entries']), 12) | ||||
|  | ||||
|     def test_bambuser_channel(self): | ||||
|         dl = FakeYDL() | ||||
|         ie = BambuserChannelIE(dl) | ||||
|         result = ie.extract('http://bambuser.com/channel/pixelversity') | ||||
|         self.assertIsPlaylist(result) | ||||
|         self.assertEqual(result['title'], u'pixelversity') | ||||
|         self.assertTrue(len(result['entries']) >= 66) | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
|   | ||||
							
								
								
									
										211
									
								
								test/test_subtitles.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										211
									
								
								test/test_subtitles.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,211 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| # Allow direct execution | ||||
| import os | ||||
| import sys | ||||
| import unittest | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from test.helper import FakeYDL, global_setup, md5 | ||||
| global_setup() | ||||
|  | ||||
|  | ||||
| from youtube_dl.extractor import ( | ||||
|     YoutubeIE, | ||||
|     DailymotionIE, | ||||
|     TEDIE, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class BaseTestSubtitles(unittest.TestCase): | ||||
|     url = None | ||||
|     IE = None | ||||
|     def setUp(self): | ||||
|         self.DL = FakeYDL() | ||||
|         self.ie = self.IE(self.DL) | ||||
|  | ||||
|     def getInfoDict(self): | ||||
|         info_dict = self.ie.extract(self.url) | ||||
|         return info_dict | ||||
|  | ||||
|     def getSubtitles(self): | ||||
|         info_dict = self.getInfoDict() | ||||
|         return info_dict['subtitles'] | ||||
|  | ||||
|  | ||||
| class TestYoutubeSubtitles(BaseTestSubtitles): | ||||
|     url = 'QRS8MkLhQmM' | ||||
|     IE = YoutubeIE | ||||
|  | ||||
|     def getSubtitles(self): | ||||
|         info_dict = self.getInfoDict() | ||||
|         return info_dict[0]['subtitles'] | ||||
|  | ||||
|     def test_youtube_no_writesubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = False | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(subtitles, None) | ||||
|  | ||||
|     def test_youtube_subtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '4cd9278a35ba2305f47354ee13472260') | ||||
|  | ||||
|     def test_youtube_subtitles_lang(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitleslangs'] = ['it'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['it']), '164a51f16f260476a05b50fe4c2f161d') | ||||
|  | ||||
|     def test_youtube_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles.keys()), 13) | ||||
|  | ||||
|     def test_youtube_subtitles_sbv_format(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitlesformat'] = 'sbv' | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '13aeaa0c245a8bed9a451cb643e3ad8b') | ||||
|  | ||||
|     def test_youtube_subtitles_vtt_format(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitlesformat'] = 'vtt' | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '356cdc577fde0c6783b9b822e7206ff7') | ||||
|  | ||||
|     def test_youtube_list_subtitles(self): | ||||
|         self.DL.expect_warning(u'Video doesn\'t have automatic captions') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|  | ||||
|     def test_youtube_automatic_captions(self): | ||||
|         self.url = '8YoUxe5ncPo' | ||||
|         self.DL.params['writeautomaticsub'] = True | ||||
|         self.DL.params['subtitleslangs'] = ['it'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertTrue(subtitles['it'] is not None) | ||||
|  | ||||
|     def test_youtube_nosubtitles(self): | ||||
|         self.DL.expect_warning(u'video doesn\'t have subtitles') | ||||
|         self.url = 'sAjKT8FhjI8' | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles), 0) | ||||
|  | ||||
|     def test_youtube_multiple_langs(self): | ||||
|         self.url = 'QRS8MkLhQmM' | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         langs = ['it', 'fr', 'de'] | ||||
|         self.DL.params['subtitleslangs'] = langs | ||||
|         subtitles = self.getSubtitles() | ||||
|         for lang in langs: | ||||
|             self.assertTrue(subtitles.get(lang) is not None, u'Subtitles for \'%s\' not extracted' % lang) | ||||
|  | ||||
|  | ||||
| class TestDailymotionSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.dailymotion.com/video/xczg00' | ||||
|     IE = DailymotionIE | ||||
|  | ||||
|     def test_no_writesubtitles(self): | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(subtitles, None) | ||||
|  | ||||
|     def test_subtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '976553874490cba125086bbfea3ff76f') | ||||
|  | ||||
|     def test_subtitles_lang(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitleslangs'] = ['fr'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['fr']), '594564ec7d588942e384e920e5341792') | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles.keys()), 5) | ||||
|  | ||||
|     def test_list_subtitles(self): | ||||
|         self.DL.expect_warning(u'Automatic Captions not supported by this server') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|  | ||||
|     def test_automatic_captions(self): | ||||
|         self.DL.expect_warning(u'Automatic Captions not supported by this server') | ||||
|         self.DL.params['writeautomaticsub'] = True | ||||
|         self.DL.params['subtitleslang'] = ['en'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertTrue(len(subtitles.keys()) == 0) | ||||
|  | ||||
|     def test_nosubtitles(self): | ||||
|         self.DL.expect_warning(u'video doesn\'t have subtitles') | ||||
|         self.url = 'http://www.dailymotion.com/video/x12u166_le-zapping-tele-star-du-08-aout-2013_tv' | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles), 0) | ||||
|  | ||||
|     def test_multiple_langs(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         langs = ['es', 'fr', 'de'] | ||||
|         self.DL.params['subtitleslangs'] = langs | ||||
|         subtitles = self.getSubtitles() | ||||
|         for lang in langs: | ||||
|             self.assertTrue(subtitles.get(lang) is not None, u'Subtitles for \'%s\' not extracted' % lang) | ||||
|  | ||||
|  | ||||
| class TestTedSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.ted.com/talks/dan_dennett_on_our_consciousness.html' | ||||
|     IE = TEDIE | ||||
|  | ||||
|     def test_no_writesubtitles(self): | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(subtitles, None) | ||||
|  | ||||
|     def test_subtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '2154f31ff9b9f89a0aa671537559c21d') | ||||
|  | ||||
|     def test_subtitles_lang(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitleslangs'] = ['fr'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['fr']), '7616cbc6df20ec2c1204083c83871cf6') | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles.keys()), 28) | ||||
|  | ||||
|     def test_list_subtitles(self): | ||||
|         self.DL.expect_warning(u'Automatic Captions not supported by this server') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|  | ||||
|     def test_automatic_captions(self): | ||||
|         self.DL.expect_warning(u'Automatic Captions not supported by this server') | ||||
|         self.DL.params['writeautomaticsub'] = True | ||||
|         self.DL.params['subtitleslang'] = ['en'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertTrue(len(subtitles.keys()) == 0) | ||||
|  | ||||
|     def test_multiple_langs(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         langs = ['es', 'fr', 'de'] | ||||
|         self.DL.params['subtitleslangs'] = langs | ||||
|         subtitles = self.getSubtitles() | ||||
|         for lang in langs: | ||||
|             self.assertTrue(subtitles.get(lang) is not None, u'Subtitles for \'%s\' not extracted' % lang) | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
| @@ -1,95 +0,0 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| # Allow direct execution | ||||
| import os | ||||
| import sys | ||||
| import unittest | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from test.helper import FakeYDL, global_setup, md5 | ||||
| global_setup() | ||||
|  | ||||
|  | ||||
| from youtube_dl.extractor import YoutubeIE | ||||
|  | ||||
|  | ||||
| class TestYoutubeSubtitles(unittest.TestCase): | ||||
|     def setUp(self): | ||||
|         self.DL = FakeYDL() | ||||
|         self.url = 'QRS8MkLhQmM' | ||||
|  | ||||
|     def getInfoDict(self): | ||||
|         IE = YoutubeIE(self.DL) | ||||
|         info_dict = IE.extract(self.url) | ||||
|         return info_dict | ||||
|  | ||||
|     def getSubtitles(self): | ||||
|         info_dict = self.getInfoDict() | ||||
|         return info_dict[0]['subtitles'] | ||||
|  | ||||
|     def test_youtube_no_writesubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = False | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(subtitles, None) | ||||
|  | ||||
|     def test_youtube_subtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '4cd9278a35ba2305f47354ee13472260') | ||||
|  | ||||
|     def test_youtube_subtitles_lang(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitleslangs'] = ['it'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['it']), '164a51f16f260476a05b50fe4c2f161d') | ||||
|  | ||||
|     def test_youtube_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles.keys()), 13) | ||||
|  | ||||
|     def test_youtube_subtitles_sbv_format(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitlesformat'] = 'sbv' | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '13aeaa0c245a8bed9a451cb643e3ad8b') | ||||
|  | ||||
|     def test_youtube_subtitles_vtt_format(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitlesformat'] = 'vtt' | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '356cdc577fde0c6783b9b822e7206ff7') | ||||
|  | ||||
|     def test_youtube_list_subtitles(self): | ||||
|         self.DL.expect_warning(u'Video doesn\'t have automatic captions') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|  | ||||
|     def test_youtube_automatic_captions(self): | ||||
|         self.url = '8YoUxe5ncPo' | ||||
|         self.DL.params['writeautomaticsub'] = True | ||||
|         self.DL.params['subtitleslangs'] = ['it'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertTrue(subtitles['it'] is not None) | ||||
|  | ||||
|     def test_youtube_nosubtitles(self): | ||||
|         self.DL.expect_warning(u'video doesn\'t have subtitles') | ||||
|         self.url = 'sAjKT8FhjI8' | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles), 0) | ||||
|  | ||||
|     def test_youtube_multiple_langs(self): | ||||
|         self.url = 'QRS8MkLhQmM' | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         langs = ['it', 'fr', 'de'] | ||||
|         self.DL.params['subtitleslangs'] = langs | ||||
|         subtitles = self.getSubtitles() | ||||
|         for lang in langs: | ||||
|             self.assertTrue(subtitles.get(lang) is not None, u'Subtitles for \'%s\' not extracted' % lang) | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
| @@ -4,12 +4,16 @@ import re | ||||
| import subprocess | ||||
| import sys | ||||
| import time | ||||
| import traceback | ||||
|  | ||||
| if os.name == 'nt': | ||||
|     import ctypes | ||||
|  | ||||
| from .utils import * | ||||
| from .utils import ( | ||||
|     compat_urllib_error, | ||||
|     compat_urllib_request, | ||||
|     ContentTooShortError, | ||||
|     determine_ext, | ||||
|     encodeFilename, | ||||
|     sanitize_open, | ||||
|     timeconvert, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class FileDownloader(object): | ||||
| @@ -144,16 +148,8 @@ class FileDownloader(object): | ||||
|     def to_stderr(self, message): | ||||
|         self.ydl.to_screen(message) | ||||
|  | ||||
|     def to_cons_title(self, message): | ||||
|         """Set console/terminal window title to message.""" | ||||
|         if not self.params.get('consoletitle', False): | ||||
|             return | ||||
|         if os.name == 'nt' and ctypes.windll.kernel32.GetConsoleWindow(): | ||||
|             # c_wchar_p() might not be necessary if `message` is | ||||
|             # already of type unicode() | ||||
|             ctypes.windll.kernel32.SetConsoleTitleW(ctypes.c_wchar_p(message)) | ||||
|         elif 'TERM' in os.environ: | ||||
|             self.to_screen('\033]0;%s\007' % message, skip_eol=True) | ||||
|     def to_console_title(self, message): | ||||
|         self.ydl.to_console_title(message) | ||||
|  | ||||
|     def trouble(self, *args, **kargs): | ||||
|         self.ydl.trouble(*args, **kargs) | ||||
| @@ -194,7 +190,7 @@ class FileDownloader(object): | ||||
|             if old_filename == new_filename: | ||||
|                 return | ||||
|             os.rename(encodeFilename(old_filename), encodeFilename(new_filename)) | ||||
|         except (IOError, OSError) as err: | ||||
|         except (IOError, OSError): | ||||
|             self.report_error(u'unable to rename file') | ||||
|  | ||||
|     def try_utime(self, filename, last_modified_hdr): | ||||
| @@ -227,8 +223,14 @@ class FileDownloader(object): | ||||
|         if self.params.get('noprogress', False): | ||||
|             return | ||||
|         clear_line = (u'\x1b[K' if sys.stderr.isatty() and os.name != 'nt' else u'') | ||||
|         eta_str = self.format_eta(eta) | ||||
|         percent_str = self.format_percent(percent) | ||||
|         if eta is not None: | ||||
|             eta_str = self.format_eta(eta) | ||||
|         else: | ||||
|             eta_str = 'Unknown ETA' | ||||
|         if percent is not None: | ||||
|             percent_str = self.format_percent(percent) | ||||
|         else: | ||||
|             percent_str = 'Unknown %' | ||||
|         speed_str = self.format_speed(speed) | ||||
|         if self.params.get('progress_with_newline', False): | ||||
|             self.to_screen(u'[download] %s of %s at %s ETA %s' % | ||||
| @@ -236,7 +238,7 @@ class FileDownloader(object): | ||||
|         else: | ||||
|             self.to_screen(u'\r%s[download] %s of %s at %s ETA %s' % | ||||
|                 (clear_line, percent_str, data_len_str, speed_str, eta_str), skip_eol=True) | ||||
|         self.to_cons_title(u'youtube-dl - %s of %s at %s ETA %s' % | ||||
|         self.to_console_title(u'youtube-dl - %s of %s at %s ETA %s' % | ||||
|                 (percent_str.strip(), data_len_str.strip(), speed_str.strip(), eta_str.strip())) | ||||
|  | ||||
|     def report_resuming_byte(self, resume_len): | ||||
| @@ -251,7 +253,7 @@ class FileDownloader(object): | ||||
|         """Report file has already been fully downloaded.""" | ||||
|         try: | ||||
|             self.to_screen(u'[download] %s has already been downloaded' % file_name) | ||||
|         except (UnicodeEncodeError) as err: | ||||
|         except UnicodeEncodeError: | ||||
|             self.to_screen(u'[download] The file has already been downloaded') | ||||
|  | ||||
|     def report_unable_to_resume(self): | ||||
| @@ -267,7 +269,7 @@ class FileDownloader(object): | ||||
|             self.to_screen(u'\r%s[download] 100%% of %s in %s' % | ||||
|                 (clear_line, data_len_str, self.format_seconds(tot_time))) | ||||
|  | ||||
|     def _download_with_rtmpdump(self, filename, url, player_url, page_url, play_path, tc_url): | ||||
|     def _download_with_rtmpdump(self, filename, url, player_url, page_url, play_path, tc_url, live): | ||||
|         self.report_destination(filename) | ||||
|         tmpfilename = self.temp_name(filename) | ||||
|         test = self.params.get('test', False) | ||||
| @@ -294,6 +296,8 @@ class FileDownloader(object): | ||||
|             basic_args += ['--tcUrl', url] | ||||
|         if test: | ||||
|             basic_args += ['--stop', '1'] | ||||
|         if live: | ||||
|             basic_args += ['--live'] | ||||
|         args = basic_args + [[], ['--resume', '--skip', '1']][self.params.get('continuedl', False)] | ||||
|         if self.params.get('verbose', False): | ||||
|             try: | ||||
| @@ -366,15 +370,20 @@ class FileDownloader(object): | ||||
|         self.report_destination(filename) | ||||
|         tmpfilename = self.temp_name(filename) | ||||
|  | ||||
|         args = ['ffmpeg', '-y', '-i', url, '-f', 'mp4', tmpfilename] | ||||
|         # Check for ffmpeg first | ||||
|         try: | ||||
|             subprocess.call(['ffmpeg', '-h'], stdout=(open(os.path.devnull, 'w')), stderr=subprocess.STDOUT) | ||||
|         except (OSError, IOError): | ||||
|             self.report_error(u'm3u8 download detected but "%s" could not be run' % args[0] ) | ||||
|             return False | ||||
|         args = ['-y', '-i', url, '-f', 'mp4', '-c', 'copy', | ||||
|             '-bsf:a', 'aac_adtstoasc', tmpfilename] | ||||
|  | ||||
|         retval = subprocess.call(args) | ||||
|         for program in ['avconv', 'ffmpeg']: | ||||
|             try: | ||||
|                 subprocess.call([program, '-version'], stdout=(open(os.path.devnull, 'w')), stderr=subprocess.STDOUT) | ||||
|                 break | ||||
|             except (OSError, IOError): | ||||
|                 pass | ||||
|         else: | ||||
|             self.report_error(u'm3u8 download detected but ffmpeg or avconv could not be found') | ||||
|         cmd = [program] + args | ||||
|  | ||||
|         retval = subprocess.call(cmd) | ||||
|         if retval == 0: | ||||
|             fsize = os.path.getsize(encodeFilename(tmpfilename)) | ||||
|             self.to_screen(u'\r[%s] %s bytes' % (args[0], fsize)) | ||||
| @@ -411,7 +420,8 @@ class FileDownloader(object): | ||||
|                                                 info_dict.get('player_url', None), | ||||
|                                                 info_dict.get('page_url', None), | ||||
|                                                 info_dict.get('play_path', None), | ||||
|                                                 info_dict.get('tc_url', None)) | ||||
|                                                 info_dict.get('tc_url', None), | ||||
|                                                 info_dict.get('rtmp_live', False)) | ||||
|  | ||||
|         # Attempt to download using mplayer | ||||
|         if url.startswith('mms') or url.startswith('rtsp'): | ||||
| @@ -550,12 +560,11 @@ class FileDownloader(object): | ||||
|             # Progress message | ||||
|             speed = self.calc_speed(start, time.time(), byte_counter - resume_len) | ||||
|             if data_len is None: | ||||
|                 self.report_progress('Unknown %', data_len_str, speed_str, 'Unknown ETA') | ||||
|                 eta = None | ||||
|                 eta = percent = None | ||||
|             else: | ||||
|                 percent = self.calc_percent(byte_counter, data_len) | ||||
|                 eta = self.calc_eta(start, time.time(), data_len - resume_len, byte_counter - resume_len) | ||||
|                 self.report_progress(percent, data_len_str, speed, eta) | ||||
|             self.report_progress(percent, data_len_str, speed, eta) | ||||
|  | ||||
|             self._hook_progress({ | ||||
|                 'downloaded_bytes': byte_counter, | ||||
|   | ||||
| @@ -501,7 +501,7 @@ class FFmpegMetadataPP(FFmpegPostProcessor): | ||||
|  | ||||
|         options = ['-c', 'copy'] | ||||
|         for (name, value) in metadata.items(): | ||||
|             options.extend(['-metadata', '%s="%s"' % (name, value)]) | ||||
|             options.extend(['-metadata', '%s=%s' % (name, value)]) | ||||
|         options.extend(['-f', ext]) | ||||
|  | ||||
|         self._downloader.to_screen(u'[ffmpeg] Adding metadata to \'%s\'' % filename) | ||||
|   | ||||
| @@ -13,7 +13,34 @@ import sys | ||||
| import time | ||||
| import traceback | ||||
|  | ||||
| from .utils import * | ||||
| if os.name == 'nt': | ||||
|     import ctypes | ||||
|  | ||||
| from .utils import ( | ||||
|     compat_http_client, | ||||
|     compat_print, | ||||
|     compat_str, | ||||
|     compat_urllib_error, | ||||
|     compat_urllib_request, | ||||
|     ContentTooShortError, | ||||
|     date_from_str, | ||||
|     DateRange, | ||||
|     determine_ext, | ||||
|     DownloadError, | ||||
|     encodeFilename, | ||||
|     ExtractorError, | ||||
|     locked_file, | ||||
|     MaxDownloadsReached, | ||||
|     PostProcessingError, | ||||
|     preferredencoding, | ||||
|     SameFileError, | ||||
|     sanitize_filename, | ||||
|     subtitles_filename, | ||||
|     takewhile_inclusive, | ||||
|     UnavailableVideoError, | ||||
|     write_json_file, | ||||
|     write_string, | ||||
| ) | ||||
| from .extractor import get_info_extractor, gen_extractors | ||||
| from .FileDownloader import FileDownloader | ||||
|  | ||||
| @@ -91,7 +118,7 @@ class YoutubeDL(object): | ||||
|     downloadarchive:   File name of a file where all downloads are recorded. | ||||
|                        Videos already present in the file are not downloaded | ||||
|                        again. | ||||
|      | ||||
|  | ||||
|     The following parameters are not used by YoutubeDL itself, they are used by | ||||
|     the FileDownloader: | ||||
|     nopart, updatetime, buffersize, ratelimit, min_filesize, max_filesize, test, | ||||
| @@ -176,6 +203,37 @@ class YoutubeDL(object): | ||||
|             output = output.encode(preferredencoding()) | ||||
|         sys.stderr.write(output) | ||||
|  | ||||
|     def to_console_title(self, message): | ||||
|         if not self.params.get('consoletitle', False): | ||||
|             return | ||||
|         if os.name == 'nt' and ctypes.windll.kernel32.GetConsoleWindow(): | ||||
|             # c_wchar_p() might not be necessary if `message` is | ||||
|             # already of type unicode() | ||||
|             ctypes.windll.kernel32.SetConsoleTitleW(ctypes.c_wchar_p(message)) | ||||
|         elif 'TERM' in os.environ: | ||||
|             write_string(u'\033]0;%s\007' % message, self._screen_file) | ||||
|  | ||||
|     def save_console_title(self): | ||||
|         if not self.params.get('consoletitle', False): | ||||
|             return | ||||
|         if 'TERM' in os.environ: | ||||
|             # Save the title on stack | ||||
|             write_string(u'\033[22;0t', self._screen_file) | ||||
|  | ||||
|     def restore_console_title(self): | ||||
|         if not self.params.get('consoletitle', False): | ||||
|             return | ||||
|         if 'TERM' in os.environ: | ||||
|             # Restore the title from stack | ||||
|             write_string(u'\033[23;0t', self._screen_file) | ||||
|  | ||||
|     def __enter__(self): | ||||
|         self.save_console_title() | ||||
|         return self | ||||
|  | ||||
|     def __exit__(self, *args): | ||||
|         self.restore_console_title() | ||||
|  | ||||
|     def fixed_template(self): | ||||
|         """Checks if the output template is fixed.""" | ||||
|         return (re.search(u'(?u)%\\(.+?\\)s', self.params['outtmpl']) is None) | ||||
| @@ -216,10 +274,10 @@ class YoutubeDL(object): | ||||
|         If stderr is a tty file the 'WARNING:' will be colored | ||||
|         ''' | ||||
|         if sys.stderr.isatty() and os.name != 'nt': | ||||
|             _msg_header=u'\033[0;33mWARNING:\033[0m' | ||||
|             _msg_header = u'\033[0;33mWARNING:\033[0m' | ||||
|         else: | ||||
|             _msg_header=u'WARNING:' | ||||
|         warning_message=u'%s %s' % (_msg_header,message) | ||||
|             _msg_header = u'WARNING:' | ||||
|         warning_message = u'%s %s' % (_msg_header, message) | ||||
|         self.to_stderr(warning_message) | ||||
|  | ||||
|     def report_error(self, message, tb=None): | ||||
| @@ -234,19 +292,6 @@ class YoutubeDL(object): | ||||
|         error_message = u'%s %s' % (_msg_header, message) | ||||
|         self.trouble(error_message, tb) | ||||
|  | ||||
|     def slow_down(self, start_time, byte_counter): | ||||
|         """Sleep if the download speed is over the rate limit.""" | ||||
|         rate_limit = self.params.get('ratelimit', None) | ||||
|         if rate_limit is None or byte_counter == 0: | ||||
|             return | ||||
|         now = time.time() | ||||
|         elapsed = now - start_time | ||||
|         if elapsed <= 0.0: | ||||
|             return | ||||
|         speed = float(byte_counter) / elapsed | ||||
|         if speed > rate_limit: | ||||
|             time.sleep((byte_counter - rate_limit * (now - start_time)) / rate_limit) | ||||
|  | ||||
|     def report_writedescription(self, descfn): | ||||
|         """ Report that the description file is being written """ | ||||
|         self.to_screen(u'[info] Writing video description to: ' + descfn) | ||||
| @@ -267,7 +312,7 @@ class YoutubeDL(object): | ||||
|         """Report file has already been fully downloaded.""" | ||||
|         try: | ||||
|             self.to_screen(u'[download] %s has already been downloaded' % file_name) | ||||
|         except (UnicodeEncodeError) as err: | ||||
|         except UnicodeEncodeError: | ||||
|             self.to_screen(u'[download] The file has already been downloaded') | ||||
|  | ||||
|     def increment_downloads(self): | ||||
| @@ -285,16 +330,18 @@ class YoutubeDL(object): | ||||
|                 autonumber_size = 5 | ||||
|             autonumber_templ = u'%0' + str(autonumber_size) + u'd' | ||||
|             template_dict['autonumber'] = autonumber_templ % self._num_downloads | ||||
|             if template_dict['playlist_index'] is not None: | ||||
|             if template_dict.get('playlist_index') is not None: | ||||
|                 template_dict['playlist_index'] = u'%05d' % template_dict['playlist_index'] | ||||
|  | ||||
|             sanitize = lambda k,v: sanitize_filename( | ||||
|             sanitize = lambda k, v: sanitize_filename( | ||||
|                 u'NA' if v is None else compat_str(v), | ||||
|                 restricted=self.params.get('restrictfilenames'), | ||||
|                 is_id=(k==u'id')) | ||||
|             template_dict = dict((k, sanitize(k, v)) for k,v in template_dict.items()) | ||||
|                 is_id=(k == u'id')) | ||||
|             template_dict = dict((k, sanitize(k, v)) | ||||
|                                  for k, v in template_dict.items()) | ||||
|  | ||||
|             filename = self.params['outtmpl'] % template_dict | ||||
|             tmpl = os.path.expanduser(self.params['outtmpl']) | ||||
|             filename = tmpl % template_dict | ||||
|             return filename | ||||
|         except KeyError as err: | ||||
|             self.report_error(u'Erroneous output template') | ||||
| @@ -328,14 +375,20 @@ class YoutubeDL(object): | ||||
|             return (u'%(title)s has already been recorded in archive' | ||||
|                     % info_dict) | ||||
|         return None | ||||
|          | ||||
|  | ||||
|     @staticmethod | ||||
|     def add_extra_info(info_dict, extra_info): | ||||
|         '''Set the keys from extra_info in info dict if they are missing''' | ||||
|         for key, value in extra_info.items(): | ||||
|             info_dict.setdefault(key, value) | ||||
|  | ||||
|     def extract_info(self, url, download=True, ie_key=None, extra_info={}): | ||||
|         ''' | ||||
|         Returns a list with a dictionary for each video we find. | ||||
|         If 'download', also downloads the videos. | ||||
|         extra_info is a dict containing the extra values to add to each result | ||||
|          ''' | ||||
|          | ||||
|  | ||||
|         if ie_key: | ||||
|             ies = [self.get_info_extractor(ie_key)] | ||||
|         else: | ||||
| @@ -355,17 +408,17 @@ class YoutubeDL(object): | ||||
|                     break | ||||
|                 if isinstance(ie_result, list): | ||||
|                     # Backwards compatibility: old IE result format | ||||
|                     for result in ie_result: | ||||
|                         result.update(extra_info) | ||||
|                     ie_result = { | ||||
|                         '_type': 'compat_list', | ||||
|                         'entries': ie_result, | ||||
|                     } | ||||
|                 else: | ||||
|                     ie_result.update(extra_info) | ||||
|                 if 'extractor' not in ie_result: | ||||
|                     ie_result['extractor'] = ie.IE_NAME | ||||
|                 return self.process_ie_result(ie_result, download=download) | ||||
|                 self.add_extra_info(ie_result, | ||||
|                     { | ||||
|                         'extractor': ie.IE_NAME, | ||||
|                         'webpage_url': url, | ||||
|                         'extractor_key': ie.ie_key(), | ||||
|                     }) | ||||
|                 return self.process_ie_result(ie_result, download, extra_info) | ||||
|             except ExtractorError as de: # An error we somewhat expected | ||||
|                 self.report_error(compat_str(de), de.format_traceback()) | ||||
|                 break | ||||
| @@ -377,7 +430,7 @@ class YoutubeDL(object): | ||||
|                     raise | ||||
|         else: | ||||
|             self.report_error(u'no suitable InfoExtractor: %s' % url) | ||||
|          | ||||
|  | ||||
|     def process_ie_result(self, ie_result, download=True, extra_info={}): | ||||
|         """ | ||||
|         Take the result of the ie(may be modified) and resolve all unresolved | ||||
| @@ -389,8 +442,8 @@ class YoutubeDL(object): | ||||
|  | ||||
|         result_type = ie_result.get('_type', 'video') # If not given we suppose it's a video, support the default old system | ||||
|         if result_type == 'video': | ||||
|             ie_result.update(extra_info) | ||||
|             return self.process_video_result(ie_result) | ||||
|             self.add_extra_info(ie_result, extra_info) | ||||
|             return self.process_video_result(ie_result, download=download) | ||||
|         elif result_type == 'url': | ||||
|             # We have to add extra_info to the results because it may be | ||||
|             # contained in a playlist | ||||
| @@ -399,9 +452,10 @@ class YoutubeDL(object): | ||||
|                                      ie_key=ie_result.get('ie_key'), | ||||
|                                      extra_info=extra_info) | ||||
|         elif result_type == 'playlist': | ||||
|             self.add_extra_info(ie_result, extra_info) | ||||
|             # We process each entry in the playlist | ||||
|             playlist = ie_result.get('title', None) or ie_result.get('id', None) | ||||
|             self.to_screen(u'[download] Downloading playlist: %s'  % playlist) | ||||
|             self.to_screen(u'[download] Downloading playlist: %s' % playlist) | ||||
|  | ||||
|             playlist_results = [] | ||||
|  | ||||
| @@ -419,17 +473,15 @@ class YoutubeDL(object): | ||||
|             self.to_screen(u"[%s] playlist '%s': Collected %d video ids (downloading %d of them)" % | ||||
|                 (ie_result['extractor'], playlist, n_all_entries, n_entries)) | ||||
|  | ||||
|             for i,entry in enumerate(entries,1): | ||||
|                 self.to_screen(u'[download] Downloading video #%s of %s' %(i, n_entries)) | ||||
|             for i, entry in enumerate(entries, 1): | ||||
|                 self.to_screen(u'[download] Downloading video #%s of %s' % (i, n_entries)) | ||||
|                 extra = { | ||||
|                          'playlist': playlist,  | ||||
|                          'playlist_index': i + playliststart, | ||||
|                          } | ||||
|                 if not 'extractor' in entry: | ||||
|                     # We set the extractor, if it's an url it will be set then to | ||||
|                     # the new extractor, but if it's already a video we must make | ||||
|                     # sure it's present: see issue #877 | ||||
|                     entry['extractor'] = ie_result['extractor'] | ||||
|                     'playlist': playlist, | ||||
|                     'playlist_index': i + playliststart, | ||||
|                     'extractor': ie_result['extractor'], | ||||
|                     'webpage_url': ie_result['webpage_url'], | ||||
|                     'extractor_key': ie_result['extractor_key'], | ||||
|                 } | ||||
|                 entry_result = self.process_ie_result(entry, | ||||
|                                                       download=download, | ||||
|                                                       extra_info=extra) | ||||
| @@ -438,16 +490,37 @@ class YoutubeDL(object): | ||||
|             return ie_result | ||||
|         elif result_type == 'compat_list': | ||||
|             def _fixup(r): | ||||
|                 r.setdefault('extractor', ie_result['extractor']) | ||||
|                 self.add_extra_info(r, | ||||
|                     { | ||||
|                         'extractor': ie_result['extractor'], | ||||
|                         'webpage_url': ie_result['webpage_url'], | ||||
|                         'extractor_key': ie_result['extractor_key'], | ||||
|                     }) | ||||
|                 return r | ||||
|             ie_result['entries'] = [ | ||||
|                 self.process_ie_result(_fixup(r), download=download) | ||||
|                 self.process_ie_result(_fixup(r), download, extra_info) | ||||
|                 for r in ie_result['entries'] | ||||
|             ] | ||||
|             return ie_result | ||||
|         else: | ||||
|             raise Exception('Invalid result type: %s' % result_type) | ||||
|  | ||||
|     def select_format(self, format_spec, available_formats): | ||||
|         if format_spec == 'best' or format_spec is None: | ||||
|             return available_formats[-1] | ||||
|         elif format_spec == 'worst': | ||||
|             return available_formats[0] | ||||
|         else: | ||||
|             extensions = [u'mp4', u'flv', u'webm', u'3gp'] | ||||
|             if format_spec in extensions: | ||||
|                 filter_f = lambda f: f['ext'] == format_spec | ||||
|             else: | ||||
|                 filter_f = lambda f: f['format_id'] == format_spec | ||||
|             matches = list(filter(filter_f, available_formats)) | ||||
|             if matches: | ||||
|                 return matches[-1] | ||||
|         return None | ||||
|  | ||||
|     def process_video_result(self, info_dict, download=True): | ||||
|         assert info_dict.get('_type', 'video') == 'video' | ||||
|  | ||||
| @@ -457,8 +530,9 @@ class YoutubeDL(object): | ||||
|             info_dict['playlist_index'] = None | ||||
|  | ||||
|         # This extractors handle format selection themselves | ||||
|         if info_dict['extractor'] in [u'youtube', u'Youku', u'YouPorn', u'mixcloud']: | ||||
|             self.process_info(info_dict) | ||||
|         if info_dict['extractor'] in [u'youtube', u'Youku']: | ||||
|             if download: | ||||
|                 self.process_info(info_dict) | ||||
|             return info_dict | ||||
|  | ||||
|         # We now pick which formats have to be downloaded | ||||
| @@ -470,17 +544,17 @@ class YoutubeDL(object): | ||||
|  | ||||
|         # We check that all the formats have the format and format_id fields | ||||
|         for (i, format) in enumerate(formats): | ||||
|             if format.get('format') is None: | ||||
|                 if format.get('height') is not None: | ||||
|                     if format.get('width') is not None: | ||||
|                         format_desc = u'%sx%s' % (format['width'], format['height']) | ||||
|                     else: | ||||
|                         format_desc = u'%sp' % format['height'] | ||||
|                 else: | ||||
|                     format_desc = '???' | ||||
|                 format['format'] = format_desc | ||||
|             if format.get('format_id') is None: | ||||
|                 format['format_id'] = compat_str(i) | ||||
|             if format.get('format') is None: | ||||
|                 format['format'] = u'{id} - {res}{note}'.format( | ||||
|                     id=format['format_id'], | ||||
|                     res=self.format_resolution(format), | ||||
|                     note=u' ({0})'.format(format['format_note']) if format.get('format_note') is not None else '', | ||||
|                 ) | ||||
|             # Automatically determine file extension if missing | ||||
|             if 'ext' not in format: | ||||
|                 format['ext'] = determine_ext(format['url']) | ||||
|  | ||||
|         if self.params.get('listformats', None): | ||||
|             self.list_formats(info_dict) | ||||
| @@ -502,25 +576,24 @@ class YoutubeDL(object): | ||||
|             formats = sorted(formats, key=_free_formats_key) | ||||
|  | ||||
|         req_format = self.params.get('format', 'best') | ||||
|         if req_format is None: | ||||
|             req_format = 'best' | ||||
|         formats_to_download = [] | ||||
|         if req_format == 'best' or req_format is None: | ||||
|             formats_to_download = [formats[-1]] | ||||
|         elif req_format == 'worst': | ||||
|             formats_to_download = [formats[0]] | ||||
|         # The -1 is for supporting YoutubeIE | ||||
|         elif req_format in ('-1', 'all'): | ||||
|         if req_format in ('-1', 'all'): | ||||
|             formats_to_download = formats | ||||
|         else: | ||||
|             # We can accept formats requestd in the format: 34/10/5, we pick | ||||
|             # We can accept formats requestd in the format: 34/5/best, we pick | ||||
|             # the first that is available, starting from left | ||||
|             req_formats = req_format.split('/') | ||||
|             for rf in req_formats: | ||||
|                 matches = filter(lambda f:f['format_id'] == rf ,formats) | ||||
|                 if matches: | ||||
|                     formats_to_download = [matches[0]] | ||||
|                 selected_format = self.select_format(rf, formats) | ||||
|                 if selected_format is not None: | ||||
|                     formats_to_download = [selected_format] | ||||
|                     break | ||||
|         if not formats_to_download: | ||||
|             raise ExtractorError(u'requested format not available') | ||||
|             raise ExtractorError(u'requested format not available', | ||||
|                                  expected=True) | ||||
|  | ||||
|         if download: | ||||
|             if len(formats_to_download) > 1: | ||||
| @@ -570,9 +643,9 @@ class YoutubeDL(object): | ||||
|         if self.params.get('forceurl', False): | ||||
|             # For RTMP URLs, also include the playpath | ||||
|             compat_print(info_dict['url'] + info_dict.get('play_path', u'')) | ||||
|         if self.params.get('forcethumbnail', False) and 'thumbnail' in info_dict: | ||||
|         if self.params.get('forcethumbnail', False) and info_dict.get('thumbnail') is not None: | ||||
|             compat_print(info_dict['thumbnail']) | ||||
|         if self.params.get('forcedescription', False) and 'description' in info_dict: | ||||
|         if self.params.get('forcedescription', False) and info_dict.get('description') is not None: | ||||
|             compat_print(info_dict['description']) | ||||
|         if self.params.get('forcefilename', False) and filename is not None: | ||||
|             compat_print(filename) | ||||
| @@ -608,24 +681,24 @@ class YoutubeDL(object): | ||||
|  | ||||
|         if self.params.get('writeannotations', False): | ||||
|             try: | ||||
|                annofn = filename + u'.annotations.xml' | ||||
|                self.report_writeannotations(annofn) | ||||
|                with io.open(encodeFilename(annofn), 'w', encoding='utf-8') as annofile: | ||||
|                    annofile.write(info_dict['annotations']) | ||||
|                 annofn = filename + u'.annotations.xml' | ||||
|                 self.report_writeannotations(annofn) | ||||
|                 with io.open(encodeFilename(annofn), 'w', encoding='utf-8') as annofile: | ||||
|                     annofile.write(info_dict['annotations']) | ||||
|             except (KeyError, TypeError): | ||||
|                 self.report_warning(u'There are no annotations to write.') | ||||
|             except (OSError, IOError): | ||||
|                  self.report_error(u'Cannot write annotations file: ' + annofn) | ||||
|                  return | ||||
|                 self.report_error(u'Cannot write annotations file: ' + annofn) | ||||
|                 return | ||||
|  | ||||
|         subtitles_are_requested = any([self.params.get('writesubtitles', False), | ||||
|                                        self.params.get('writeautomaticsub')]) | ||||
|  | ||||
|         if  subtitles_are_requested and 'subtitles' in info_dict and info_dict['subtitles']: | ||||
|         if subtitles_are_requested and 'subtitles' in info_dict and info_dict['subtitles']: | ||||
|             # subtitles download errors are already managed as troubles in relevant IE | ||||
|             # that way it will silently go on when used with unsupporting IE | ||||
|             subtitles = info_dict['subtitles'] | ||||
|             sub_format = self.params.get('subtitlesformat') | ||||
|             sub_format = self.params.get('subtitlesformat', 'srt') | ||||
|             for sub_lang in subtitles.keys(): | ||||
|                 sub = subtitles[sub_lang] | ||||
|                 if sub is None: | ||||
| @@ -643,7 +716,7 @@ class YoutubeDL(object): | ||||
|             infofn = filename + u'.info.json' | ||||
|             self.report_writeinfojson(infofn) | ||||
|             try: | ||||
|                 json_info_dict = dict((k, v) for k,v in info_dict.items() if not k in ['urlhandle']) | ||||
|                 json_info_dict = dict((k, v) for k, v in info_dict.items() if not k in ['urlhandle']) | ||||
|                 write_json_file(json_info_dict, encodeFilename(infofn)) | ||||
|             except (OSError, IOError): | ||||
|                 self.report_error(u'Cannot write metadata to JSON file ' + infofn) | ||||
| @@ -713,7 +786,7 @@ class YoutubeDL(object): | ||||
|         keep_video = None | ||||
|         for pp in self._pps: | ||||
|             try: | ||||
|                 keep_video_wish,new_info = pp.run(info) | ||||
|                 keep_video_wish, new_info = pp.run(info) | ||||
|                 if keep_video_wish is not None: | ||||
|                     if keep_video_wish: | ||||
|                         keep_video = keep_video_wish | ||||
| @@ -752,16 +825,59 @@ class YoutubeDL(object): | ||||
|         with locked_file(fn, 'a', encoding='utf-8') as archive_file: | ||||
|             archive_file.write(vid_id + u'\n') | ||||
|  | ||||
|     @staticmethod | ||||
|     def format_resolution(format, default='unknown'): | ||||
|         if format.get('_resolution') is not None: | ||||
|             return format['_resolution'] | ||||
|         if format.get('height') is not None: | ||||
|             if format.get('width') is not None: | ||||
|                 res = u'%sx%s' % (format['width'], format['height']) | ||||
|             else: | ||||
|                 res = u'%sp' % format['height'] | ||||
|         else: | ||||
|             res = default | ||||
|         return res | ||||
|  | ||||
|     def list_formats(self, info_dict): | ||||
|         formats_s = [] | ||||
|         for format in info_dict.get('formats', [info_dict]): | ||||
|             formats_s.append("%s\t:\t%s\t[%s]" % (format['format_id'], | ||||
|                                                 format['ext'], | ||||
|                                                 format.get('format', '???'), | ||||
|                                                 ) | ||||
|                             ) | ||||
|         if len(formats_s) != 1: | ||||
|             formats_s[0]  += ' (worst)' | ||||
|             formats_s[-1] += ' (best)' | ||||
|         formats_s = "\n".join(formats_s) | ||||
|         self.to_screen(u"[info] Available formats for %s:\nformat code\textension\n%s" % (info_dict['id'], formats_s))  | ||||
|         def format_note(fdict): | ||||
|             if fdict.get('format_note') is not None: | ||||
|                 return fdict['format_note'] | ||||
|             res = u'' | ||||
|             if fdict.get('vcodec') is not None: | ||||
|                 res += u'%-5s' % fdict['vcodec'] | ||||
|             elif fdict.get('vbr') is not None: | ||||
|                 res += u'video' | ||||
|             if fdict.get('vbr') is not None: | ||||
|                 res += u'@%4dk' % fdict['vbr'] | ||||
|             if fdict.get('acodec') is not None: | ||||
|                 if res: | ||||
|                     res += u', ' | ||||
|                 res += u'%-5s' % fdict['acodec'] | ||||
|             elif fdict.get('abr') is not None: | ||||
|                 if res: | ||||
|                     res += u', ' | ||||
|                 res += 'audio' | ||||
|             if fdict.get('abr') is not None: | ||||
|                 res += u'@%3dk' % fdict['abr'] | ||||
|             return res | ||||
|  | ||||
|         def line(format): | ||||
|             return (u'%-20s%-10s%-12s%s' % ( | ||||
|                 format['format_id'], | ||||
|                 format['ext'], | ||||
|                 self.format_resolution(format), | ||||
|                 format_note(format), | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|         formats = info_dict.get('formats', [info_dict]) | ||||
|         formats_s = list(map(line, formats)) | ||||
|         if len(formats) > 1: | ||||
|             formats_s[0] += (' ' if format_note(formats[0]) else '') + '(worst)' | ||||
|             formats_s[-1] += (' ' if format_note(formats[-1]) else '') + '(best)' | ||||
|  | ||||
|         header_line = line({ | ||||
|             'format_id': u'format code', 'ext': u'extension', | ||||
|             '_resolution': u'resolution', 'format_note': u'note'}) | ||||
|         self.to_screen(u'[info] Available formats for %s:\n%s\n%s' % | ||||
|                        (info_dict['id'], header_line, u"\n".join(formats_s))) | ||||
|   | ||||
| @@ -32,6 +32,8 @@ __authors__  = ( | ||||
|     'Ismael Mejía', | ||||
|     'Steffan \'Ruirize\' James', | ||||
|     'Andras Elso', | ||||
|     'Jelle van der Waa', | ||||
|     'Marcin Cieślak', | ||||
| ) | ||||
|  | ||||
| __license__ = 'Public Domain' | ||||
| @@ -133,7 +135,7 @@ def parseOpts(overrideArguments=None): | ||||
|  | ||||
|     def _hide_login_info(opts): | ||||
|         opts = list(opts) | ||||
|         for private_opt in ['-p', '--password', '-u', '--username']: | ||||
|         for private_opt in ['-p', '--password', '-u', '--username', '--video-password']: | ||||
|             try: | ||||
|                 i = opts.index(private_opt) | ||||
|                 opts[i+1] = '<PRIVATE>' | ||||
| @@ -179,6 +181,9 @@ def parseOpts(overrideArguments=None): | ||||
|             action='store_true', dest='update_self', help='update this program to latest version. Make sure that you have sufficient permissions (run with sudo if needed)') | ||||
|     general.add_option('-i', '--ignore-errors', | ||||
|             action='store_true', dest='ignoreerrors', help='continue on download errors, for example to to skip unavailable videos in a playlist', default=False) | ||||
|     general.add_option('--abort-on-error', | ||||
|             action='store_false', dest='ignoreerrors', | ||||
|             help='Abort downloading of further videos (in the playlist or the command line) if an error occurs') | ||||
|     general.add_option('--dump-user-agent', | ||||
|             action='store_true', dest='dump_user_agent', | ||||
|             help='display the current browser identification', default=False) | ||||
| @@ -196,7 +201,7 @@ def parseOpts(overrideArguments=None): | ||||
|     general.add_option('--proxy', dest='proxy', default=None, help='Use the specified HTTP/HTTPS proxy', metavar='URL') | ||||
|     general.add_option('--no-check-certificate', action='store_true', dest='no_check_certificate', default=False, help='Suppress HTTPS certificate validation.') | ||||
|     general.add_option( | ||||
|         '--cache-dir', dest='cachedir', default=get_cachedir(), | ||||
|         '--cache-dir', dest='cachedir', default=get_cachedir(), metavar='DIR', | ||||
|         help='Location in the filesystem where youtube-dl can store downloaded information permanently. By default $XDG_CACHE_HOME/youtube-dl or ~/.cache/youtube-dl .') | ||||
|     general.add_option( | ||||
|         '--no-cache-dir', action='store_const', const=None, dest='cachedir', | ||||
| @@ -313,6 +318,9 @@ def parseOpts(overrideArguments=None): | ||||
|     verbosity.add_option('--dump-intermediate-pages', | ||||
|             action='store_true', dest='dump_intermediate_pages', default=False, | ||||
|             help='print downloaded pages to debug problems(very verbose)') | ||||
|     verbosity.add_option('--write-pages', | ||||
|             action='store_true', dest='write_pages', default=False, | ||||
|             help='Write downloaded pages to files in the current directory') | ||||
|     verbosity.add_option('--youtube-print-sig-code', | ||||
|             action='store_true', dest='youtube_print_sig_code', default=False, | ||||
|             help=optparse.SUPPRESS_HELP) | ||||
| @@ -332,7 +340,10 @@ def parseOpts(overrideArguments=None): | ||||
|             help=('output filename template. Use %(title)s to get the title, ' | ||||
|                   '%(uploader)s for the uploader name, %(uploader_id)s for the uploader nickname if different, ' | ||||
|                   '%(autonumber)s to get an automatically incremented number, ' | ||||
|                   '%(ext)s for the filename extension, %(upload_date)s for the upload date (YYYYMMDD), ' | ||||
|                   '%(ext)s for the filename extension, ' | ||||
|                   '%(format)s for the format description (like "22 - 1280x720" or "HD"),' | ||||
|                   '%(format_id)s for the unique id of the format (like Youtube\'s itags: "137"),' | ||||
|                   '%(upload_date)s for the upload date (YYYYMMDD), ' | ||||
|                   '%(extractor)s for the provider (youtube, metacafe, etc), ' | ||||
|                   '%(id)s for the video id , %(playlist)s for the playlist the video is in, ' | ||||
|                   '%(playlist_index)s for the position in the playlist and %% for a literal percent. ' | ||||
| @@ -340,7 +351,7 @@ def parseOpts(overrideArguments=None): | ||||
|                   'for example with -o \'/my/downloads/%(uploader)s/%(title)s-%(id)s.%(ext)s\' .')) | ||||
|     filesystem.add_option('--autonumber-size', | ||||
|             dest='autonumber_size', metavar='NUMBER', | ||||
|             help='Specifies the number of digits in %(autonumber)s when it is present in output filename template or --autonumber option is given') | ||||
|             help='Specifies the number of digits in %(autonumber)s when it is present in output filename template or --auto-number option is given') | ||||
|     filesystem.add_option('--restrict-filenames', | ||||
|             action='store_true', dest='restrictfilenames', | ||||
|             help='Restrict filenames to only ASCII characters, and avoid "&" and spaces in filenames', default=False) | ||||
| @@ -349,7 +360,7 @@ def parseOpts(overrideArguments=None): | ||||
|     filesystem.add_option('-w', '--no-overwrites', | ||||
|             action='store_true', dest='nooverwrites', help='do not overwrite files', default=False) | ||||
|     filesystem.add_option('-c', '--continue', | ||||
|             action='store_true', dest='continue_dl', help='resume partially downloaded files', default=True) | ||||
|             action='store_true', dest='continue_dl', help='force resume of partially downloaded files. By default, youtube-dl will resume downloads if possible.', default=True) | ||||
|     filesystem.add_option('--no-continue', | ||||
|             action='store_false', dest='continue_dl', | ||||
|             help='do not resume partially downloaded files (restart from beginning)') | ||||
| @@ -592,8 +603,7 @@ def _real_main(argv=None): | ||||
|                      u' file! Use "%%(ext)s" instead of %r' % | ||||
|                      determine_ext(outtmpl, u'')) | ||||
|  | ||||
|     # YoutubeDL | ||||
|     ydl = YoutubeDL({ | ||||
|     ydl_opts = { | ||||
|         'usenetrc': opts.usenetrc, | ||||
|         'username': opts.username, | ||||
|         'password': opts.password, | ||||
| @@ -646,6 +656,7 @@ def _real_main(argv=None): | ||||
|         'prefer_free_formats': opts.prefer_free_formats, | ||||
|         'verbose': opts.verbose, | ||||
|         'dump_intermediate_pages': opts.dump_intermediate_pages, | ||||
|         'write_pages': opts.write_pages, | ||||
|         'test': opts.test, | ||||
|         'keepvideo': opts.keepvideo, | ||||
|         'min_filesize': opts.min_filesize, | ||||
| @@ -655,61 +666,63 @@ def _real_main(argv=None): | ||||
|         'youtube_print_sig_code': opts.youtube_print_sig_code, | ||||
|         'age_limit': opts.age_limit, | ||||
|         'download_archive': opts.download_archive, | ||||
|         }) | ||||
|     } | ||||
|  | ||||
|     if opts.verbose: | ||||
|         write_string(u'[debug] youtube-dl version ' + __version__ + u'\n') | ||||
|         try: | ||||
|             sp = subprocess.Popen( | ||||
|                 ['git', 'rev-parse', '--short', 'HEAD'], | ||||
|                 stdout=subprocess.PIPE, stderr=subprocess.PIPE, | ||||
|                 cwd=os.path.dirname(os.path.abspath(__file__))) | ||||
|             out, err = sp.communicate() | ||||
|             out = out.decode().strip() | ||||
|             if re.match('[0-9a-f]+', out): | ||||
|                 write_string(u'[debug] Git HEAD: ' + out + u'\n') | ||||
|         except: | ||||
|     with YoutubeDL(ydl_opts) as ydl: | ||||
|         if opts.verbose: | ||||
|             write_string(u'[debug] youtube-dl version ' + __version__ + u'\n') | ||||
|             try: | ||||
|                 sys.exc_clear() | ||||
|                 sp = subprocess.Popen( | ||||
|                     ['git', 'rev-parse', '--short', 'HEAD'], | ||||
|                     stdout=subprocess.PIPE, stderr=subprocess.PIPE, | ||||
|                     cwd=os.path.dirname(os.path.abspath(__file__))) | ||||
|                 out, err = sp.communicate() | ||||
|                 out = out.decode().strip() | ||||
|                 if re.match('[0-9a-f]+', out): | ||||
|                     write_string(u'[debug] Git HEAD: ' + out + u'\n') | ||||
|             except: | ||||
|                 pass | ||||
|         write_string(u'[debug] Python version %s - %s' %(platform.python_version(), platform_name()) + u'\n') | ||||
|                 try: | ||||
|                     sys.exc_clear() | ||||
|                 except: | ||||
|                     pass | ||||
|             write_string(u'[debug] Python version %s - %s' % | ||||
|                          (platform.python_version(), platform_name()) + u'\n') | ||||
|  | ||||
|         proxy_map = {} | ||||
|         for handler in opener.handlers: | ||||
|             if hasattr(handler, 'proxies'): | ||||
|                 proxy_map.update(handler.proxies) | ||||
|         write_string(u'[debug] Proxy map: ' + compat_str(proxy_map) + u'\n') | ||||
|             proxy_map = {} | ||||
|             for handler in opener.handlers: | ||||
|                 if hasattr(handler, 'proxies'): | ||||
|                     proxy_map.update(handler.proxies) | ||||
|             write_string(u'[debug] Proxy map: ' + compat_str(proxy_map) + u'\n') | ||||
|  | ||||
|     ydl.add_default_info_extractors() | ||||
|         ydl.add_default_info_extractors() | ||||
|  | ||||
|     # PostProcessors | ||||
|     # Add the metadata pp first, the other pps will copy it | ||||
|     if opts.addmetadata: | ||||
|         ydl.add_post_processor(FFmpegMetadataPP()) | ||||
|     if opts.extractaudio: | ||||
|         ydl.add_post_processor(FFmpegExtractAudioPP(preferredcodec=opts.audioformat, preferredquality=opts.audioquality, nopostoverwrites=opts.nopostoverwrites)) | ||||
|     if opts.recodevideo: | ||||
|         ydl.add_post_processor(FFmpegVideoConvertor(preferedformat=opts.recodevideo)) | ||||
|     if opts.embedsubtitles: | ||||
|         ydl.add_post_processor(FFmpegEmbedSubtitlePP(subtitlesformat=opts.subtitlesformat)) | ||||
|         # PostProcessors | ||||
|         # Add the metadata pp first, the other pps will copy it | ||||
|         if opts.addmetadata: | ||||
|             ydl.add_post_processor(FFmpegMetadataPP()) | ||||
|         if opts.extractaudio: | ||||
|             ydl.add_post_processor(FFmpegExtractAudioPP(preferredcodec=opts.audioformat, preferredquality=opts.audioquality, nopostoverwrites=opts.nopostoverwrites)) | ||||
|         if opts.recodevideo: | ||||
|             ydl.add_post_processor(FFmpegVideoConvertor(preferedformat=opts.recodevideo)) | ||||
|         if opts.embedsubtitles: | ||||
|             ydl.add_post_processor(FFmpegEmbedSubtitlePP(subtitlesformat=opts.subtitlesformat)) | ||||
|  | ||||
|     # Update version | ||||
|     if opts.update_self: | ||||
|         update_self(ydl.to_screen, opts.verbose) | ||||
|         # Update version | ||||
|         if opts.update_self: | ||||
|             update_self(ydl.to_screen, opts.verbose) | ||||
|  | ||||
|     # Maybe do nothing | ||||
|     if len(all_urls) < 1: | ||||
|         if not opts.update_self: | ||||
|             parser.error(u'you must provide at least one URL') | ||||
|         else: | ||||
|             sys.exit() | ||||
|         # Maybe do nothing | ||||
|         if len(all_urls) < 1: | ||||
|             if not opts.update_self: | ||||
|                 parser.error(u'you must provide at least one URL') | ||||
|             else: | ||||
|                 sys.exit() | ||||
|  | ||||
|     try: | ||||
|         retcode = ydl.download(all_urls) | ||||
|     except MaxDownloadsReached: | ||||
|         ydl.to_screen(u'--max-download limit reached, aborting.') | ||||
|         retcode = 101 | ||||
|         try: | ||||
|             retcode = ydl.download(all_urls) | ||||
|         except MaxDownloadsReached: | ||||
|             ydl.to_screen(u'--max-download limit reached, aborting.') | ||||
|             retcode = 101 | ||||
|  | ||||
|     # Dump cookie jar if requested | ||||
|     if opts.cookiefile is not None: | ||||
|   | ||||
| @@ -9,6 +9,7 @@ from .arte import ( | ||||
|     ArteTVFutureIE, | ||||
| ) | ||||
| from .auengine import AUEngineIE | ||||
| from .bambuser import BambuserIE, BambuserChannelIE | ||||
| from .bandcamp import BandcampIE | ||||
| from .bliptv import BlipTVIE, BlipTVUserIE | ||||
| from .bloomberg import BloombergIE | ||||
| @@ -37,8 +38,10 @@ from .defense import DefenseGouvFrIE | ||||
| from .ebaumsworld import EbaumsWorldIE | ||||
| from .ehow import EHowIE | ||||
| from .eighttracks import EightTracksIE | ||||
| from .eitb import EitbIE | ||||
| from .escapist import EscapistIE | ||||
| from .exfm import ExfmIE | ||||
| from .extremetube import ExtremeTubeIE | ||||
| from .facebook import FacebookIE | ||||
| from .faz import FazIE | ||||
| from .fktv import ( | ||||
| @@ -54,6 +57,7 @@ from .francetv import ( | ||||
| ) | ||||
| from .freesound import FreesoundIE | ||||
| from .funnyordie import FunnyOrDieIE | ||||
| from .gamekings import GamekingsIE | ||||
| from .gamespot import GameSpotIE | ||||
| from .gametrailers import GametrailersIE | ||||
| from .generic import GenericIE | ||||
| @@ -72,16 +76,19 @@ from .jeuxvideo import JeuxVideoIE | ||||
| from .jukebox import JukeboxIE | ||||
| from .justintv import JustinTVIE | ||||
| from .kankan import KankanIE | ||||
| from .keezmovies import KeezMoviesIE | ||||
| from .kickstarter import KickStarterIE | ||||
| from .keek import KeekIE | ||||
| from .liveleak import LiveLeakIE | ||||
| from .livestream import LivestreamIE | ||||
| from .livestream import LivestreamIE, LivestreamOriginalIE | ||||
| from .metacafe import MetacafeIE | ||||
| from .metacritic import MetacriticIE | ||||
| from .mit import TechTVMITIE, MITIE | ||||
| from .mixcloud import MixcloudIE | ||||
| from .mofosex import MofosexIE | ||||
| from .mtv import MTVIE | ||||
| from .muzu import MuzuTVIE | ||||
| from .myspace import MySpaceIE | ||||
| from .myspass import MySpassIE | ||||
| from .myvideo import MyVideoIE | ||||
| from .naver import NaverIE | ||||
| @@ -94,6 +101,7 @@ from .ooyala import OoyalaIE | ||||
| from .orf import ORFIE | ||||
| from .pbs import PBSIE | ||||
| from .photobucket import PhotobucketIE | ||||
| from .pornhub import PornHubIE | ||||
| from .pornotube import PornotubeIE | ||||
| from .rbmaradio import RBMARadioIE | ||||
| from .redtube import RedTubeIE | ||||
| @@ -108,7 +116,12 @@ from .slashdot import SlashdotIE | ||||
| from .slideshare import SlideshareIE | ||||
| from .sohu import SohuIE | ||||
| from .soundcloud import SoundcloudIE, SoundcloudSetIE, SoundcloudUserIE | ||||
| from .southparkstudios import SouthParkStudiosIE | ||||
| from .southparkstudios import ( | ||||
|     SouthParkStudiosIE, | ||||
|     SouthparkDeIE, | ||||
| ) | ||||
| from .space import SpaceIE | ||||
| from .spankwire import SpankwireIE | ||||
| from .spiegel import SpiegelIE | ||||
| from .stanfordoc import StanfordOpenClassroomIE | ||||
| from .statigram import StatigramIE | ||||
| @@ -121,9 +134,11 @@ from .tf1 import TF1IE | ||||
| from .thisav import ThisAVIE | ||||
| from .traileraddict import TrailerAddictIE | ||||
| from .trilulilu import TriluliluIE | ||||
| from .tube8 import Tube8IE | ||||
| from .tudou import TudouIE | ||||
| from .tumblr import TumblrIE | ||||
| from .tutv import TutvIE | ||||
| from .tvp import TvpIE | ||||
| from .unistra import UnistraIE | ||||
| from .ustream import UstreamIE, UstreamChannelIE | ||||
| from .vbox7 import Vbox7IE | ||||
| @@ -137,6 +152,7 @@ from .videofyme import VideofyMeIE | ||||
| from .videopremium import VideoPremiumIE | ||||
| from .vimeo import VimeoIE, VimeoChannelIE | ||||
| from .vine import VineIE | ||||
| from .vk import VKIE | ||||
| from .wat import WatIE | ||||
| from .websurg import WeBSurgIE | ||||
| from .weibo import WeiboIE | ||||
| @@ -145,6 +161,7 @@ from .worldstarhiphop import WorldStarHipHopIE | ||||
| from .xhamster import XHamsterIE | ||||
| from .xnxx import XNXXIE | ||||
| from .xvideos import XVideosIE | ||||
| from .xtube import XTubeIE | ||||
| from .yahoo import YahooIE, YahooSearchIE | ||||
| from .youjizz import YouJizzIE | ||||
| from .youku import YoukuIE | ||||
| @@ -153,6 +170,7 @@ from .youtube import ( | ||||
|     YoutubeIE, | ||||
|     YoutubePlaylistIE, | ||||
|     YoutubeSearchIE, | ||||
|     YoutubeSearchDateIE, | ||||
|     YoutubeUserIE, | ||||
|     YoutubeChannelIE, | ||||
|     YoutubeShowIE, | ||||
|   | ||||
| @@ -17,8 +17,8 @@ class AddAnimeIE(InfoExtractor): | ||||
|     IE_NAME = u'AddAnime' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.add-anime.net/watch_video.php?v=24MR3YO5SAS9', | ||||
|         u'file': u'24MR3YO5SAS9.flv', | ||||
|         u'md5': u'1036a0e0cd307b95bd8a8c3a5c8cfaf1', | ||||
|         u'file': u'24MR3YO5SAS9.mp4', | ||||
|         u'md5': u'72954ea10bc979ab5e2eb288b21425a0', | ||||
|         u'info_dict': { | ||||
|             u"description": u"One Piece 606", | ||||
|             u"title": u"One Piece 606" | ||||
| @@ -31,7 +31,8 @@ class AddAnimeIE(InfoExtractor): | ||||
|             video_id = mobj.group('video_id') | ||||
|             webpage = self._download_webpage(url, video_id) | ||||
|         except ExtractorError as ee: | ||||
|             if not isinstance(ee.cause, compat_HTTPError): | ||||
|             if not isinstance(ee.cause, compat_HTTPError) or \ | ||||
|                ee.cause.code != 503: | ||||
|                 raise | ||||
|  | ||||
|             redir_webpage = ee.cause.read().decode('utf-8') | ||||
| @@ -60,16 +61,26 @@ class AddAnimeIE(InfoExtractor): | ||||
|                 note=u'Confirming after redirect') | ||||
|             webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         video_url = self._search_regex(r"var normal_video_file = '(.*?)';", | ||||
|                                        webpage, u'video file URL') | ||||
|         formats = [] | ||||
|         for format_id in ('normal', 'hq'): | ||||
|             rex = r"var %s_video_file = '(.*?)';" % re.escape(format_id) | ||||
|             video_url = self._search_regex(rex, webpage, u'video file URLx', | ||||
|                                            fatal=False) | ||||
|             if not video_url: | ||||
|                 continue | ||||
|             formats.append({ | ||||
|                 'format_id': format_id, | ||||
|                 'url': video_url, | ||||
|             }) | ||||
|         if not formats: | ||||
|             raise ExtractorError(u'Cannot find any video format!') | ||||
|         video_title = self._og_search_title(webpage) | ||||
|         video_description = self._og_search_description(webpage) | ||||
|  | ||||
|         return { | ||||
|             '_type': 'video', | ||||
|             'id':  video_id, | ||||
|             'url': video_url, | ||||
|             'ext': 'flv', | ||||
|             'formats': formats, | ||||
|             'title': video_title, | ||||
|             'description': video_description | ||||
|         } | ||||
|   | ||||
| @@ -10,6 +10,7 @@ from ..utils import ( | ||||
|     unified_strdate, | ||||
|     determine_ext, | ||||
|     get_element_by_id, | ||||
|     compat_str, | ||||
| ) | ||||
|  | ||||
| # There are different sources of video in arte.tv, the extraction process  | ||||
| @@ -68,7 +69,7 @@ class ArteTvIE(InfoExtractor): | ||||
|             lang = mobj.group('lang') | ||||
|             return self._extract_liveweb(url, name, lang) | ||||
|  | ||||
|         if re.search(self._LIVE_URL, video_id) is not None: | ||||
|         if re.search(self._LIVE_URL, url) is not None: | ||||
|             raise ExtractorError(u'Arte live streams are not yet supported, sorry') | ||||
|             # self.extractLiveStream(url) | ||||
|             # return | ||||
| @@ -114,7 +115,7 @@ class ArteTvIE(InfoExtractor): | ||||
|         event_doc = config_doc.find('event') | ||||
|         url_node = event_doc.find('video').find('urlHd') | ||||
|         if url_node is None: | ||||
|             url_node = video_doc.find('urlSd') | ||||
|             url_node = event_doc.find('urlSd') | ||||
|  | ||||
|         return {'id': video_id, | ||||
|                 'title': event_doc.find('name%s' % lang.capitalize()).text, | ||||
| @@ -158,7 +159,9 @@ class ArteTVPlus7IE(InfoExtractor): | ||||
|             'thumbnail': player_info.get('programImage') or player_info.get('VTU', {}).get('IUR'), | ||||
|         } | ||||
|  | ||||
|         formats = player_info['VSR'].values() | ||||
|         all_formats = player_info['VSR'].values() | ||||
|         # Some formats use the m3u8 protocol | ||||
|         all_formats = list(filter(lambda f: f.get('videoFormat') != 'M3U8', all_formats)) | ||||
|         def _match_lang(f): | ||||
|             if f.get('versionCode') is None: | ||||
|                 return True | ||||
| @@ -170,18 +173,48 @@ class ArteTVPlus7IE(InfoExtractor): | ||||
|             regexes = [r'VO?%s' % l, r'VO?.-ST%s' % l] | ||||
|             return any(re.match(r, f['versionCode']) for r in regexes) | ||||
|         # Some formats may not be in the same language as the url | ||||
|         formats = filter(_match_lang, formats) | ||||
|         # Some formats use the m3u8 protocol | ||||
|         formats = filter(lambda f: f.get('videoFormat') != 'M3U8', formats) | ||||
|         # We order the formats by quality | ||||
|         formats = sorted(formats, key=lambda f: int(f.get('height',-1))) | ||||
|         # Prefer videos without subtitles in the same language | ||||
|         formats = sorted(formats, key=lambda f: re.match(r'VO(F|A)-STM\1', f.get('versionCode', '')) is None) | ||||
|         # Pick the best quality | ||||
|         formats = filter(_match_lang, all_formats) | ||||
|         formats = list(formats) # in python3 filter returns an iterator | ||||
|         if not formats: | ||||
|             # Some videos are only available in the 'Originalversion' | ||||
|             # they aren't tagged as being in French or German | ||||
|             if all(f['versionCode'] == 'VO' for f in all_formats): | ||||
|                 formats = all_formats | ||||
|             else: | ||||
|                 raise ExtractorError(u'The formats list is empty') | ||||
|  | ||||
|         if re.match(r'[A-Z]Q', formats[0]['quality']) is not None: | ||||
|             def sort_key(f): | ||||
|                 return ['HQ', 'MQ', 'EQ', 'SQ'].index(f['quality']) | ||||
|         else: | ||||
|             def sort_key(f): | ||||
|                 return ( | ||||
|                     # Sort first by quality | ||||
|                     int(f.get('height',-1)), | ||||
|                     int(f.get('bitrate',-1)), | ||||
|                     # The original version with subtitles has lower relevance | ||||
|                     re.match(r'VO-ST(F|A)', f.get('versionCode', '')) is None, | ||||
|                     # The version with sourds/mal subtitles has also lower relevance | ||||
|                     re.match(r'VO?(F|A)-STM\1', f.get('versionCode', '')) is None, | ||||
|                 ) | ||||
|         formats = sorted(formats, key=sort_key) | ||||
|         def _format(format_info): | ||||
|             quality = '' | ||||
|             height = format_info.get('height') | ||||
|             if height is not None: | ||||
|                 quality = compat_str(height) | ||||
|             bitrate = format_info.get('bitrate') | ||||
|             if bitrate is not None: | ||||
|                 quality += '-%d' % bitrate | ||||
|             if format_info.get('versionCode') is not None: | ||||
|                 format_id = u'%s-%s' % (quality, format_info['versionCode']) | ||||
|             else: | ||||
|                 format_id = quality | ||||
|             info = { | ||||
|                 'format_id': format_id, | ||||
|                 'format_note': format_info.get('versionLibelle'), | ||||
|                 'width': format_info.get('width'), | ||||
|                 'height': format_info.get('height'), | ||||
|                 'height': height, | ||||
|             } | ||||
|             if format_info['mediaType'] == u'rtmp': | ||||
|                 info['url'] = format_info['streamer'] | ||||
| @@ -192,8 +225,6 @@ class ArteTVPlus7IE(InfoExtractor): | ||||
|                 info['ext'] = determine_ext(info['url']) | ||||
|             return info | ||||
|         info_dict['formats'] = [_format(f) for f in formats] | ||||
|         # TODO: Remove when #980 has been merged  | ||||
|         info_dict.update(info_dict['formats'][-1]) | ||||
|  | ||||
|         return info_dict | ||||
|  | ||||
| @@ -207,7 +238,7 @@ class ArteTVCreativeIE(ArteTVPlus7IE): | ||||
|         u'url': u'http://creative.arte.tv/de/magazin/agentur-amateur-corporate-design', | ||||
|         u'file': u'050489-002.mp4', | ||||
|         u'info_dict': { | ||||
|             u'title': u'Agentur Amateur #2 - Corporate Design', | ||||
|             u'title': u'Agentur Amateur / Agence Amateur #2 : Corporate Design', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -1,10 +1,10 @@ | ||||
| import os.path | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urllib_parse, | ||||
|     compat_urllib_parse_urlparse, | ||||
|     determine_ext, | ||||
|     ExtractorError, | ||||
| ) | ||||
|  | ||||
| class AUEngineIE(InfoExtractor): | ||||
| @@ -25,22 +25,25 @@ class AUEngineIE(InfoExtractor): | ||||
|         title = self._html_search_regex(r'<title>(?P<title>.+?)</title>', | ||||
|                 webpage, u'title') | ||||
|         title = title.strip() | ||||
|         links = re.findall(r'[^A-Za-z0-9]?(?:file|url):\s*["\'](http[^\'"&]*)', webpage) | ||||
|         links = [compat_urllib_parse.unquote(l) for l in links] | ||||
|         links = re.findall(r'\s(?:file|url):\s*["\']([^\'"]+)["\']', webpage) | ||||
|         links = map(compat_urllib_parse.unquote, links) | ||||
|  | ||||
|         thumbnail = None | ||||
|         video_url = None | ||||
|         for link in links: | ||||
|             root, pathext = os.path.splitext(compat_urllib_parse_urlparse(link).path) | ||||
|             if pathext == '.png': | ||||
|             if link.endswith('.png'): | ||||
|                 thumbnail = link | ||||
|             elif pathext == '.mp4': | ||||
|                 url = link | ||||
|                 ext = pathext | ||||
|             elif '/videos/' in link: | ||||
|                 video_url = link | ||||
|         if not video_url: | ||||
|             raise ExtractorError(u'Could not find video URL') | ||||
|         ext = u'.' + determine_ext(video_url) | ||||
|         if ext == title[-len(ext):]: | ||||
|             title = title[:-len(ext)] | ||||
|         ext = ext[1:] | ||||
|         return [{ | ||||
|  | ||||
|         return { | ||||
|             'id':        video_id, | ||||
|             'url':       url, | ||||
|             'ext':       ext, | ||||
|             'url':       video_url, | ||||
|             'title':     title, | ||||
|             'thumbnail': thumbnail, | ||||
|         }] | ||||
|         } | ||||
|   | ||||
							
								
								
									
										81
									
								
								youtube_dl/extractor/bambuser.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								youtube_dl/extractor/bambuser.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,81 @@ | ||||
| import re | ||||
| import json | ||||
| import itertools | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urllib_request, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class BambuserIE(InfoExtractor): | ||||
|     IE_NAME = u'bambuser' | ||||
|     _VALID_URL = r'https?://bambuser\.com/v/(?P<id>\d+)' | ||||
|     _API_KEY = '005f64509e19a868399060af746a00aa' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'url': u'http://bambuser.com/v/4050584', | ||||
|         # MD5 seems to be flaky, see https://travis-ci.org/rg3/youtube-dl/jobs/14051016#L388 | ||||
|         #u'md5': u'fba8f7693e48fd4e8641b3fd5539a641', | ||||
|         u'info_dict': { | ||||
|             u'id': u'4050584', | ||||
|             u'ext': u'flv', | ||||
|             u'title': u'Education engineering days - lightning talks', | ||||
|             u'duration': 3741, | ||||
|             u'uploader': u'pixelversity', | ||||
|             u'uploader_id': u'344706', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         info_url = ('http://player-c.api.bambuser.com/getVideo.json?' | ||||
|             '&api_key=%s&vid=%s' % (self._API_KEY, video_id)) | ||||
|         info_json = self._download_webpage(info_url, video_id) | ||||
|         info = json.loads(info_json)['result'] | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': info['title'], | ||||
|             'url': info['url'], | ||||
|             'thumbnail': info.get('preview'), | ||||
|             'duration': int(info['length']), | ||||
|             'view_count': int(info['views_total']), | ||||
|             'uploader': info['username'], | ||||
|             'uploader_id': info['uid'], | ||||
|         } | ||||
|  | ||||
|  | ||||
| class BambuserChannelIE(InfoExtractor): | ||||
|     IE_NAME = u'bambuser:channel' | ||||
|     _VALID_URL = r'http://bambuser.com/channel/(?P<user>.*?)(?:/|#|\?|$)' | ||||
|     # The maximum number we can get with each request | ||||
|     _STEP = 50 | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         user = mobj.group('user') | ||||
|         urls = [] | ||||
|         last_id = '' | ||||
|         for i in itertools.count(1): | ||||
|             req_url = ('http://bambuser.com/xhr-api/index.php?username={user}' | ||||
|                 '&sort=created&access_mode=0%2C1%2C2&limit={count}' | ||||
|                 '&method=broadcast&format=json&vid_older_than={last}' | ||||
|                 ).format(user=user, count=self._STEP, last=last_id) | ||||
|             req = compat_urllib_request.Request(req_url) | ||||
|             # Without setting this header, we wouldn't get any result | ||||
|             req.add_header('Referer', 'http://bambuser.com/channel/%s' % user) | ||||
|             info_json = self._download_webpage(req, user, | ||||
|                 u'Downloading page %d' % i) | ||||
|             results = json.loads(info_json)['result'] | ||||
|             if len(results) == 0: | ||||
|                 break | ||||
|             last_id = results[-1]['vid'] | ||||
|             urls.extend(self.url_result(v['page'], 'Bambuser') for v in results) | ||||
|  | ||||
|         return { | ||||
|             '_type': 'playlist', | ||||
|             'title': user, | ||||
|             'entries': urls, | ||||
|         } | ||||
| @@ -9,10 +9,13 @@ from ..utils import ( | ||||
|     compat_urllib_parse, | ||||
|     find_xpath_attr, | ||||
|     compat_urlparse, | ||||
|     compat_str, | ||||
|     compat_urllib_request, | ||||
|  | ||||
|     ExtractorError, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class BrightcoveIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://.*brightcove\.com/(services|viewer).*\?(?P<query>.*)' | ||||
|     _FEDERATED_URL_TEMPLATE = 'http://c.brightcove.com/services/viewer/htmlFederated?%s' | ||||
| @@ -23,7 +26,7 @@ class BrightcoveIE(InfoExtractor): | ||||
|             # From http://www.8tv.cat/8aldia/videos/xavier-sala-i-martin-aquesta-tarda-a-8-al-dia/ | ||||
|             u'url': u'http://c.brightcove.com/services/viewer/htmlFederated?playerID=1654948606001&flashID=myExperience&%40videoPlayer=2371591881001', | ||||
|             u'file': u'2371591881001.mp4', | ||||
|             u'md5': u'9e80619e0a94663f0bdc849b4566af19', | ||||
|             u'md5': u'8eccab865181d29ec2958f32a6a754f5', | ||||
|             u'note': u'Test Brightcove downloads and detection in GenericIE', | ||||
|             u'info_dict': { | ||||
|                 u'title': u'Xavier Sala i Martín: “Un banc que no presta és un banc zombi que no serveix per a res”', | ||||
| @@ -41,6 +44,17 @@ class BrightcoveIE(InfoExtractor): | ||||
|                 u'uploader': u'Oracle', | ||||
|             }, | ||||
|         }, | ||||
|         { | ||||
|             # From http://mashable.com/2013/10/26/thermoelectric-bracelet-lets-you-control-your-body-temperature/ | ||||
|             u'url': u'http://c.brightcove.com/services/viewer/federated_f9?&playerID=1265504713001&publisherID=AQ%7E%7E%2CAAABBzUwv1E%7E%2CxP-xFHVUstiMFlNYfvF4G9yFnNaqCw_9&videoID=2750934548001', | ||||
|             u'info_dict': { | ||||
|                 u'id': u'2750934548001', | ||||
|                 u'ext': u'mp4', | ||||
|                 u'title': u'This Bracelet Acts as a Personal Thermostat', | ||||
|                 u'description': u'md5:547b78c64f4112766ccf4e151c20b6a0', | ||||
|                 u'uploader': u'Mashable', | ||||
|             }, | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     @classmethod | ||||
| @@ -68,24 +82,48 @@ class BrightcoveIE(InfoExtractor): | ||||
|         videoPlayer = find_xpath_attr(object_doc, './param', 'name', '@videoPlayer') | ||||
|         if videoPlayer is not None: | ||||
|             params['@videoPlayer'] = videoPlayer.attrib['value'] | ||||
|         linkBase = find_xpath_attr(object_doc, './param', 'name', 'linkBaseURL') | ||||
|         if linkBase is not None: | ||||
|             params['linkBaseURL'] = linkBase.attrib['value'] | ||||
|         data = compat_urllib_parse.urlencode(params) | ||||
|         return cls._FEDERATED_URL_TEMPLATE % data | ||||
|  | ||||
|     @classmethod | ||||
|     def _extract_brightcove_url(cls, webpage): | ||||
|         """Try to extract the brightcove url from the wepbage, returns None | ||||
|         if it can't be found | ||||
|         """ | ||||
|         m_brightcove = re.search( | ||||
|             r'<object[^>]+?class=([\'"])[^>]*?BrightcoveExperience.*?\1.+?</object>', | ||||
|             webpage, re.DOTALL) | ||||
|         if m_brightcove is not None: | ||||
|             return cls._build_brighcove_url(m_brightcove.group()) | ||||
|         else: | ||||
|             return None | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         # Change the 'videoId' and others field to '@videoPlayer' | ||||
|         url = re.sub(r'(?<=[?&])(videoI(d|D)|bctid)', '%40videoPlayer', url) | ||||
|         # Change bckey (used by bcove.me urls) to playerKey | ||||
|         url = re.sub(r'(?<=[?&])bckey', 'playerKey', url) | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         query_str = mobj.group('query') | ||||
|         query = compat_urlparse.parse_qs(query_str) | ||||
|  | ||||
|         videoPlayer = query.get('@videoPlayer') | ||||
|         if videoPlayer: | ||||
|             return self._get_video_info(videoPlayer[0], query_str) | ||||
|             return self._get_video_info(videoPlayer[0], query_str, query) | ||||
|         else: | ||||
|             player_key = query['playerKey'] | ||||
|             return self._get_playlist_info(player_key[0]) | ||||
|  | ||||
|     def _get_video_info(self, video_id, query): | ||||
|         request_url = self._FEDERATED_URL_TEMPLATE % query | ||||
|         webpage = self._download_webpage(request_url, video_id) | ||||
|     def _get_video_info(self, video_id, query_str, query): | ||||
|         request_url = self._FEDERATED_URL_TEMPLATE % query_str | ||||
|         req = compat_urllib_request.Request(request_url) | ||||
|         linkBase = query.get('linkBaseURL') | ||||
|         if linkBase is not None: | ||||
|             req.add_header('Referer', linkBase[0]) | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         self.report_extraction(video_id) | ||||
|         info = self._search_regex(r'var experienceJSON = ({.*?});', webpage, 'json') | ||||
| @@ -109,7 +147,7 @@ class BrightcoveIE(InfoExtractor): | ||||
|  | ||||
|     def _extract_video_info(self, video_info): | ||||
|         info = { | ||||
|             'id': video_info['id'], | ||||
|             'id': compat_str(video_info['id']), | ||||
|             'title': video_info['displayName'], | ||||
|             'description': video_info.get('shortDescription'), | ||||
|             'thumbnail': video_info.get('videoStillURL') or video_info.get('thumbnailURL'), | ||||
| @@ -119,15 +157,14 @@ class BrightcoveIE(InfoExtractor): | ||||
|         renditions = video_info.get('renditions') | ||||
|         if renditions: | ||||
|             renditions = sorted(renditions, key=lambda r: r['size']) | ||||
|             best_format = renditions[-1] | ||||
|             info.update({ | ||||
|                 'url': best_format['defaultURL'], | ||||
|                 'ext': 'mp4', | ||||
|             }) | ||||
|             info['formats'] = [{ | ||||
|                 'url': rend['defaultURL'], | ||||
|                 'height': rend.get('frameHeight'), | ||||
|                 'width': rend.get('frameWidth'), | ||||
|             } for rend in renditions] | ||||
|         elif video_info.get('FLVFullLengthURL') is not None: | ||||
|             info.update({ | ||||
|                 'url': video_info['FLVFullLengthURL'], | ||||
|                 'ext': 'flv', | ||||
|             }) | ||||
|         else: | ||||
|             raise ExtractorError(u'Unable to extract video url for %s' % info['id']) | ||||
|   | ||||
| @@ -6,7 +6,7 @@ from .common import InfoExtractor | ||||
|  | ||||
| class Canalc2IE(InfoExtractor): | ||||
|     IE_NAME = 'canalc2.tv' | ||||
|     _VALID_URL = r'http://.*?\.canalc2\.tv/video\.asp\?idVideo=(\d+)&voir=oui' | ||||
|     _VALID_URL = r'http://.*?\.canalc2\.tv/video\.asp\?.*?idVideo=(?P<id>\d+)' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.canalc2.tv/video.asp?idVideo=12163&voir=oui', | ||||
| @@ -18,7 +18,9 @@ class Canalc2IE(InfoExtractor): | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = re.match(self._VALID_URL, url).group(1) | ||||
|         video_id = re.match(self._VALID_URL, url).group('id') | ||||
|         # We need to set the voir field for getting the file name | ||||
|         url = 'http://www.canalc2.tv/video.asp?idVideo=%s&voir=oui' % video_id | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         file_name = self._search_regex( | ||||
|             r"so\.addVariable\('file','(.*?)'\);", | ||||
|   | ||||
| @@ -41,7 +41,7 @@ class CinemassacreIE(InfoExtractor): | ||||
|         webpage_url = u'http://' + mobj.group('url') | ||||
|         webpage = self._download_webpage(webpage_url, None) # Don't know video id yet | ||||
|         video_date = mobj.group('date_Y') + mobj.group('date_m') + mobj.group('date_d') | ||||
|         mobj = re.search(r'src="(?P<embed_url>http://player\.screenwavemedia\.com/play/(?:embed|player)\.php\?id=(?:Cinemassacre-)?(?P<video_id>.+?))"', webpage) | ||||
|         mobj = re.search(r'src="(?P<embed_url>http://player\.screenwavemedia\.com/play/[a-zA-Z]+\.php\?id=(?:Cinemassacre-)?(?P<video_id>.+?))"', webpage) | ||||
|         if not mobj: | ||||
|             raise ExtractorError(u'Can\'t extract embed url and video id') | ||||
|         playerdata_url = mobj.group(u'embed_url') | ||||
| @@ -55,30 +55,32 @@ class CinemassacreIE(InfoExtractor): | ||||
|             video_description = None | ||||
|  | ||||
|         playerdata = self._download_webpage(playerdata_url, video_id) | ||||
|         base_url = self._html_search_regex(r'\'streamer\': \'(?P<base_url>rtmp://.*?)/(?:vod|Cinemassacre)\'', | ||||
|             playerdata, u'base_url') | ||||
|         base_url += '/Cinemassacre/' | ||||
|         # Important: The file names in playerdata are not used by the player and even wrong for some videos | ||||
|         sd_file = 'Cinemassacre-%s_high.mp4' % video_id | ||||
|         hd_file = 'Cinemassacre-%s.mp4' % video_id | ||||
|         video_thumbnail = 'http://image.screenwavemedia.com/Cinemassacre/Cinemassacre-%s_thumb_640x360.jpg' % video_id | ||||
|         url = self._html_search_regex(r'\'streamer\': \'(?P<url>[^\']+)\'', playerdata, u'url') | ||||
|  | ||||
|         sd_file = self._html_search_regex(r'\'file\': \'(?P<sd_file>[^\']+)\'', playerdata, u'sd_file') | ||||
|         hd_file = self._html_search_regex(r'\'?file\'?: "(?P<hd_file>[^"]+)"', playerdata, u'hd_file') | ||||
|         video_thumbnail = self._html_search_regex(r'\'image\': \'(?P<thumbnail>[^\']+)\'', playerdata, u'thumbnail', fatal=False) | ||||
|  | ||||
|         formats = [ | ||||
|             { | ||||
|                 'url': base_url + sd_file, | ||||
|                 'url': url, | ||||
|                 'play_path': 'mp4:' + sd_file, | ||||
|                 'rtmp_live': True, # workaround | ||||
|                 'ext': 'flv', | ||||
|                 'format': 'sd', | ||||
|                 'format_id': 'sd', | ||||
|             }, | ||||
|             { | ||||
|                 'url': base_url + hd_file, | ||||
|                 'url': url, | ||||
|                 'play_path': 'mp4:' + hd_file, | ||||
|                 'rtmp_live': True, # workaround | ||||
|                 'ext': 'flv', | ||||
|                 'format': 'hd', | ||||
|                 'format_id': 'hd', | ||||
|             }, | ||||
|         ] | ||||
|  | ||||
|         info = { | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': video_title, | ||||
|             'formats': formats, | ||||
| @@ -86,6 +88,3 @@ class CinemassacreIE(InfoExtractor): | ||||
|             'upload_date': video_date, | ||||
|             'thumbnail': video_thumbnail, | ||||
|         } | ||||
|         # TODO: Remove when #980 has been merged | ||||
|         info.update(formats[-1]) | ||||
|         return info | ||||
|   | ||||
| @@ -6,7 +6,7 @@ from ..utils import determine_ext | ||||
|  | ||||
|  | ||||
| class CNNIE(InfoExtractor): | ||||
|     _VALID_URL = r'''(?x)https?://(edition\.)?cnn\.com/video/(data/.+?|\?)/ | ||||
|     _VALID_URL = r'''(?x)https?://((edition|www)\.)?cnn\.com/video/(data/.+?|\?)/ | ||||
|         (?P<path>.+?/(?P<title>[^/]+?)(?:\.cnn|(?=&)))''' | ||||
|  | ||||
|     _TESTS = [{ | ||||
|   | ||||
| @@ -14,6 +14,8 @@ from ..utils import ( | ||||
|     clean_html, | ||||
|     compiled_regex_type, | ||||
|     ExtractorError, | ||||
|     RegexNotFoundError, | ||||
|     sanitize_filename, | ||||
|     unescapeHTML, | ||||
| ) | ||||
|  | ||||
| @@ -61,11 +63,21 @@ class InfoExtractor(object): | ||||
|                     * ext       Will be calculated from url if missing | ||||
|                     * format    A human-readable description of the format | ||||
|                                 ("mp4 container with h264/opus"). | ||||
|                                 Calculated from width and height if missing. | ||||
|                                 Calculated from the format_id, width, height. | ||||
|                                 and format_note fields if missing. | ||||
|                     * format_id A short description of the format | ||||
|                                 ("mp4_h264_opus" or "19") | ||||
|                     * format_note Additional info about the format | ||||
|                                 ("3D" or "DASH video") | ||||
|                     * width     Width of the video, if known | ||||
|                     * height    Height of the video, if known | ||||
|                     * abr       Average audio bitrate in KBit/s | ||||
|                     * acodec    Name of the audio codec in use | ||||
|                     * vbr       Average video bitrate in KBit/s | ||||
|                     * vcodec    Name of the video codec in use | ||||
|     webpage_url:    The url to the video webpage, if given to youtube-dl it | ||||
|                     should allow to get the same result again. (It will be set | ||||
|                     by YoutubeDL if it's missing) | ||||
|  | ||||
|     Unless mentioned otherwise, the fields should be Unicode strings. | ||||
|  | ||||
| @@ -178,6 +190,17 @@ class InfoExtractor(object): | ||||
|             self.to_screen(u'Dumping request to ' + url) | ||||
|             dump = base64.b64encode(webpage_bytes).decode('ascii') | ||||
|             self._downloader.to_screen(dump) | ||||
|         if self._downloader.params.get('write_pages', False): | ||||
|             try: | ||||
|                 url = url_or_request.get_full_url() | ||||
|             except AttributeError: | ||||
|                 url = url_or_request | ||||
|             raw_filename = ('%s_%s.dump' % (video_id, url)) | ||||
|             filename = sanitize_filename(raw_filename, restricted=True) | ||||
|             self.to_screen(u'Saving request to ' + filename) | ||||
|             with open(filename, 'wb') as outf: | ||||
|                 outf.write(webpage_bytes) | ||||
|  | ||||
|         content = webpage_bytes.decode(encoding, 'replace') | ||||
|         return (content, urlh) | ||||
|  | ||||
| @@ -228,7 +251,7 @@ class InfoExtractor(object): | ||||
|         Perform a regex search on the given string, using a single or a list of | ||||
|         patterns returning the first matching group. | ||||
|         In case of failure return a default value or raise a WARNING or a | ||||
|         ExtractorError, depending on fatal, specifying the field name. | ||||
|         RegexNotFoundError, depending on fatal, specifying the field name. | ||||
|         """ | ||||
|         if isinstance(pattern, (str, compat_str, compiled_regex_type)): | ||||
|             mobj = re.search(pattern, string, flags) | ||||
| @@ -248,7 +271,7 @@ class InfoExtractor(object): | ||||
|         elif default is not None: | ||||
|             return default | ||||
|         elif fatal: | ||||
|             raise ExtractorError(u'Unable to extract %s' % _name) | ||||
|             raise RegexNotFoundError(u'Unable to extract %s' % _name) | ||||
|         else: | ||||
|             self._downloader.report_warning(u'unable to extract %s; ' | ||||
|                 u'please report this issue on http://yt-dl.org/bug' % _name) | ||||
| @@ -296,13 +319,21 @@ class InfoExtractor(object): | ||||
|  | ||||
|     # Helper functions for extracting OpenGraph info | ||||
|     @staticmethod | ||||
|     def _og_regex(prop): | ||||
|         return r'<meta.+?property=[\'"]og:%s[\'"].+?content=(?:"(.+?)"|\'(.+?)\')' % re.escape(prop) | ||||
|     def _og_regexes(prop): | ||||
|         content_re = r'content=(?:"([^>]+?)"|\'(.+?)\')' | ||||
|         property_re = r'property=[\'"]og:%s[\'"]' % re.escape(prop) | ||||
|         template = r'<meta[^>]+?%s[^>]+?%s' | ||||
|         return [ | ||||
|             template % (property_re, content_re), | ||||
|             template % (content_re, property_re), | ||||
|         ] | ||||
|  | ||||
|     def _og_search_property(self, prop, html, name=None, **kargs): | ||||
|         if name is None: | ||||
|             name = 'OpenGraph %s' % prop | ||||
|         escaped = self._search_regex(self._og_regex(prop), html, name, flags=re.DOTALL, **kargs) | ||||
|         escaped = self._search_regex(self._og_regexes(prop), html, name, flags=re.DOTALL, **kargs) | ||||
|         if escaped is None: | ||||
|             return None | ||||
|         return unescapeHTML(escaped) | ||||
|  | ||||
|     def _og_search_thumbnail(self, html, **kargs): | ||||
| @@ -314,10 +345,10 @@ class InfoExtractor(object): | ||||
|     def _og_search_title(self, html, **kargs): | ||||
|         return self._og_search_property('title', html, **kargs) | ||||
|  | ||||
|     def _og_search_video_url(self, html, name='video url', **kargs): | ||||
|         return self._html_search_regex([self._og_regex('video:secure_url'), | ||||
|                                         self._og_regex('video')], | ||||
|                                        html, name, **kargs) | ||||
|     def _og_search_video_url(self, html, name='video url', secure=True, **kargs): | ||||
|         regexes = self._og_regexes('video') | ||||
|         if secure: regexes = self._og_regexes('video:secure_url') + regexes | ||||
|         return self._html_search_regex(regexes, html, name, **kargs) | ||||
|  | ||||
|     def _rta_search(self, html): | ||||
|         # See http://www.rtalabel.org/index.php?content=howtofaq#single | ||||
|   | ||||
| @@ -21,6 +21,7 @@ class DailymotionBaseInfoExtractor(InfoExtractor): | ||||
|         """Build a request with the family filter disabled""" | ||||
|         request = compat_urllib_request.Request(url) | ||||
|         request.add_header('Cookie', 'family_filter=off') | ||||
|         request.add_header('Cookie', 'ff=off') | ||||
|         return request | ||||
|  | ||||
| class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
| @@ -28,6 +29,15 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|  | ||||
|     _VALID_URL = r'(?i)(?:https?://)?(?:www\.)?dailymotion\.[a-z]{2,3}/(?:embed/)?video/([^/]+)' | ||||
|     IE_NAME = u'dailymotion' | ||||
|  | ||||
|     _FORMATS = [ | ||||
|         (u'stream_h264_ld_url', u'ld'), | ||||
|         (u'stream_h264_url', u'standard'), | ||||
|         (u'stream_h264_hq_url', u'hq'), | ||||
|         (u'stream_h264_hd_url', u'hd'), | ||||
|         (u'stream_h264_hd1080_url', u'hd180'), | ||||
|     ] | ||||
|  | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             u'url': u'http://www.dailymotion.com/video/x33vw9_tutoriel-de-youtubeur-dl-des-video_tech', | ||||
| @@ -52,6 +62,18 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|             }, | ||||
|             u'skip': u'VEVO is only available in some countries', | ||||
|         }, | ||||
|         # age-restricted video | ||||
|         { | ||||
|             u'url': u'http://www.dailymotion.com/video/xyh2zz_leanna-decker-cyber-girl-of-the-year-desires-nude-playboy-plus_redband', | ||||
|             u'file': u'xyh2zz.mp4', | ||||
|             u'md5': u'0d667a7b9cebecc3c89ee93099c4159d', | ||||
|             u'info_dict': { | ||||
|                 u'title': 'Leanna Decker - Cyber Girl Of The Year Desires Nude [Playboy Plus]', | ||||
|                 u'uploader': 'HotWaves1012', | ||||
|                 u'age_limit': 18, | ||||
|             } | ||||
|  | ||||
|         } | ||||
|     ] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
| @@ -60,7 +82,6 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|  | ||||
|         video_id = mobj.group(1).split('_')[0].split('?')[0] | ||||
|  | ||||
|         video_extension = 'mp4' | ||||
|         url = 'http://www.dailymotion.com/video/%s' % video_id | ||||
|  | ||||
|         # Retrieve video webpage to extract further information | ||||
| @@ -82,7 +103,8 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|         video_uploader = self._search_regex([r'(?im)<span class="owner[^\"]+?">[^<]+?<a [^>]+?>([^<]+?)</a>', | ||||
|                                              # Looking for official user | ||||
|                                              r'<(?:span|a) .*?rel="author".*?>([^<]+?)</'], | ||||
|                                             webpage, 'video uploader') | ||||
|                                             webpage, 'video uploader', fatal=False) | ||||
|         age_limit = self._rta_search(webpage) | ||||
|  | ||||
|         video_upload_date = None | ||||
|         mobj = re.search(r'<div class="[^"]*uploaded_cont[^"]*" title="[^"]*">([0-9]{2})-([0-9]{2})-([0-9]{4})</div>', webpage) | ||||
| @@ -99,37 +121,43 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|             msg = 'Couldn\'t get video, Dailymotion says: %s' % info['error']['title'] | ||||
|             raise ExtractorError(msg, expected=True) | ||||
|  | ||||
|         # TODO: support choosing qualities | ||||
|  | ||||
|         for key in ['stream_h264_hd1080_url','stream_h264_hd_url', | ||||
|                     'stream_h264_hq_url','stream_h264_url', | ||||
|                     'stream_h264_ld_url']: | ||||
|             if info.get(key):#key in info and info[key]: | ||||
|                 max_quality = key | ||||
|                 self.to_screen(u'Using %s' % key) | ||||
|                 break | ||||
|         else: | ||||
|         formats = [] | ||||
|         for (key, format_id) in self._FORMATS: | ||||
|             video_url = info.get(key) | ||||
|             if video_url is not None: | ||||
|                 m_size = re.search(r'H264-(\d+)x(\d+)', video_url) | ||||
|                 if m_size is not None: | ||||
|                     width, height = m_size.group(1), m_size.group(2) | ||||
|                 else: | ||||
|                     width, height = None, None | ||||
|                 formats.append({ | ||||
|                     'url': video_url, | ||||
|                     'ext': 'mp4', | ||||
|                     'format_id': format_id, | ||||
|                     'width': width, | ||||
|                     'height': height, | ||||
|                 }) | ||||
|         if not formats: | ||||
|             raise ExtractorError(u'Unable to extract video URL') | ||||
|         video_url = info[max_quality] | ||||
|  | ||||
|         # subtitles | ||||
|         video_subtitles = self.extract_subtitles(video_id) | ||||
|         video_subtitles = self.extract_subtitles(video_id, webpage) | ||||
|         if self._downloader.params.get('listsubtitles', False): | ||||
|             self._list_available_subtitles(video_id) | ||||
|             self._list_available_subtitles(video_id, webpage) | ||||
|             return | ||||
|  | ||||
|         return [{ | ||||
|         return { | ||||
|             'id':       video_id, | ||||
|             'url':      video_url, | ||||
|             'formats': formats, | ||||
|             'uploader': video_uploader, | ||||
|             'upload_date':  video_upload_date, | ||||
|             'title':    self._og_search_title(webpage), | ||||
|             'ext':      video_extension, | ||||
|             'subtitles':    video_subtitles, | ||||
|             'thumbnail': info['thumbnail_url'] | ||||
|         }] | ||||
|             'thumbnail': info['thumbnail_url'], | ||||
|             'age_limit': age_limit, | ||||
|         } | ||||
|  | ||||
|     def _get_available_subtitles(self, video_id): | ||||
|     def _get_available_subtitles(self, video_id, webpage): | ||||
|         try: | ||||
|             sub_list = self._download_webpage( | ||||
|                 'https://api.dailymotion.com/video/%s/subtitles?fields=id,language,url' % video_id, | ||||
| @@ -158,7 +186,7 @@ class DailymotionPlaylistIE(DailymotionBaseInfoExtractor): | ||||
|             webpage = self._download_webpage(request, | ||||
|                                              id, u'Downloading page %s' % pagenum) | ||||
|  | ||||
|             playlist_el = get_element_by_attribute(u'class', u'video_list', webpage) | ||||
|             playlist_el = get_element_by_attribute(u'class', u'row video_list', webpage) | ||||
|             video_ids.extend(re.findall(r'data-id="(.+?)"', playlist_el)) | ||||
|  | ||||
|             if re.search(self._MORE_PAGES_INDICATOR, webpage, re.DOTALL) is None: | ||||
|   | ||||
| @@ -25,7 +25,7 @@ class DepositFilesIE(InfoExtractor): | ||||
|         url = 'http://depositfiles.com/en/files/' + file_id | ||||
|  | ||||
|         # Retrieve file webpage with 'Free download' button pressed | ||||
|         free_download_indication = { 'gateway_result' : '1' } | ||||
|         free_download_indication = {'gateway_result' : '1'} | ||||
|         request = compat_urllib_request.Request(url, compat_urllib_parse.urlencode(free_download_indication)) | ||||
|         try: | ||||
|             self.report_download_webpage(file_id) | ||||
|   | ||||
| @@ -101,7 +101,7 @@ class EightTracksIE(InfoExtractor): | ||||
|         first_url = 'http://8tracks.com/sets/%s/play?player=sm&mix_id=%s&format=jsonh' % (session, mix_id) | ||||
|         next_url = first_url | ||||
|         res = [] | ||||
|         for i in itertools.count(): | ||||
|         for i in range(track_count): | ||||
|             api_json = self._download_webpage(next_url, playlist_id, | ||||
|                 note=u'Downloading song information %s/%s' % (str(i+1), track_count), | ||||
|                 errnote=u'Failed to download song information') | ||||
| @@ -116,7 +116,5 @@ class EightTracksIE(InfoExtractor): | ||||
|                 'ext': 'm4a', | ||||
|             } | ||||
|             res.append(info) | ||||
|             if api_data['set']['at_last_track']: | ||||
|                 break | ||||
|             next_url = 'http://8tracks.com/sets/%s/next?player=sm&mix_id=%s&format=jsonh&track_id=%s' % (session, mix_id, track_data['id']) | ||||
|         return res | ||||
|   | ||||
							
								
								
									
										37
									
								
								youtube_dl/extractor/eitb.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								youtube_dl/extractor/eitb.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| # encoding: utf-8 | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from .brightcove import BrightcoveIE | ||||
| from ..utils import ExtractorError | ||||
|  | ||||
|  | ||||
| class EitbIE(InfoExtractor): | ||||
|     IE_NAME = u'eitb.tv' | ||||
|     _VALID_URL = r'https?://www\.eitb\.tv/(eu/bideoa|es/video)/[^/]+/(?P<playlist_id>\d+)/(?P<chapter_id>\d+)' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'add_ie': ['Brightcove'], | ||||
|         u'url': u'http://www.eitb.tv/es/video/60-minutos-60-minutos-2013-2014/2677100210001/2743577154001/lasa-y-zabala-30-anos/', | ||||
|         u'md5': u'edf4436247185adee3ea18ce64c47998', | ||||
|         u'info_dict': { | ||||
|             u'id': u'2743577154001', | ||||
|             u'ext': u'mp4', | ||||
|             u'title': u'60 minutos (Lasa y Zabala, 30 años)', | ||||
|             # All videos from eitb has this description in the brightcove info | ||||
|             u'description': u'.', | ||||
|             u'uploader': u'Euskal Telebista', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         chapter_id = mobj.group('chapter_id') | ||||
|         webpage = self._download_webpage(url, chapter_id) | ||||
|         bc_url = BrightcoveIE._extract_brightcove_url(webpage) | ||||
|         if bc_url is None: | ||||
|             raise ExtractorError(u'Could not extract the Brightcove url') | ||||
|         # The BrightcoveExperience object doesn't contain the video id, we set | ||||
|         # it manually | ||||
|         bc_url += '&%40videoPlayer={0}'.format(chapter_id) | ||||
|         return self.url_result(bc_url, BrightcoveIE.ie_key()) | ||||
| @@ -11,16 +11,17 @@ class ExfmIE(InfoExtractor): | ||||
|     _SOUNDCLOUD_URL = r'(?:http://)?(?:www\.)?api\.soundcloud.com/tracks/([^/]+)/stream' | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             u'url': u'http://ex.fm/song/1bgtzg', | ||||
|             u'file': u'95223130.mp3', | ||||
|             u'md5': u'8a7967a3fef10e59a1d6f86240fd41cf', | ||||
|             u'url': u'http://ex.fm/song/eh359', | ||||
|             u'file': u'44216187.mp3', | ||||
|             u'md5': u'e45513df5631e6d760970b14cc0c11e7', | ||||
|             u'info_dict': { | ||||
|                 u"title": u"We Can't Stop - Miley Cyrus", | ||||
|                 u"uploader": u"Miley Cyrus", | ||||
|                 u'upload_date': u'20130603', | ||||
|                 u'description': u'Download "We Can\'t Stop" \r\niTunes: http://smarturl.it/WeCantStop?IQid=SC\r\nAmazon: http://smarturl.it/WeCantStopAMZ?IQid=SC', | ||||
|                 u"title": u"Test House \"Love Is Not Enough\" (Extended Mix) DeadJournalist Exclusive", | ||||
|                 u"uploader": u"deadjournalist", | ||||
|                 u'upload_date': u'20120424', | ||||
|                 u'description': u'Test House \"Love Is Not Enough\" (Extended Mix) DeadJournalist Exclusive', | ||||
|             }, | ||||
|             u'note': u'Soundcloud song', | ||||
|             u'skip': u'The site is down too often', | ||||
|         }, | ||||
|         { | ||||
|             u'url': u'http://ex.fm/song/wddt8', | ||||
| @@ -30,6 +31,7 @@ class ExfmIE(InfoExtractor): | ||||
|                 u'title': u'Safe and Sound', | ||||
|                 u'uploader': u'Capital Cities', | ||||
|             }, | ||||
|             u'skip': u'The site is down too often', | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|   | ||||
							
								
								
									
										50
									
								
								youtube_dl/extractor/extremetube.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								youtube_dl/extractor/extremetube.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | ||||
| import os | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urllib_parse_urlparse, | ||||
|     compat_urllib_request, | ||||
|     compat_urllib_parse, | ||||
| ) | ||||
|  | ||||
| class ExtremeTubeIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?(?P<url>extremetube\.com/video/.+?(?P<videoid>[0-9]+))(?:[/?&]|$)' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.extremetube.com/video/music-video-14-british-euro-brit-european-cumshots-swallow-652431', | ||||
|         u'file': u'652431.mp4', | ||||
|         u'md5': u'1fb9228f5e3332ec8c057d6ac36f33e0', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Music Video 14 british euro brit european cumshots swallow", | ||||
|             u"uploader": u"unknown", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('videoid') | ||||
|         url = 'http://www.' + mobj.group('url') | ||||
|  | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('Cookie', 'age_verified=1') | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         video_title = self._html_search_regex(r'<h1 [^>]*?title="([^"]+)"[^>]*>\1<', webpage, u'title') | ||||
|         uploader = self._html_search_regex(r'>Posted by:(?=<)(?:\s|<[^>]*>)*(.+?)\|', webpage, u'uploader', fatal=False) | ||||
|         video_url = compat_urllib_parse.unquote(self._html_search_regex(r'video_url=(.+?)&', webpage, u'video_url')) | ||||
|         path = compat_urllib_parse_urlparse(video_url).path | ||||
|         extension = os.path.splitext(path)[1][1:] | ||||
|         format = path.split('/')[5].split('_')[:2] | ||||
|         format = "-".join(format) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': video_title, | ||||
|             'uploader': uploader, | ||||
|             'url': video_url, | ||||
|             'ext': extension, | ||||
|             'format': format, | ||||
|             'format_id': format, | ||||
|             'age_limit': 18, | ||||
|         } | ||||
| @@ -19,7 +19,8 @@ class FacebookIE(InfoExtractor): | ||||
|     """Information Extractor for Facebook""" | ||||
|  | ||||
|     _VALID_URL = r'^(?:https?://)?(?:\w+\.)?facebook\.com/(?:video/video|photo)\.php\?(?:.*?)v=(?P<ID>\d+)(?:.*)' | ||||
|     _LOGIN_URL = 'https://login.facebook.com/login.php?m&next=http%3A%2F%2Fm.facebook.com%2Fhome.php&' | ||||
|     _LOGIN_URL = 'https://www.facebook.com/login.php?next=http%3A%2F%2Ffacebook.com%2Fhome.php&login_attempt=1' | ||||
|     _CHECKPOINT_URL = 'https://www.facebook.com/checkpoint/?next=http%3A%2F%2Ffacebook.com%2Fhome.php&_fb_noscript=1' | ||||
|     _NETRC_MACHINE = 'facebook' | ||||
|     IE_NAME = u'facebook' | ||||
|     _TEST = { | ||||
| @@ -36,50 +37,56 @@ class FacebookIE(InfoExtractor): | ||||
|         """Report attempt to log in.""" | ||||
|         self.to_screen(u'Logging in') | ||||
|  | ||||
|     def _real_initialize(self): | ||||
|         if self._downloader is None: | ||||
|             return | ||||
|  | ||||
|         useremail = None | ||||
|         password = None | ||||
|         downloader_params = self._downloader.params | ||||
|  | ||||
|         # Attempt to use provided username and password or .netrc data | ||||
|         if downloader_params.get('username', None) is not None: | ||||
|             useremail = downloader_params['username'] | ||||
|             password = downloader_params['password'] | ||||
|         elif downloader_params.get('usenetrc', False): | ||||
|             try: | ||||
|                 info = netrc.netrc().authenticators(self._NETRC_MACHINE) | ||||
|                 if info is not None: | ||||
|                     useremail = info[0] | ||||
|                     password = info[2] | ||||
|                 else: | ||||
|                     raise netrc.NetrcParseError('No authenticators for %s' % self._NETRC_MACHINE) | ||||
|             except (IOError, netrc.NetrcParseError) as err: | ||||
|                 self._downloader.report_warning(u'parsing .netrc: %s' % compat_str(err)) | ||||
|                 return | ||||
|  | ||||
|     def _login(self): | ||||
|         (useremail, password) = self._get_login_info() | ||||
|         if useremail is None: | ||||
|             return | ||||
|  | ||||
|         # Log in | ||||
|         login_page_req = compat_urllib_request.Request(self._LOGIN_URL) | ||||
|         login_page_req.add_header('Cookie', 'locale=en_US') | ||||
|         self.report_login() | ||||
|         login_page = self._download_webpage(login_page_req, None, note=False, | ||||
|             errnote=u'Unable to download login page') | ||||
|         lsd = self._search_regex(r'"lsd":"(\w*?)"', login_page, u'lsd') | ||||
|         lgnrnd = self._search_regex(r'name="lgnrnd" value="([^"]*?)"', login_page, u'lgnrnd') | ||||
|  | ||||
|         login_form = { | ||||
|             'email': useremail, | ||||
|             'pass': password, | ||||
|             'login': 'Log+In' | ||||
|             'lsd': lsd, | ||||
|             'lgnrnd': lgnrnd, | ||||
|             'next': 'http://facebook.com/home.php', | ||||
|             'default_persistent': '0', | ||||
|             'legacy_return': '1', | ||||
|             'timezone': '-60', | ||||
|             'trynum': '1', | ||||
|             } | ||||
|         request = compat_urllib_request.Request(self._LOGIN_URL, compat_urllib_parse.urlencode(login_form)) | ||||
|         request.add_header('Content-Type', 'application/x-www-form-urlencoded') | ||||
|         try: | ||||
|             self.report_login() | ||||
|             login_results = compat_urllib_request.urlopen(request).read() | ||||
|             if re.search(r'<form(.*)name="login"(.*)</form>', login_results) is not None: | ||||
|                 self._downloader.report_warning(u'unable to log in: bad username/password, or exceded login rate limit (~3/min). Check credentials or wait.') | ||||
|                 return | ||||
|  | ||||
|             check_form = { | ||||
|                 'fb_dtsg': self._search_regex(r'"fb_dtsg":"(.*?)"', login_results, u'fb_dtsg'), | ||||
|                 'nh': self._search_regex(r'name="nh" value="(\w*?)"', login_results, u'nh'), | ||||
|                 'name_action_selected': 'dont_save', | ||||
|                 'submit[Continue]': self._search_regex(r'<input value="(.*?)" name="submit\[Continue\]"', login_results, u'continue'), | ||||
|             } | ||||
|             check_req = compat_urllib_request.Request(self._CHECKPOINT_URL, compat_urllib_parse.urlencode(check_form)) | ||||
|             check_req.add_header('Content-Type', 'application/x-www-form-urlencoded') | ||||
|             check_response = compat_urllib_request.urlopen(check_req).read() | ||||
|             if re.search(r'id="checkpointSubmitButton"', check_response) is not None: | ||||
|                 self._downloader.report_warning(u'Unable to confirm login, you have to login in your brower and authorize the login.') | ||||
|         except (compat_urllib_error.URLError, compat_http_client.HTTPException, socket.error) as err: | ||||
|             self._downloader.report_warning(u'unable to log in: %s' % compat_str(err)) | ||||
|             return | ||||
|  | ||||
|     def _real_initialize(self): | ||||
|         self._login() | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         if mobj is None: | ||||
| @@ -93,7 +100,13 @@ class FacebookIE(InfoExtractor): | ||||
|         AFTER = '.forEach(function(variable) {swf.addVariable(variable[0], variable[1]);});' | ||||
|         m = re.search(re.escape(BEFORE) + '(.*?)' + re.escape(AFTER), webpage) | ||||
|         if not m: | ||||
|             raise ExtractorError(u'Cannot parse data') | ||||
|             m_msg = re.search(r'class="[^"]*uiInterstitialContent[^"]*"><div>(.*?)</div>', webpage) | ||||
|             if m_msg is not None: | ||||
|                 raise ExtractorError( | ||||
|                     u'The video is not available, Facebook said: "%s"' % m_msg.group(1), | ||||
|                     expected=True) | ||||
|             else: | ||||
|                 raise ExtractorError(u'Cannot parse data') | ||||
|         data = dict(json.loads(m.group(1))) | ||||
|         params_raw = compat_urllib_parse.unquote(data['params']) | ||||
|         params = json.loads(params_raw) | ||||
|   | ||||
| @@ -5,8 +5,6 @@ import xml.etree.ElementTree | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     determine_ext, | ||||
|     clean_html, | ||||
|     get_element_by_attribute, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -47,12 +45,12 @@ class FazIE(InfoExtractor): | ||||
|                 'format_id': code.lower(), | ||||
|             }) | ||||
|  | ||||
|         descr_html = get_element_by_attribute('class', 'Content Copy', webpage) | ||||
|         descr = self._html_search_regex(r'<p class="Content Copy">(.*?)</p>', webpage, u'description') | ||||
|         info = { | ||||
|             'id': video_id, | ||||
|             'title': self._og_search_title(webpage), | ||||
|             'formats': formats, | ||||
|             'description': clean_html(descr_html), | ||||
|             'description': descr, | ||||
|             'thumbnail': config.find('STILL/STILL_BIG').text, | ||||
|         } | ||||
|         # TODO: Remove when #980 has been merged | ||||
|   | ||||
							
								
								
									
										38
									
								
								youtube_dl/extractor/gamekings.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								youtube_dl/extractor/gamekings.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class GamekingsIE(InfoExtractor): | ||||
|     _VALID_URL = r'http?://www\.gamekings\.tv/videos/(?P<name>[0-9a-z\-]+)' | ||||
|     _TEST = { | ||||
|         u"url": u"http://www.gamekings.tv/videos/phoenix-wright-ace-attorney-dual-destinies-review/", | ||||
|         u'file': u'20130811.mp4', | ||||
|         # MD5 is flaky, seems to change regularly | ||||
|         #u'md5': u'2f32b1f7b80fdc5cb616efb4f387f8a3', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Phoenix Wright: Ace Attorney \u2013 Dual Destinies Review", | ||||
|             u"description": u"Melle en Steven hebben voor de review een week in de rechtbank doorbracht met Phoenix Wright: Ace Attorney - Dual Destinies.", | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|  | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         name = mobj.group('name') | ||||
|         webpage = self._download_webpage(url, name) | ||||
|         video_url = self._og_search_video_url(webpage) | ||||
|  | ||||
|         video = re.search(r'[0-9]+', video_url) | ||||
|         video_id = video.group(0) | ||||
|  | ||||
|         # Todo: add medium format | ||||
|         video_url = video_url.replace(video_id, 'large/' + video_id) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'ext': 'mp4', | ||||
|             'url': video_url, | ||||
|             'title': self._og_search_title(webpage), | ||||
|             'description': self._og_search_description(webpage), | ||||
|         } | ||||
| @@ -25,7 +25,7 @@ class GenericIE(InfoExtractor): | ||||
|         { | ||||
|             u'url': u'http://www.hodiho.fr/2013/02/regis-plante-sa-jeep.html', | ||||
|             u'file': u'13601338388002.mp4', | ||||
|             u'md5': u'85b90ccc9d73b4acd9138d3af4c27f89', | ||||
|             u'md5': u'6e15c93721d7ec9e9ca3fdbf07982cfd', | ||||
|             u'info_dict': { | ||||
|                 u"uploader": u"www.hodiho.fr", | ||||
|                 u"title": u"R\u00e9gis plante sa Jeep" | ||||
| @@ -33,6 +33,7 @@ class GenericIE(InfoExtractor): | ||||
|         }, | ||||
|         # embedded vimeo video | ||||
|         { | ||||
|             u'add_ie': ['Vimeo'], | ||||
|             u'url': u'http://skillsmatter.com/podcast/home/move-semanticsperfect-forwarding-and-rvalue-references', | ||||
|             u'file': u'22444065.mp4', | ||||
|             u'md5': u'2903896e23df39722c33f015af0666e2', | ||||
| @@ -41,7 +42,35 @@ class GenericIE(InfoExtractor): | ||||
|                 u"uploader_id": u"skillsmatter", | ||||
|                 u"uploader": u"Skills Matter", | ||||
|             } | ||||
|         } | ||||
|         }, | ||||
|         # bandcamp page with custom domain | ||||
|         { | ||||
|             u'add_ie': ['Bandcamp'], | ||||
|             u'url': u'http://bronyrock.com/track/the-pony-mash', | ||||
|             u'file': u'3235767654.mp3', | ||||
|             u'info_dict': { | ||||
|                 u'title': u'The Pony Mash', | ||||
|                 u'uploader': u'M_Pallante', | ||||
|             }, | ||||
|             u'skip': u'There is a limit of 200 free downloads / month for the test song', | ||||
|         }, | ||||
|         # embedded brightcove video | ||||
|         # it also tests brightcove videos that need to set the 'Referer' in the | ||||
|         # http requests | ||||
|         { | ||||
|             u'add_ie': ['Brightcove'], | ||||
|             u'url': u'http://www.bfmtv.com/video/bfmbusiness/cours-bourse/cours-bourse-l-analyse-technique-154522/', | ||||
|             u'info_dict': { | ||||
|                 u'id': u'2765128793001', | ||||
|                 u'ext': u'mp4', | ||||
|                 u'title': u'Le cours de bourse : l’analyse technique', | ||||
|                 u'description': u'md5:7e9ad046e968cb2d1114004aba466fd9', | ||||
|                 u'uploader': u'BFM BUSINESS', | ||||
|             }, | ||||
|             u'params': { | ||||
|                 u'skip_download': True, | ||||
|             }, | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     def report_download_webpage(self, video_id): | ||||
| @@ -133,11 +162,20 @@ class GenericIE(InfoExtractor): | ||||
|             raise ExtractorError(u'Failed to download URL: %s' % url) | ||||
|  | ||||
|         self.report_extraction(video_id) | ||||
|  | ||||
|         # it's tempting to parse this further, but you would | ||||
|         # have to take into account all the variations like | ||||
|         #   Video Title - Site Name | ||||
|         #   Site Name | Video Title | ||||
|         #   Video Title - Tagline | Site Name | ||||
|         # and so on and so forth; it's just not practical | ||||
|         video_title = self._html_search_regex(r'<title>(.*)</title>', | ||||
|             webpage, u'video title', default=u'video', flags=re.DOTALL) | ||||
|  | ||||
|         # Look for BrightCove: | ||||
|         m_brightcove = re.search(r'<object[^>]+?class=([\'"])[^>]*?BrightcoveExperience.*?\1.+?</object>', webpage, re.DOTALL) | ||||
|         if m_brightcove is not None: | ||||
|         bc_url = BrightcoveIE._extract_brightcove_url(webpage) | ||||
|         if bc_url is not None: | ||||
|             self.to_screen(u'Brightcove video detected.') | ||||
|             bc_url = BrightcoveIE._build_brighcove_url(m_brightcove.group()) | ||||
|             return self.url_result(bc_url, 'Brightcove') | ||||
|  | ||||
|         # Look for embedded Vimeo player | ||||
| @@ -149,11 +187,19 @@ class GenericIE(InfoExtractor): | ||||
|             return self.url_result(surl, 'Vimeo') | ||||
|  | ||||
|         # Look for embedded YouTube player | ||||
|         mobj = re.search( | ||||
|             r'<iframe[^>]+?src="(https?://(?:www\.)?youtube.com/embed/.+?)"', webpage) | ||||
|         if mobj: | ||||
|             surl = unescapeHTML(mobj.group(1)) | ||||
|             return self.url_result(surl, 'Youtube') | ||||
|         matches = re.findall( | ||||
|             r'<iframe[^>]+?src=(["\'])(?P<url>(?:https?:)?//(?:www\.)?youtube.com/embed/.+?)\1', webpage) | ||||
|         if matches: | ||||
|             urlrs = [self.url_result(unescapeHTML(tuppl[1]), 'Youtube') | ||||
|                      for tuppl in matches] | ||||
|             return self.playlist_result( | ||||
|                 urlrs, playlist_id=video_id, playlist_title=video_title) | ||||
|  | ||||
|         # Look for Bandcamp pages with custom domain | ||||
|         mobj = re.search(r'<meta property="og:url"[^>]*?content="(.*?bandcamp\.com.*?)"', webpage) | ||||
|         if mobj is not None: | ||||
|             burl = unescapeHTML(mobj.group(1)) | ||||
|             return self.url_result(burl, 'Bandcamp') | ||||
|  | ||||
|         # Start with something easy: JW Player in SWFObject | ||||
|         mobj = re.search(r'flashvars: [\'"](?:.*&)?file=(http[^\'"&]*)', webpage) | ||||
| @@ -192,15 +238,6 @@ class GenericIE(InfoExtractor): | ||||
|         video_extension = os.path.splitext(video_id)[1][1:] | ||||
|         video_id = os.path.splitext(video_id)[0] | ||||
|  | ||||
|         # it's tempting to parse this further, but you would | ||||
|         # have to take into account all the variations like | ||||
|         #   Video Title - Site Name | ||||
|         #   Site Name | Video Title | ||||
|         #   Video Title - Tagline | Site Name | ||||
|         # and so on and so forth; it's just not practical | ||||
|         video_title = self._html_search_regex(r'<title>(.*)</title>', | ||||
|             webpage, u'video title', default=u'video', flags=re.DOTALL) | ||||
|  | ||||
|         # video uploader is domain name | ||||
|         video_uploader = self._search_regex(r'(?:https?://)?([^/]*)/.*', | ||||
|             url, u'video uploader') | ||||
|   | ||||
| @@ -41,9 +41,9 @@ class GooglePlusIE(InfoExtractor): | ||||
|  | ||||
|         # Extract update date | ||||
|         upload_date = self._html_search_regex( | ||||
|             r'''(?x)<a.+?class="o-T-s\s[^"]+"\s+style="display:\s*none"\s*> | ||||
|             r'''(?x)<a.+?class="o-U-s\s[^"]+"\s+style="display:\s*none"\s*> | ||||
|                     ([0-9]{4}-[0-9]{2}-[0-9]{2})</a>''', | ||||
|             webpage, u'upload date', fatal=False) | ||||
|             webpage, u'upload date', fatal=False, flags=re.VERBOSE) | ||||
|         if upload_date: | ||||
|             # Convert timestring to a format suitable for filename | ||||
|             upload_date = datetime.datetime.strptime(upload_date, "%Y-%m-%d") | ||||
|   | ||||
| @@ -30,7 +30,7 @@ class HypemIE(InfoExtractor): | ||||
|             raise ExtractorError(u'Invalid URL: %s' % url) | ||||
|         track_id = mobj.group(1) | ||||
|  | ||||
|         data = { 'ax': 1, 'ts': time.time() } | ||||
|         data = {'ax': 1, 'ts': time.time()} | ||||
|         data_encoded = compat_urllib_parse.urlencode(data) | ||||
|         complete_url = url + "?" + data_encoded | ||||
|         request = compat_urllib_request.Request(complete_url) | ||||
| @@ -68,4 +68,4 @@ class HypemIE(InfoExtractor): | ||||
|             'ext':      "mp3", | ||||
|             'title':    title, | ||||
|             'artist':   artist, | ||||
|         }] | ||||
|         }] | ||||
|   | ||||
| @@ -26,7 +26,7 @@ class InstagramIE(InfoExtractor): | ||||
|  | ||||
|         return [{ | ||||
|             'id':        video_id, | ||||
|             'url':       self._og_search_video_url(webpage), | ||||
|             'url':       self._og_search_video_url(webpage, secure=False), | ||||
|             'ext':       'mp4', | ||||
|             'title':     u'Video by %s' % uploader_id, | ||||
|             'thumbnail': self._og_search_thumbnail(webpage), | ||||
|   | ||||
| @@ -19,7 +19,7 @@ class InternetVideoArchiveIE(InfoExtractor): | ||||
|         u'info_dict': { | ||||
|             u'title': u'SKYFALL', | ||||
|             u'description': u'In SKYFALL, Bond\'s loyalty to M is tested as her past comes back to haunt her. As MI6 comes under attack, 007 must track down and destroy the threat, no matter how personal the cost.', | ||||
|             u'duration': 156, | ||||
|             u'duration': 153, | ||||
|         }, | ||||
|     } | ||||
|  | ||||
| @@ -74,7 +74,7 @@ class InternetVideoArchiveIE(InfoExtractor): | ||||
|             }) | ||||
|         formats = sorted(formats, key=lambda f: f['bitrate']) | ||||
|  | ||||
|         info = { | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': item.find('title').text, | ||||
|             'formats': formats, | ||||
| @@ -82,6 +82,3 @@ class InternetVideoArchiveIE(InfoExtractor): | ||||
|             'description': item.find('description').text, | ||||
|             'duration': int(attr['duration']), | ||||
|         } | ||||
|         # TODO: Remove when #980 has been merged | ||||
|         info.update(formats[-1]) | ||||
|         return info | ||||
|   | ||||
| @@ -1,8 +1,10 @@ | ||||
| import re | ||||
| import hashlib | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import determine_ext | ||||
|  | ||||
| _md5 = lambda s: hashlib.md5(s.encode('utf-8')).hexdigest() | ||||
|  | ||||
| class KankanIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:.*?\.)?kankan\.com/.+?/(?P<id>\d+)\.shtml' | ||||
| @@ -30,7 +32,10 @@ class KankanIE(InfoExtractor): | ||||
|                                                  video_id, u'Downloading video url info') | ||||
|         ip = self._search_regex(r'ip:"(.+?)"', video_info_page, u'video url ip') | ||||
|         path = self._search_regex(r'path:"(.+?)"', video_info_page, u'video url path') | ||||
|         video_url = 'http://%s%s' % (ip, path) | ||||
|         param1 = self._search_regex(r'param1:(\d+)', video_info_page, u'param1') | ||||
|         param2 = self._search_regex(r'param2:(\d+)', video_info_page, u'param2') | ||||
|         key = _md5('xl_mp43651' + param1 + param2) | ||||
|         video_url = 'http://%s%s?key=%s&key1=%s' % (ip, path, key, param2) | ||||
|  | ||||
|         return {'id': video_id, | ||||
|                 'title': title, | ||||
|   | ||||
							
								
								
									
										61
									
								
								youtube_dl/extractor/keezmovies.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								youtube_dl/extractor/keezmovies.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| import os | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urllib_parse_urlparse, | ||||
|     compat_urllib_request, | ||||
|     compat_urllib_parse, | ||||
| ) | ||||
| from ..aes import ( | ||||
|     aes_decrypt_text | ||||
| ) | ||||
|  | ||||
| class KeezMoviesIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?(?P<url>keezmovies\.com/video/.+?(?P<videoid>[0-9]+))(?:[/?&]|$)' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.keezmovies.com/video/petite-asian-lady-mai-playing-in-bathtub-1214711', | ||||
|         u'file': u'1214711.mp4', | ||||
|         u'md5': u'6e297b7e789329923fcf83abb67c9289', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Petite Asian Lady Mai Playing In Bathtub", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('videoid') | ||||
|         url = 'http://www.' + mobj.group('url') | ||||
|  | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('Cookie', 'age_verified=1') | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         # embedded video | ||||
|         mobj = re.search(r'href="([^"]+)"></iframe>', webpage) | ||||
|         if mobj: | ||||
|             embedded_url = mobj.group(1) | ||||
|             return self.url_result(embedded_url) | ||||
|  | ||||
|         video_title = self._html_search_regex(r'<h1 [^>]*>([^<]+)', webpage, u'title') | ||||
|         video_url = compat_urllib_parse.unquote(self._html_search_regex(r'video_url=(.+?)&', webpage, u'video_url')) | ||||
|         if webpage.find('encrypted=true')!=-1: | ||||
|             password = self._html_search_regex(r'video_title=(.+?)&', webpage, u'password') | ||||
|             video_url = aes_decrypt_text(video_url, password, 32).decode('utf-8') | ||||
|         path = compat_urllib_parse_urlparse(video_url).path | ||||
|         extension = os.path.splitext(path)[1][1:] | ||||
|         format = path.split('/')[4].split('_')[:2] | ||||
|         format = "-".join(format) | ||||
|  | ||||
|         age_limit = self._rta_search(webpage) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': video_title, | ||||
|             'url': video_url, | ||||
|             'ext': extension, | ||||
|             'format': format, | ||||
|             'format_id': format, | ||||
|             'age_limit': age_limit, | ||||
|         } | ||||
| @@ -1,16 +1,19 @@ | ||||
| import re | ||||
| import json | ||||
| import xml.etree.ElementTree | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urllib_parse_urlparse, | ||||
|     compat_urlparse, | ||||
|     get_meta_content, | ||||
|     xpath_with_ns, | ||||
|     ExtractorError, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class LivestreamIE(InfoExtractor): | ||||
|     IE_NAME = u'livestream' | ||||
|     _VALID_URL = r'http://new.livestream.com/.*?/(?P<event_name>.*?)(/videos/(?P<id>\d+))?/?$' | ||||
|     _TEST = { | ||||
|         u'url': u'http://new.livestream.com/CoheedandCambria/WebsterHall/videos/4719370', | ||||
| @@ -40,13 +43,9 @@ class LivestreamIE(InfoExtractor): | ||||
|  | ||||
|         if video_id is None: | ||||
|             # This is an event page: | ||||
|             player = get_meta_content('twitter:player', webpage) | ||||
|             if player is None: | ||||
|                 raise ExtractorError('Couldn\'t extract event api url') | ||||
|             api_url = player.replace('/player', '') | ||||
|             api_url = re.sub(r'^(https?://)(new\.)', r'\1api.\2', api_url) | ||||
|             info = json.loads(self._download_webpage(api_url, event_name, | ||||
|                                                      u'Downloading event info')) | ||||
|             config_json = self._search_regex(r'window.config = ({.*?});', | ||||
|                 webpage, u'window config') | ||||
|             info = json.loads(config_json)['event'] | ||||
|             videos = [self._extract_video_info(video_data['data']) | ||||
|                 for video_data in info['feed']['data'] if video_data['type'] == u'video'] | ||||
|             return self.playlist_result(videos, info['id'], info['full_name']) | ||||
| @@ -58,3 +57,44 @@ class LivestreamIE(InfoExtractor): | ||||
|             info = json.loads(self._download_webpage(api_url, video_id, | ||||
|                                                      u'Downloading video info')) | ||||
|             return self._extract_video_info(info) | ||||
|  | ||||
|  | ||||
| # The original version of Livestream uses a different system | ||||
| class LivestreamOriginalIE(InfoExtractor): | ||||
|     IE_NAME = u'livestream:original' | ||||
|     _VALID_URL = r'https?://www\.livestream\.com/(?P<user>[^/]+)/video\?.*?clipId=(?P<id>.*?)(&|$)' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.livestream.com/dealbook/video?clipId=pla_8aa4a3f1-ba15-46a4-893b-902210e138fb', | ||||
|         u'info_dict': { | ||||
|             u'id': u'pla_8aa4a3f1-ba15-46a4-893b-902210e138fb', | ||||
|             u'ext': u'flv', | ||||
|             u'title': u'Spark 1 (BitCoin) with Cameron Winklevoss & Tyler Winklevoss of Winklevoss Capital', | ||||
|         }, | ||||
|         u'params': { | ||||
|             # rtmp | ||||
|             u'skip_download': True, | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         user = mobj.group('user') | ||||
|         api_url = 'http://x{0}x.api.channel.livestream.com/2.0/clipdetails?extendedInfo=true&id={1}'.format(user, video_id) | ||||
|  | ||||
|         api_response = self._download_webpage(api_url, video_id) | ||||
|         info = xml.etree.ElementTree.fromstring(api_response.encode('utf-8')) | ||||
|         item = info.find('channel').find('item') | ||||
|         ns = {'media': 'http://search.yahoo.com/mrss'} | ||||
|         thumbnail_url = item.find(xpath_with_ns('media:thumbnail', ns)).attrib['url'] | ||||
|         # Remove the extension and number from the path (like 1.jpg) | ||||
|         path = self._search_regex(r'(user-files/.+)_.*?\.jpg$', thumbnail_url, u'path') | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': item.find('title').text, | ||||
|             'url': 'rtmp://extondemand.livestream.com/ondemand', | ||||
|             'play_path': 'mp4:trans/dv15/mogulus-{0}.mp4'.format(path), | ||||
|             'ext': 'flv', | ||||
|             'thumbnail': thumbnail_url, | ||||
|         } | ||||
|   | ||||
| @@ -20,10 +20,12 @@ class MetacafeIE(InfoExtractor): | ||||
|     _DISCLAIMER = 'http://www.metacafe.com/family_filter/' | ||||
|     _FILTER_POST = 'http://www.metacafe.com/f/index.php?inputType=filter&controllerGroup=user' | ||||
|     IE_NAME = u'metacafe' | ||||
|     _TESTS = [{ | ||||
|     _TESTS = [ | ||||
|     # Youtube video | ||||
|     { | ||||
|         u"add_ie": ["Youtube"], | ||||
|         u"url":  u"http://metacafe.com/watch/yt-_aUehQsCQtM/the_electric_company_short_i_pbs_kids_go/", | ||||
|         u"file":  u"_aUehQsCQtM.flv", | ||||
|         u"file":  u"_aUehQsCQtM.mp4", | ||||
|         u"info_dict": { | ||||
|             u"upload_date": u"20090102", | ||||
|             u"title": u"The Electric Company | \"Short I\" | PBS KIDS GO!", | ||||
| @@ -32,15 +34,42 @@ class MetacafeIE(InfoExtractor): | ||||
|             u"uploader_id": u"PBS" | ||||
|         } | ||||
|     }, | ||||
|     # Normal metacafe video | ||||
|     { | ||||
|         u'url': u'http://www.metacafe.com/watch/11121940/news_stuff_you_wont_do_with_your_playstation_4/', | ||||
|         u'md5': u'6e0bca200eaad2552e6915ed6fd4d9ad', | ||||
|         u'info_dict': { | ||||
|             u'id': u'11121940', | ||||
|             u'ext': u'mp4', | ||||
|             u'title': u'News: Stuff You Won\'t Do with Your PlayStation 4', | ||||
|             u'uploader': u'ign', | ||||
|             u'description': u'Sony released a massive FAQ on the PlayStation Blog detailing the PS4\'s capabilities and limitations.', | ||||
|         }, | ||||
|     }, | ||||
|     # AnyClip video | ||||
|     { | ||||
|         u"url": u"http://www.metacafe.com/watch/an-dVVXnuY7Jh77J/the_andromeda_strain_1971_stop_the_bomb_part_3/", | ||||
|         u"file": u"an-dVVXnuY7Jh77J.mp4", | ||||
|         u"info_dict": { | ||||
|             u"title": u"The Andromeda Strain (1971): Stop the Bomb Part 3", | ||||
|             u"uploader": u"anyclip", | ||||
|             u"description": u"md5:38c711dd98f5bb87acf973d573442e67" | ||||
|         } | ||||
|     }] | ||||
|             u"description": u"md5:38c711dd98f5bb87acf973d573442e67", | ||||
|         }, | ||||
|     }, | ||||
|     # age-restricted video | ||||
|     { | ||||
|         u'url': u'http://www.metacafe.com/watch/5186653/bbc_internal_christmas_tape_79_uncensored_outtakes_etc/', | ||||
|         u'md5': u'98dde7c1a35d02178e8ab7560fe8bd09', | ||||
|         u'info_dict': { | ||||
|             u'id': u'5186653', | ||||
|             u'ext': u'mp4', | ||||
|             u'title': u'BBC INTERNAL Christmas Tape \'79 - UNCENSORED Outtakes, Etc.', | ||||
|             u'uploader': u'Dwayne Pipe', | ||||
|             u'description': u'md5:950bf4c581e2c059911fa3ffbe377e4b', | ||||
|             u'age_limit': 18, | ||||
|         }, | ||||
|     }, | ||||
|     ] | ||||
|  | ||||
|  | ||||
|     def report_disclaimer(self): | ||||
| @@ -62,6 +91,7 @@ class MetacafeIE(InfoExtractor): | ||||
|             'submit': "Continue - I'm over 18", | ||||
|             } | ||||
|         request = compat_urllib_request.Request(self._FILTER_POST, compat_urllib_parse.urlencode(disclaimer_form)) | ||||
|         request.add_header('Content-Type', 'application/x-www-form-urlencoded') | ||||
|         try: | ||||
|             self.report_age_confirmation() | ||||
|             compat_urllib_request.urlopen(request).read() | ||||
| @@ -83,7 +113,12 @@ class MetacafeIE(InfoExtractor): | ||||
|  | ||||
|         # Retrieve video webpage to extract further information | ||||
|         req = compat_urllib_request.Request('http://www.metacafe.com/watch/%s/' % video_id) | ||||
|         req.headers['Cookie'] = 'flashVersion=0;' | ||||
|  | ||||
|         # AnyClip videos require the flashversion cookie so that we get the link | ||||
|         # to the mp4 file | ||||
|         mobj_an = re.match(r'^an-(.*?)$', video_id) | ||||
|         if mobj_an: | ||||
|             req.headers['Cookie'] = 'flashVersion=0;' | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         # Extract URL, uploader and title from webpage | ||||
| @@ -125,6 +160,11 @@ class MetacafeIE(InfoExtractor): | ||||
|                 r'submitter=(.*?);|googletag\.pubads\(\)\.setTargeting\("(?:channel|submiter)","([^"]+)"\);', | ||||
|                 webpage, u'uploader nickname', fatal=False) | ||||
|  | ||||
|         if re.search(r'"contentRating":"restricted"', webpage) is not None: | ||||
|             age_limit = 18 | ||||
|         else: | ||||
|             age_limit = 0 | ||||
|  | ||||
|         return { | ||||
|             '_type':    'video', | ||||
|             'id':       video_id, | ||||
| @@ -134,4 +174,5 @@ class MetacafeIE(InfoExtractor): | ||||
|             'upload_date':  None, | ||||
|             'title':    video_title, | ||||
|             'ext':      video_ext, | ||||
|             'age_limit': age_limit, | ||||
|         } | ||||
|   | ||||
							
								
								
									
										49
									
								
								youtube_dl/extractor/mofosex.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								youtube_dl/extractor/mofosex.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | ||||
| import os | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urllib_parse_urlparse, | ||||
|     compat_urllib_request, | ||||
|     compat_urllib_parse, | ||||
| ) | ||||
|  | ||||
| class MofosexIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?(?P<url>mofosex\.com/videos/(?P<videoid>[0-9]+)/.*?\.html)' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.mofosex.com/videos/5018/japanese-teen-music-video.html', | ||||
|         u'file': u'5018.mp4', | ||||
|         u'md5': u'1b2eb47ac33cc75d4a80e3026b613c5a', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Japanese Teen Music Video", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('videoid') | ||||
|         url = 'http://www.' + mobj.group('url') | ||||
|  | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('Cookie', 'age_verified=1') | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         video_title = self._html_search_regex(r'<h1>(.+?)<', webpage, u'title') | ||||
|         video_url = compat_urllib_parse.unquote(self._html_search_regex(r'flashvars.video_url = \'([^\']+)', webpage, u'video_url')) | ||||
|         path = compat_urllib_parse_urlparse(video_url).path | ||||
|         extension = os.path.splitext(path)[1][1:] | ||||
|         format = path.split('/')[5].split('_')[:2] | ||||
|         format = "-".join(format) | ||||
|  | ||||
|         age_limit = self._rta_search(webpage) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': video_title, | ||||
|             'url': video_url, | ||||
|             'ext': extension, | ||||
|             'format': format, | ||||
|             'format_id': format, | ||||
|             'age_limit': age_limit, | ||||
|         } | ||||
| @@ -26,6 +26,7 @@ class MTVIE(InfoExtractor): | ||||
|             }, | ||||
|         }, | ||||
|         { | ||||
|             u'add_ie': ['Vevo'], | ||||
|             u'url': u'http://www.mtv.com/videos/taylor-swift/916187/everything-has-changed-ft-ed-sheeran.jhtml', | ||||
|             u'file': u'USCJY1331283.mp4', | ||||
|             u'md5': u'73b4e7fcadd88929292fe52c3ced8caf', | ||||
| @@ -47,7 +48,7 @@ class MTVIE(InfoExtractor): | ||||
|     def _transform_rtmp_url(rtmp_video_url): | ||||
|         m = re.match(r'^rtmpe?://.*?/(?P<finalid>gsp\..+?/.*)$', rtmp_video_url) | ||||
|         if not m: | ||||
|             raise ExtractorError(u'Cannot transform RTMP url') | ||||
|             return rtmp_video_url | ||||
|         base = 'http://mtvnmobile.vo.llnwd.net/kip0/_pxn=1+_pxI0=Ripod-h264+_pxL0=undefined+_pxM0=+_pxK=18639+_pxE=mp4/44620/mtvnorigin/' | ||||
|         return base + m.group('finalid') | ||||
|  | ||||
| @@ -80,6 +81,8 @@ class MTVIE(InfoExtractor): | ||||
|         video_id = self._id_from_uri(uri) | ||||
|         self.report_extraction(video_id) | ||||
|         mediagen_url = itemdoc.find('%s/%s' % (_media_xml_tag('group'), _media_xml_tag('content'))).attrib['url'] | ||||
|         # Remove the templates, like &device={device} | ||||
|         mediagen_url = re.sub(r'&[^=]*?={.*?}(?=(&|$))', u'', mediagen_url) | ||||
|         if 'acceptMethods' not in mediagen_url: | ||||
|             mediagen_url += '&acceptMethods=fms' | ||||
|         mediagen_page = self._download_webpage(mediagen_url, video_id, | ||||
|   | ||||
							
								
								
									
										48
									
								
								youtube_dl/extractor/myspace.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								youtube_dl/extractor/myspace.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| import re | ||||
| import json | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_str, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class MySpaceIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://myspace\.com/([^/]+)/video/[^/]+/(?P<id>\d+)' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'url': u'https://myspace.com/coldplay/video/viva-la-vida/100008689', | ||||
|         u'info_dict': { | ||||
|             u'id': u'100008689', | ||||
|             u'ext': u'flv', | ||||
|             u'title': u'Viva La Vida', | ||||
|             u'description': u'The official Viva La Vida video, directed by Hype Williams', | ||||
|             u'uploader': u'Coldplay', | ||||
|             u'uploader_id': u'coldplay', | ||||
|         }, | ||||
|         u'params': { | ||||
|             # rtmp download | ||||
|             u'skip_download': True, | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         context = json.loads(self._search_regex(r'context = ({.*?});', webpage, | ||||
|             u'context')) | ||||
|         video = context['video'] | ||||
|         rtmp_url, play_path = video['streamUrl'].split(';', 1) | ||||
|  | ||||
|         return { | ||||
|             'id': compat_str(video['mediaId']), | ||||
|             'title': video['title'], | ||||
|             'url': rtmp_url, | ||||
|             'play_path': play_path, | ||||
|             'ext': 'flv', | ||||
|             'description': video['description'], | ||||
|             'thumbnail': video['imageUrl'], | ||||
|             'uploader': video['artistName'], | ||||
|             'uploader_id': video['artistUsername'], | ||||
|         } | ||||
| @@ -90,8 +90,8 @@ class NHLVideocenterIE(NHLBaseInfoExtractor): | ||||
|              r'{statusIndex:0,index:0,.*?id:(.*?),'], | ||||
|             webpage, u'category id') | ||||
|         playlist_title = self._html_search_regex( | ||||
|             r'\?catid=%s">(.*?)</a>' % cat_id, | ||||
|             webpage, u'playlist title', flags=re.DOTALL) | ||||
|             r'tab0"[^>]*?>(.*?)</td>', | ||||
|             webpage, u'playlist title', flags=re.DOTALL).lower().capitalize() | ||||
|  | ||||
|         data = compat_urllib_parse.urlencode({ | ||||
|             'cid': cat_id, | ||||
|   | ||||
| @@ -20,7 +20,10 @@ class NowVideoIE(InfoExtractor): | ||||
|  | ||||
|         video_id = mobj.group('id') | ||||
|         webpage_url = 'http://www.nowvideo.ch/video/' + video_id | ||||
|         embed_url = 'http://embed.nowvideo.ch/embed.php?v=' + video_id | ||||
|         webpage = self._download_webpage(webpage_url, video_id) | ||||
|         embed_page = self._download_webpage(embed_url, video_id, | ||||
|             u'Downloading embed page') | ||||
|  | ||||
|         self.report_extraction(video_id) | ||||
|  | ||||
| @@ -28,7 +31,7 @@ class NowVideoIE(InfoExtractor): | ||||
|             webpage, u'video title') | ||||
|  | ||||
|         video_key = self._search_regex(r'var fkzd="(.*)";', | ||||
|             webpage, u'video key') | ||||
|             embed_page, u'video key') | ||||
|  | ||||
|         api_call = "http://www.nowvideo.ch/api/player.api.php?file={0}&numOfErrors=0&cid=1&key={1}".format(video_id, video_key) | ||||
|         api_response = self._download_webpage(api_call, video_id, | ||||
|   | ||||
							
								
								
									
										69
									
								
								youtube_dl/extractor/pornhub.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								youtube_dl/extractor/pornhub.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| import os | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urllib_parse_urlparse, | ||||
|     compat_urllib_request, | ||||
|     compat_urllib_parse, | ||||
|     unescapeHTML, | ||||
| ) | ||||
| from ..aes import ( | ||||
|     aes_decrypt_text | ||||
| ) | ||||
|  | ||||
| class PornHubIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?(?P<url>pornhub\.com/view_video\.php\?viewkey=(?P<videoid>[0-9]+))' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.pornhub.com/view_video.php?viewkey=648719015', | ||||
|         u'file': u'648719015.mp4', | ||||
|         u'md5': u'882f488fa1f0026f023f33576004a2ed', | ||||
|         u'info_dict': { | ||||
|             u"uploader": u"BABES-COM",  | ||||
|             u"title": u"Seductive Indian beauty strips down and fingers her pink pussy", | ||||
|             u"age_limit": 18 | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('videoid') | ||||
|         url = 'http://www.' + mobj.group('url') | ||||
|  | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('Cookie', 'age_verified=1') | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         video_title = self._html_search_regex(r'<h1 [^>]+>([^<]+)', webpage, u'title') | ||||
|         video_uploader = self._html_search_regex(r'<b>From: </b>(?:\s|<[^>]*>)*(.+?)<', webpage, u'uploader', fatal=False) | ||||
|         thumbnail = self._html_search_regex(r'"image_url":"([^"]+)', webpage, u'thumbnail', fatal=False) | ||||
|         if thumbnail: | ||||
|             thumbnail = compat_urllib_parse.unquote(thumbnail) | ||||
|  | ||||
|         video_urls = list(map(compat_urllib_parse.unquote , re.findall(r'"quality_[0-9]{3}p":"([^"]+)', webpage))) | ||||
|         if webpage.find('"encrypted":true') != -1: | ||||
|             password = self._html_search_regex(r'"video_title":"([^"]+)', webpage, u'password').replace('+', ' ') | ||||
|             video_urls = list(map(lambda s: aes_decrypt_text(s, password, 32).decode('utf-8'), video_urls)) | ||||
|  | ||||
|         formats = [] | ||||
|         for video_url in video_urls: | ||||
|             path = compat_urllib_parse_urlparse(video_url).path | ||||
|             extension = os.path.splitext(path)[1][1:] | ||||
|             format = path.split('/')[5].split('_')[:2] | ||||
|             format = "-".join(format) | ||||
|             formats.append({ | ||||
|                 'url': video_url, | ||||
|                 'ext': extension, | ||||
|                 'format': format, | ||||
|                 'format_id': format, | ||||
|             }) | ||||
|         formats.sort(key=lambda format: list(map(lambda s: s.zfill(6), format['format'].split('-')))) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'uploader': video_uploader, | ||||
|             'title': video_title, | ||||
|             'thumbnail': thumbnail, | ||||
|             'formats': formats, | ||||
|             'age_limit': 18, | ||||
|         } | ||||
| @@ -16,7 +16,8 @@ class PornotubeIE(InfoExtractor): | ||||
|         u'md5': u'374dd6dcedd24234453b295209aa69b6', | ||||
|         u'info_dict': { | ||||
|             u"upload_date": u"20090708",  | ||||
|             u"title": u"Marilyn-Monroe-Bathing" | ||||
|             u"title": u"Marilyn-Monroe-Bathing", | ||||
|             u"age_limit": 18 | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -8,9 +8,12 @@ class RedTubeIE(InfoExtractor): | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.redtube.com/66418', | ||||
|         u'file': u'66418.mp4', | ||||
|         u'md5': u'7b8c22b5e7098a3e1c09709df1126d2d', | ||||
|         # md5 varies from time to time, as in | ||||
|         # https://travis-ci.org/rg3/youtube-dl/jobs/14052463#L295 | ||||
|         #u'md5': u'7b8c22b5e7098a3e1c09709df1126d2d', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Sucked on a toilet" | ||||
|             u"title": u"Sucked on a toilet", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -62,19 +62,6 @@ class RTLnowIE(InfoExtractor): | ||||
|             u'skip_download': True, | ||||
|         }, | ||||
|     }, | ||||
|     { | ||||
|         u'url': u'http://www.rtlnitronow.de/recht-ordnung/lebensmittelkontrolle-erlangenordnungsamt-berlin.php?film_id=127367&player=1&season=1', | ||||
|         u'file': u'127367.flv', | ||||
|         u'info_dict': { | ||||
|             u'upload_date': u'20130926',  | ||||
|             u'title': u'Recht & Ordnung - Lebensmittelkontrolle Erlangen/Ordnungsamt...', | ||||
|             u'description': u'Lebensmittelkontrolle Erlangen/Ordnungsamt Berlin', | ||||
|             u'thumbnail': u'http://autoimg.static-fra.de/nitronow/344787/1500x1500/image2.jpg', | ||||
|         }, | ||||
|         u'params': { | ||||
|             u'skip_download': True, | ||||
|         }, | ||||
|     }, | ||||
|     { | ||||
|         u'url': u'http://www.n-tvnow.de/top-gear/episode-1-2013-01-01-00-00-00.php?film_id=124903&player=1&season=10', | ||||
|         u'file': u'124903.flv', | ||||
|   | ||||
| @@ -7,6 +7,7 @@ class SlashdotIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://tv.slashdot.org/video/\?embed=(?P<id>.*?)(&|$)' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'add_ie': ['Ooyala'], | ||||
|         u'url': u'http://tv.slashdot.org/video/?embed=JscHMzZDplD0p-yNLOzTfzC3Q3xzJaUz', | ||||
|         u'file': u'JscHMzZDplD0p-yNLOzTfzC3Q3xzJaUz.mp4', | ||||
|         u'md5': u'd2222e7a4a4c1541b3e0cf732fb26735', | ||||
|   | ||||
| @@ -29,17 +29,34 @@ class SoundcloudIE(InfoExtractor): | ||||
|                     ) | ||||
|                     ''' | ||||
|     IE_NAME = u'soundcloud' | ||||
|     _TEST = { | ||||
|         u'url': u'http://soundcloud.com/ethmusic/lostin-powers-she-so-heavy', | ||||
|         u'file': u'62986583.mp3', | ||||
|         u'md5': u'ebef0a451b909710ed1d7787dddbf0d7', | ||||
|         u'info_dict': { | ||||
|             u"upload_date": u"20121011",  | ||||
|             u"description": u"No Downloads untill we record the finished version this weekend, i was too pumped n i had to post it , earl is prolly gonna b hella p.o'd",  | ||||
|             u"uploader": u"E.T. ExTerrestrial Music",  | ||||
|             u"title": u"Lostin Powers - She so Heavy (SneakPreview) Adrian Ackers Blueprint 1" | ||||
|         } | ||||
|     } | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             u'url': u'http://soundcloud.com/ethmusic/lostin-powers-she-so-heavy', | ||||
|             u'file': u'62986583.mp3', | ||||
|             u'md5': u'ebef0a451b909710ed1d7787dddbf0d7', | ||||
|             u'info_dict': { | ||||
|                 u"upload_date": u"20121011",  | ||||
|                 u"description": u"No Downloads untill we record the finished version this weekend, i was too pumped n i had to post it , earl is prolly gonna b hella p.o'd",  | ||||
|                 u"uploader": u"E.T. ExTerrestrial Music",  | ||||
|                 u"title": u"Lostin Powers - She so Heavy (SneakPreview) Adrian Ackers Blueprint 1" | ||||
|             } | ||||
|         }, | ||||
|         # not streamable song | ||||
|         { | ||||
|             u'url': u'https://soundcloud.com/the-concept-band/goldrushed-mastered?in=the-concept-band/sets/the-royal-concept-ep', | ||||
|             u'info_dict': { | ||||
|                 u'id': u'47127627', | ||||
|                 u'ext': u'mp3', | ||||
|                 u'title': u'Goldrushed', | ||||
|                 u'uploader': u'The Royal Concept', | ||||
|                 u'upload_date': u'20120521', | ||||
|             }, | ||||
|             u'params': { | ||||
|                 # rtmp | ||||
|                 u'skip_download': True, | ||||
|             }, | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     _CLIENT_ID = 'b45b1aa10f1ac2941910a7f0d10f8e28' | ||||
|  | ||||
| @@ -56,24 +73,39 @@ class SoundcloudIE(InfoExtractor): | ||||
|         return 'http://api.soundcloud.com/resolve.json?url=' + url + '&client_id=' + cls._CLIENT_ID | ||||
|  | ||||
|     def _extract_info_dict(self, info, full_title=None, quiet=False): | ||||
|         video_id = info['id'] | ||||
|         name = full_title or video_id | ||||
|         track_id = compat_str(info['id']) | ||||
|         name = full_title or track_id | ||||
|         if quiet == False: | ||||
|             self.report_extraction(name) | ||||
|  | ||||
|         thumbnail = info['artwork_url'] | ||||
|         if thumbnail is not None: | ||||
|             thumbnail = thumbnail.replace('-large', '-t500x500') | ||||
|         return { | ||||
|             'id':       info['id'], | ||||
|         result = { | ||||
|             'id':       track_id, | ||||
|             'url':      info['stream_url'] + '?client_id=' + self._CLIENT_ID, | ||||
|             'uploader': info['user']['username'], | ||||
|             'upload_date': unified_strdate(info['created_at']), | ||||
|             'title':    info['title'], | ||||
|             'ext':      u'mp3', | ||||
|             'ext':      info.get('original_format', u'mp3'), | ||||
|             'description': info['description'], | ||||
|             'thumbnail': thumbnail, | ||||
|         } | ||||
|         if info.get('downloadable', False): | ||||
|             result['url'] = 'https://api.soundcloud.com/tracks/{0}/download?client_id={1}'.format(track_id, self._CLIENT_ID) | ||||
|         if not info.get('streamable', False): | ||||
|             # We have to get the rtmp url | ||||
|             stream_json = self._download_webpage( | ||||
|                 'http://api.soundcloud.com/i1/tracks/{0}/streams?client_id={1}'.format(track_id, self._CLIENT_ID), | ||||
|                 track_id, u'Downloading track url') | ||||
|             rtmp_url = json.loads(stream_json)['rtmp_mp3_128_url'] | ||||
|             # The url doesn't have an rtmp app, we have to extract the playpath | ||||
|             url, path = rtmp_url.split('mp3:', 1) | ||||
|             result.update({ | ||||
|                 'url': url, | ||||
|                 'play_path': 'mp3:' + path, | ||||
|             }) | ||||
|         return result | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url, flags=re.VERBOSE) | ||||
| @@ -106,70 +138,8 @@ class SoundcloudIE(InfoExtractor): | ||||
| class SoundcloudSetIE(SoundcloudIE): | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?soundcloud\.com/([\w\d-]+)/sets/([\w\d-]+)(?:[?].*)?$' | ||||
|     IE_NAME = u'soundcloud:set' | ||||
|     _TEST = { | ||||
|         u"url":"https://soundcloud.com/the-concept-band/sets/the-royal-concept-ep", | ||||
|         u"playlist": [ | ||||
|             { | ||||
|                 u"file":"30510138.mp3", | ||||
|                 u"md5":"f9136bf103901728f29e419d2c70f55d", | ||||
|                 u"info_dict": { | ||||
|                     u"upload_date": u"20111213", | ||||
|                     u"description": u"The Royal Concept from Stockholm\r\nFilip / Povel / David / Magnus\r\nwww.royalconceptband.com", | ||||
|                     u"uploader": u"The Royal Concept", | ||||
|                     u"title": u"D-D-Dance" | ||||
|                 } | ||||
|             }, | ||||
|             { | ||||
|                 u"file":"47127625.mp3", | ||||
|                 u"md5":"09b6758a018470570f8fd423c9453dd8", | ||||
|                 u"info_dict": { | ||||
|                     u"upload_date": u"20120521", | ||||
|                     u"description": u"The Royal Concept from Stockholm\r\nFilip / Povel / David / Magnus\r\nwww.royalconceptband.com", | ||||
|                     u"uploader": u"The Royal Concept", | ||||
|                     u"title": u"The Royal Concept - Gimme Twice" | ||||
|                 } | ||||
|             }, | ||||
|             { | ||||
|                 u"file":"47127627.mp3", | ||||
|                 u"md5":"154abd4e418cea19c3b901f1e1306d9c", | ||||
|                 u"info_dict": { | ||||
|                     u"upload_date": u"20120521", | ||||
|                     u"uploader": u"The Royal Concept", | ||||
|                     u"title": u"Goldrushed" | ||||
|                 } | ||||
|             }, | ||||
|             { | ||||
|                 u"file":"47127629.mp3", | ||||
|                 u"md5":"2f5471edc79ad3f33a683153e96a79c1", | ||||
|                 u"info_dict": { | ||||
|                     u"upload_date": u"20120521", | ||||
|                     u"description": u"The Royal Concept from Stockholm\r\nFilip / Povel / David / Magnus\r\nwww.royalconceptband.com", | ||||
|                     u"uploader": u"The Royal Concept", | ||||
|                     u"title": u"In the End" | ||||
|                 } | ||||
|             }, | ||||
|             { | ||||
|                 u"file":"47127631.mp3", | ||||
|                 u"md5":"f9ba87aa940af7213f98949254f1c6e2", | ||||
|                 u"info_dict": { | ||||
|                     u"upload_date": u"20120521", | ||||
|                     u"description": u"The Royal Concept from Stockholm\r\nFilip / David / Povel / Magnus\r\nwww.theroyalconceptband.com", | ||||
|                     u"uploader": u"The Royal Concept", | ||||
|                     u"title": u"Knocked Up" | ||||
|                 } | ||||
|             }, | ||||
|             { | ||||
|                 u"file":"75206121.mp3", | ||||
|                 u"md5":"f9d1fe9406717e302980c30de4af9353", | ||||
|                 u"info_dict": { | ||||
|                     u"upload_date": u"20130116", | ||||
|                     u"description": u"The unreleased track World on Fire premiered on the CW's hit show Arrow (8pm/7pm central).  \r\nAs a gift to our fans we would like to offer you a free download of the track!  ", | ||||
|                     u"uploader": u"The Royal Concept", | ||||
|                     u"title": u"World On Fire" | ||||
|                 } | ||||
|             } | ||||
|         ] | ||||
|     } | ||||
|     # it's in tests/test_playlists.py | ||||
|     _TESTS = [] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
| @@ -208,7 +178,7 @@ class SoundcloudUserIE(SoundcloudIE): | ||||
|     IE_NAME = u'soundcloud:user' | ||||
|  | ||||
|     # it's in tests/test_playlists.py | ||||
|     _TEST = None | ||||
|     _TESTS = [] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|   | ||||
| @@ -5,21 +5,19 @@ from .mtv import MTVIE, _media_xml_tag | ||||
|  | ||||
| class SouthParkStudiosIE(MTVIE): | ||||
|     IE_NAME = u'southparkstudios.com' | ||||
|     _VALID_URL = r'https?://www\.southparkstudios\.com/(clips|full-episodes)/(?P<id>.+?)(\?|#|$)' | ||||
|     _VALID_URL = r'(https?://)?(www\.)?(?P<url>southparkstudios\.com/(clips|full-episodes)/(?P<id>.+?)(\?|#|$))' | ||||
|  | ||||
|     _FEED_URL = 'http://www.southparkstudios.com/feeds/video-player/mrss' | ||||
|  | ||||
|     _TEST = { | ||||
|     # Overwrite MTVIE properties we don't want | ||||
|     _TESTS = [{ | ||||
|         u'url': u'http://www.southparkstudios.com/clips/104437/bat-daded#tab=featured', | ||||
|         u'file': u'a7bff6c2-ed00-11e0-aca6-0026b9414f30.mp4', | ||||
|         u'info_dict': { | ||||
|             u'title': u'Bat Daded', | ||||
|             u'description': u'Randy disqualifies South Park by getting into a fight with Bat Dad.', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     # Overwrite MTVIE properties we don't want | ||||
|     _TESTS = [] | ||||
|     }] | ||||
|  | ||||
|     def _get_thumbnail_url(self, uri, itemdoc): | ||||
|         search_path = '%s/%s' % (_media_xml_tag('group'), _media_xml_tag('thumbnail')) | ||||
| @@ -31,8 +29,23 @@ class SouthParkStudiosIE(MTVIE): | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         url = u'http://www.' + mobj.group(u'url') | ||||
|         video_id = mobj.group('id') | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         mgid = self._search_regex(r'swfobject.embedSWF\(".*?(mgid:.*?)"', | ||||
|                                   webpage, u'mgid') | ||||
|         return self._get_videos_info(mgid) | ||||
|  | ||||
| class SouthparkDeIE(SouthParkStudiosIE): | ||||
|     IE_NAME = u'southpark.de' | ||||
|     _VALID_URL = r'(https?://)?(www\.)?(?P<url>southpark\.de/(clips|alle-episoden)/(?P<id>.+?)(\?|#|$))' | ||||
|     _FEED_URL = 'http://www.southpark.de/feeds/video-player/mrss/' | ||||
|  | ||||
|     _TESTS = [{ | ||||
|         u'url': u'http://www.southpark.de/clips/uygssh/the-government-wont-respect-my-privacy#tab=featured', | ||||
|         u'file': u'85487c96-b3b9-4e39-9127-ad88583d9bf2.mp4', | ||||
|         u'info_dict': { | ||||
|             u'title': u'The Government Won\'t Respect My Privacy', | ||||
|             u'description': u'Cartman explains the benefits of "Shitter" to Stan, Kyle and Craig.', | ||||
|         }, | ||||
|     }] | ||||
|   | ||||
							
								
								
									
										35
									
								
								youtube_dl/extractor/space.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								youtube_dl/extractor/space.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from .brightcove import BrightcoveIE | ||||
| from ..utils import RegexNotFoundError, ExtractorError | ||||
|  | ||||
|  | ||||
| class SpaceIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://www\.space\.com/\d+-(?P<title>[^/\.\?]*?)-video.html' | ||||
|     _TEST = { | ||||
|         u'add_ie': ['Brightcove'], | ||||
|         u'url': u'http://www.space.com/23373-huge-martian-landforms-detail-revealed-by-european-probe-video.html', | ||||
|         u'info_dict': { | ||||
|             u'id': u'2780937028001', | ||||
|             u'ext': u'mp4', | ||||
|             u'title': u'Huge Martian Landforms\' Detail Revealed By European Probe | Video', | ||||
|             u'description': u'md5:db81cf7f3122f95ed234b631a6ea1e61', | ||||
|             u'uploader': u'TechMedia Networks', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         title = mobj.group('title') | ||||
|         webpage = self._download_webpage(url, title) | ||||
|         try: | ||||
|             # Some videos require the playerKey field, which isn't define in | ||||
|             # the BrightcoveExperience object | ||||
|             brightcove_url = self._og_search_video_url(webpage) | ||||
|         except RegexNotFoundError: | ||||
|             # Other videos works fine with the info from the object | ||||
|             brightcove_url = BrightcoveIE._extract_brightcove_url(webpage) | ||||
|         if brightcove_url is None: | ||||
|             raise ExtractorError(u'The webpage does not contain a video', expected=True) | ||||
|         return self.url_result(brightcove_url, BrightcoveIE.ie_key()) | ||||
							
								
								
									
										74
									
								
								youtube_dl/extractor/spankwire.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								youtube_dl/extractor/spankwire.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,74 @@ | ||||
| import os | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urllib_parse_urlparse, | ||||
|     compat_urllib_request, | ||||
|     compat_urllib_parse, | ||||
|     unescapeHTML, | ||||
| ) | ||||
| from ..aes import ( | ||||
|     aes_decrypt_text | ||||
| ) | ||||
|  | ||||
| class SpankwireIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?(?P<url>spankwire\.com/[^/]*/video(?P<videoid>[0-9]+)/?)' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.spankwire.com/Buckcherry-s-X-Rated-Music-Video-Crazy-Bitch/video103545/', | ||||
|         u'file': u'103545.mp4', | ||||
|         u'md5': u'1b3f55e345500552dbc252a3e9c1af43', | ||||
|         u'info_dict': { | ||||
|             u"uploader": u"oreusz",  | ||||
|             u"title": u"Buckcherry`s X Rated Music Video Crazy Bitch", | ||||
|             u"description": u"Crazy Bitch X rated music video.", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('videoid') | ||||
|         url = 'http://www.' + mobj.group('url') | ||||
|  | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('Cookie', 'age_verified=1') | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         video_title = self._html_search_regex(r'<h1>([^<]+)', webpage, u'title') | ||||
|         video_uploader = self._html_search_regex(r'by:\s*<a [^>]*>(.+?)</a>', webpage, u'uploader', fatal=False) | ||||
|         thumbnail = self._html_search_regex(r'flashvars\.image_url = "([^"]+)', webpage, u'thumbnail', fatal=False) | ||||
|         description = self._html_search_regex(r'>\s*Description:</div>\s*<[^>]*>([^<]+)', webpage, u'description', fatal=False) | ||||
|         if len(description) == 0: | ||||
|             description = None | ||||
|  | ||||
|         video_urls = list(map(compat_urllib_parse.unquote , re.findall(r'flashvars\.quality_[0-9]{3}p = "([^"]+)', webpage))) | ||||
|         if webpage.find('flashvars\.encrypted = "true"') != -1: | ||||
|             password = self._html_search_regex(r'flashvars\.video_title = "([^"]+)', webpage, u'password').replace('+', ' ') | ||||
|             video_urls = list(map(lambda s: aes_decrypt_text(s, password, 32).decode('utf-8'), video_urls)) | ||||
|  | ||||
|         formats = [] | ||||
|         for video_url in video_urls: | ||||
|             path = compat_urllib_parse_urlparse(video_url).path | ||||
|             extension = os.path.splitext(path)[1][1:] | ||||
|             format = path.split('/')[4].split('_')[:2] | ||||
|             format = "-".join(format) | ||||
|             formats.append({ | ||||
|                 'url': video_url, | ||||
|                 'ext': extension, | ||||
|                 'format': format, | ||||
|                 'format_id': format, | ||||
|             }) | ||||
|         formats.sort(key=lambda format: list(map(lambda s: s.zfill(6), format['format'].split('-')))) | ||||
|  | ||||
|         age_limit = self._rta_search(webpage) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'uploader': video_uploader, | ||||
|             'title': video_title, | ||||
|             'thumbnail': thumbnail, | ||||
|             'description': description, | ||||
|             'formats': formats, | ||||
|             'age_limit': age_limit, | ||||
|         } | ||||
| @@ -2,18 +2,27 @@ import re | ||||
| import xml.etree.ElementTree | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import determine_ext | ||||
|  | ||||
|  | ||||
| class SpiegelIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?spiegel\.de/video/[^/]*-(?P<videoID>[0-9]+)(?:\.html)?(?:#.*)?$' | ||||
|     _TEST = { | ||||
|     _TESTS = [{ | ||||
|         u'url': u'http://www.spiegel.de/video/vulkan-tungurahua-in-ecuador-ist-wieder-aktiv-video-1259285.html', | ||||
|         u'file': u'1259285.mp4', | ||||
|         u'md5': u'2c2754212136f35fb4b19767d242f66e', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Vulkanausbruch in Ecuador: Der \"Feuerschlund\" ist wieder aktiv" | ||||
|         } | ||||
|     } | ||||
|     }, | ||||
|     { | ||||
|         u'url': u'http://www.spiegel.de/video/schach-wm-videoanalyse-des-fuenften-spiels-video-1309159.html', | ||||
|         u'file': u'1309159.mp4', | ||||
|         u'md5': u'f2cdf638d7aa47654e251e1aee360af1', | ||||
|         u'info_dict': { | ||||
|             u'title': u'Schach-WM in der Videoanalyse: Carlsen nutzt die Fehlgriffe des Titelverteidigers' | ||||
|         } | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         m = re.match(self._VALID_URL, url) | ||||
| @@ -21,25 +30,38 @@ class SpiegelIE(InfoExtractor): | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         video_title = self._html_search_regex(r'<div class="module-title">(.*?)</div>', | ||||
|             webpage, u'title') | ||||
|         video_title = self._html_search_regex( | ||||
|             r'<div class="module-title">(.*?)</div>', webpage, u'title') | ||||
|  | ||||
|         xml_url = u'http://video2.spiegel.de/flash/' + video_id + u'.xml' | ||||
|         xml_code = self._download_webpage(xml_url, video_id, | ||||
|                     note=u'Downloading XML', errnote=u'Failed to download XML') | ||||
|         xml_code = self._download_webpage( | ||||
|             xml_url, video_id, | ||||
|             note=u'Downloading XML', errnote=u'Failed to download XML') | ||||
|  | ||||
|         idoc = xml.etree.ElementTree.fromstring(xml_code) | ||||
|         last_type = idoc[-1] | ||||
|         filename = last_type.findall('./filename')[0].text | ||||
|         duration = float(last_type.findall('./duration')[0].text) | ||||
|  | ||||
|         video_url = 'http://video2.spiegel.de/flash/' + filename | ||||
|         video_ext = filename.rpartition('.')[2] | ||||
|         formats = [ | ||||
|             { | ||||
|                 'format_id': n.tag.rpartition('type')[2], | ||||
|                 'url': u'http://video2.spiegel.de/flash/' + n.find('./filename').text, | ||||
|                 'width': int(n.find('./width').text), | ||||
|                 'height': int(n.find('./height').text), | ||||
|                 'abr': int(n.find('./audiobitrate').text), | ||||
|                 'vbr': int(n.find('./videobitrate').text), | ||||
|                 'vcodec': n.find('./codec').text, | ||||
|                 'acodec': 'MP4A', | ||||
|             } | ||||
|             for n in list(idoc) | ||||
|             # Blacklist type 6, it's extremely LQ and not available on the same server | ||||
|             if n.tag.startswith('type') and n.tag != 'type6' | ||||
|         ] | ||||
|         formats.sort(key=lambda f: f['vbr']) | ||||
|         duration = float(idoc[0].findall('./duration')[0].text) | ||||
|  | ||||
|         info = { | ||||
|             'id': video_id, | ||||
|             'url': video_url, | ||||
|             'ext': video_ext, | ||||
|             'title': video_title, | ||||
|             'duration': duration, | ||||
|             'formats': formats, | ||||
|         } | ||||
|         return [info] | ||||
|         return info | ||||
|   | ||||
| @@ -12,9 +12,9 @@ class SubtitlesInfoExtractor(InfoExtractor): | ||||
|         return any([self._downloader.params.get('writesubtitles', False), | ||||
|                     self._downloader.params.get('writeautomaticsub')]) | ||||
|  | ||||
|     def _list_available_subtitles(self, video_id, webpage=None): | ||||
|     def _list_available_subtitles(self, video_id, webpage): | ||||
|         """ outputs the available subtitles for the video """ | ||||
|         sub_lang_list = self._get_available_subtitles(video_id) | ||||
|         sub_lang_list = self._get_available_subtitles(video_id, webpage) | ||||
|         auto_captions_list = self._get_available_automatic_caption(video_id, webpage) | ||||
|         sub_lang = ",".join(list(sub_lang_list.keys())) | ||||
|         self.to_screen(u'%s: Available subtitles for video: %s' % | ||||
| @@ -23,7 +23,7 @@ class SubtitlesInfoExtractor(InfoExtractor): | ||||
|         self.to_screen(u'%s: Available automatic captions for video: %s' % | ||||
|                        (video_id, auto_lang)) | ||||
|  | ||||
|     def extract_subtitles(self, video_id, video_webpage=None): | ||||
|     def extract_subtitles(self, video_id, webpage): | ||||
|         """ | ||||
|         returns {sub_lang: sub} ,{} if subtitles not found or None if the | ||||
|         subtitles aren't requested. | ||||
| @@ -32,9 +32,9 @@ class SubtitlesInfoExtractor(InfoExtractor): | ||||
|             return None | ||||
|         available_subs_list = {} | ||||
|         if self._downloader.params.get('writeautomaticsub', False): | ||||
|             available_subs_list.update(self._get_available_automatic_caption(video_id, video_webpage)) | ||||
|             available_subs_list.update(self._get_available_automatic_caption(video_id, webpage)) | ||||
|         if self._downloader.params.get('writesubtitles', False): | ||||
|             available_subs_list.update(self._get_available_subtitles(video_id)) | ||||
|             available_subs_list.update(self._get_available_subtitles(video_id, webpage)) | ||||
|  | ||||
|         if not available_subs_list:  # error, it didn't get the available subtitles | ||||
|             return {} | ||||
| @@ -74,7 +74,7 @@ class SubtitlesInfoExtractor(InfoExtractor): | ||||
|             return | ||||
|         return sub | ||||
|  | ||||
|     def _get_available_subtitles(self, video_id): | ||||
|     def _get_available_subtitles(self, video_id, webpage): | ||||
|         """ | ||||
|         returns {sub_lang: url} or {} if not available | ||||
|         Must be redefined by the subclasses | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import re | ||||
| import xml.etree.ElementTree | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
| @@ -11,7 +12,7 @@ class TeamcocoIE(InfoExtractor): | ||||
|     _TEST = { | ||||
|         u'url': u'http://teamcoco.com/video/louis-ck-interview-george-w-bush', | ||||
|         u'file': u'19705.mp4', | ||||
|         u'md5': u'27b6f7527da5acf534b15f21b032656e', | ||||
|         u'md5': u'cde9ba0fa3506f5f017ce11ead928f9a', | ||||
|         u'info_dict': { | ||||
|             u"description": u"Louis C.K. got starstruck by George W. Bush, so what? Part one.",  | ||||
|             u"title": u"Louis C.K. Interview Pt. 1 11/3/11" | ||||
| @@ -31,16 +32,40 @@ class TeamcocoIE(InfoExtractor): | ||||
|         self.report_extraction(video_id) | ||||
|  | ||||
|         data_url = 'http://teamcoco.com/cvp/2.0/%s.xml' % video_id | ||||
|         data = self._download_webpage(data_url, video_id, 'Downloading data webpage') | ||||
|         data_xml = self._download_webpage(data_url, video_id, 'Downloading data webpage') | ||||
|         data = xml.etree.ElementTree.fromstring(data_xml.encode('utf-8')) | ||||
|  | ||||
|         video_url = self._html_search_regex(r'<file [^>]*type="high".*?>(.*?)</file>', | ||||
|             data, u'video URL') | ||||
|  | ||||
|         return [{ | ||||
|         qualities = ['500k', '480p', '1000k', '720p', '1080p'] | ||||
|         formats = [] | ||||
|         for file in data.findall('files/file'): | ||||
|             if file.attrib.get('playmode') == 'all': | ||||
|                 # it just duplicates one of the entries | ||||
|                 break | ||||
|             file_url = file.text | ||||
|             m_format = re.search(r'(\d+(k|p))\.mp4', file_url) | ||||
|             if m_format is not None: | ||||
|                 format_id = m_format.group(1) | ||||
|             else: | ||||
|                 format_id = file.attrib['bitrate'] | ||||
|             formats.append({ | ||||
|                 'url': file_url, | ||||
|                 'ext': 'mp4', | ||||
|                 'format_id': format_id, | ||||
|             }) | ||||
|         def sort_key(f): | ||||
|             try: | ||||
|                 return qualities.index(f['format_id']) | ||||
|             except ValueError: | ||||
|                 return -1 | ||||
|         formats.sort(key=sort_key) | ||||
|         if not formats: | ||||
|             raise RegexNotFoundError(u'Unable to extract video URL') | ||||
|  | ||||
|         return { | ||||
|             'id':          video_id, | ||||
|             'url':         video_url, | ||||
|             'ext':         'mp4', | ||||
|             'formats': formats, | ||||
|             'title':       self._og_search_title(webpage), | ||||
|             'thumbnail':   self._og_search_thumbnail(webpage), | ||||
|             'description': self._og_search_description(webpage), | ||||
|         }] | ||||
|         } | ||||
|   | ||||
| @@ -1,10 +1,14 @@ | ||||
| import json | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from .subtitles import SubtitlesInfoExtractor | ||||
|  | ||||
| from ..utils import ( | ||||
|     compat_str, | ||||
|     RegexNotFoundError, | ||||
| ) | ||||
|  | ||||
| class TEDIE(InfoExtractor): | ||||
| class TEDIE(SubtitlesInfoExtractor): | ||||
|     _VALID_URL=r'''http://www\.ted\.com/ | ||||
|                    ( | ||||
|                         ((?P<type_playlist>playlists)/(?P<playlist_id>\d+)) # We have a playlist | ||||
| @@ -32,33 +36,32 @@ class TEDIE(InfoExtractor): | ||||
|     def _real_extract(self, url): | ||||
|         m=re.match(self._VALID_URL, url, re.VERBOSE) | ||||
|         if m.group('type_talk'): | ||||
|             return [self._talk_info(url)] | ||||
|             return self._talk_info(url) | ||||
|         else : | ||||
|             playlist_id=m.group('playlist_id') | ||||
|             name=m.group('name') | ||||
|             self.to_screen(u'Getting info of playlist %s: "%s"' % (playlist_id,name)) | ||||
|             return [self._playlist_videos_info(url,name,playlist_id)] | ||||
|  | ||||
|     def _playlist_videos_info(self,url,name,playlist_id=0): | ||||
|  | ||||
|     def _playlist_videos_info(self, url, name, playlist_id): | ||||
|         '''Returns the videos of the playlist''' | ||||
|         video_RE=r''' | ||||
|                      <li\ id="talk_(\d+)"([.\s]*?)data-id="(?P<video_id>\d+)" | ||||
|                      ([.\s]*?)data-playlist_item_id="(\d+)" | ||||
|                      ([.\s]*?)data-mediaslug="(?P<mediaSlug>.+?)" | ||||
|                      ''' | ||||
|         video_name_RE=r'<p\ class="talk-title"><a href="(?P<talk_url>/talks/(.+).html)">(?P<fullname>.+?)</a></p>' | ||||
|         webpage=self._download_webpage(url, playlist_id, 'Downloading playlist webpage') | ||||
|         m_videos=re.finditer(video_RE,webpage,re.VERBOSE) | ||||
|         m_names=re.finditer(video_name_RE,webpage) | ||||
|  | ||||
|         webpage = self._download_webpage( | ||||
|             url, playlist_id, u'Downloading playlist webpage') | ||||
|         matches = re.finditer( | ||||
|             r'<p\s+class="talk-title[^"]*"><a\s+href="(?P<talk_url>/talks/[^"]+\.html)">[^<]*</a></p>', | ||||
|             webpage) | ||||
|  | ||||
|         playlist_title = self._html_search_regex(r'div class="headline">\s*?<h1>\s*?<span>(.*?)</span>', | ||||
|                                                  webpage, 'playlist title') | ||||
|  | ||||
|         playlist_entries = [] | ||||
|         for m_video, m_name in zip(m_videos,m_names): | ||||
|             talk_url='http://www.ted.com%s' % m_name.group('talk_url') | ||||
|             playlist_entries.append(self.url_result(talk_url, 'TED')) | ||||
|         return self.playlist_result(playlist_entries, playlist_id = playlist_id, playlist_title = playlist_title) | ||||
|         playlist_entries = [ | ||||
|             self.url_result(u'http://www.ted.com' + m.group('talk_url'), 'TED') | ||||
|             for m in matches | ||||
|         ] | ||||
|         return self.playlist_result( | ||||
|             playlist_entries, playlist_id=playlist_id, playlist_title=playlist_title) | ||||
|  | ||||
|     def _talk_info(self, url, video_id=0): | ||||
|         """Return the video for the talk in the url""" | ||||
| @@ -81,16 +84,35 @@ class TEDIE(InfoExtractor): | ||||
|             'ext': 'mp4', | ||||
|             'url': stream['file'], | ||||
|             'format': stream['id'] | ||||
|             } for stream in info['htmlStreams']] | ||||
|         info = { | ||||
|             'id': info['id'], | ||||
|         } for stream in info['htmlStreams']] | ||||
|  | ||||
|         video_id = info['id'] | ||||
|  | ||||
|         # subtitles | ||||
|         video_subtitles = self.extract_subtitles(video_id, webpage) | ||||
|         if self._downloader.params.get('listsubtitles', False): | ||||
|             self._list_available_subtitles(video_id, webpage) | ||||
|             return | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'thumbnail': thumbnail, | ||||
|             'description': desc, | ||||
|             'subtitles': video_subtitles, | ||||
|             'formats': formats, | ||||
|         } | ||||
|  | ||||
|         # TODO: Remove when #980 has been merged | ||||
|         info.update(info['formats'][-1]) | ||||
|  | ||||
|         return info | ||||
|     def _get_available_subtitles(self, video_id, webpage): | ||||
|         try: | ||||
|             options = self._search_regex(r'(?:<select name="subtitles_language_select" id="subtitles_language_select">)(.*?)(?:</select>)', webpage, 'subtitles_language_select', flags=re.DOTALL) | ||||
|             languages = re.findall(r'(?:<option value=")(\S+)"', options) | ||||
|             if languages: | ||||
|                 sub_lang_list = {} | ||||
|                 for l in languages: | ||||
|                     url = 'http://www.ted.com/talks/subtitles/id/%s/lang/%s/format/srt' % (video_id, l) | ||||
|                     sub_lang_list[l] = url | ||||
|                 return sub_lang_list | ||||
|         except RegexNotFoundError as err: | ||||
|             self._downloader.report_warning(u'video doesn\'t have subtitles') | ||||
|         return {} | ||||
|   | ||||
							
								
								
									
										65
									
								
								youtube_dl/extractor/tube8.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								youtube_dl/extractor/tube8.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | ||||
| import os | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urllib_parse_urlparse, | ||||
|     compat_urllib_request, | ||||
|     compat_urllib_parse, | ||||
|     unescapeHTML, | ||||
| ) | ||||
| from ..aes import ( | ||||
|     aes_decrypt_text | ||||
| ) | ||||
|  | ||||
| class Tube8IE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?(?P<url>tube8\.com/[^/]+/[^/]+/(?P<videoid>[0-9]+)/?)' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.tube8.com/teen/kasia-music-video/229795/', | ||||
|         u'file': u'229795.mp4', | ||||
|         u'md5': u'e9e0b0c86734e5e3766e653509475db0', | ||||
|         u'info_dict': { | ||||
|             u"description": u"hot teen Kasia grinding",  | ||||
|             u"uploader": u"unknown",  | ||||
|             u"title": u"Kasia music video", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('videoid') | ||||
|         url = 'http://www.' + mobj.group('url') | ||||
|  | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('Cookie', 'age_verified=1') | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         video_title = self._html_search_regex(r'videotitle	="([^"]+)', webpage, u'title') | ||||
|         video_description = self._html_search_regex(r'>Description:</strong>(.+?)<', webpage, u'description', fatal=False) | ||||
|         video_uploader = self._html_search_regex(r'>Submitted by:</strong>(?:\s|<[^>]*>)*(.+?)<', webpage, u'uploader', fatal=False) | ||||
|         thumbnail = self._html_search_regex(r'"image_url":"([^"]+)', webpage, u'thumbnail', fatal=False) | ||||
|         if thumbnail: | ||||
|             thumbnail = thumbnail.replace('\\/', '/') | ||||
|  | ||||
|         video_url = self._html_search_regex(r'"video_url":"([^"]+)', webpage, u'video_url') | ||||
|         if webpage.find('"encrypted":true')!=-1: | ||||
|             password = self._html_search_regex(r'"video_title":"([^"]+)', webpage, u'password') | ||||
|             video_url = aes_decrypt_text(video_url, password, 32).decode('utf-8') | ||||
|         path = compat_urllib_parse_urlparse(video_url).path | ||||
|         extension = os.path.splitext(path)[1][1:] | ||||
|         format = path.split('/')[4].split('_')[:2] | ||||
|         format = "-".join(format) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'uploader': video_uploader, | ||||
|             'title': video_title, | ||||
|             'thumbnail': thumbnail, | ||||
|             'description': video_description, | ||||
|             'url': video_url, | ||||
|             'ext': extension, | ||||
|             'format': format, | ||||
|             'format_id': format, | ||||
|             'age_limit': 18, | ||||
|         } | ||||
							
								
								
									
										42
									
								
								youtube_dl/extractor/tvp.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								youtube_dl/extractor/tvp.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| import json | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class TvpIE(InfoExtractor): | ||||
|     IE_NAME = u'tvp.pl' | ||||
|     _VALID_URL = r'https?://www\.tvp\.pl/.*?wideo/(?P<date>\d+)/(?P<id>\d+)' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.tvp.pl/warszawa/magazyny/campusnews/wideo/31102013/12878238', | ||||
|         u'md5': u'148408967a6a468953c0a75cbdaf0d7a', | ||||
|         u'file': u'12878238.wmv', | ||||
|         u'info_dict': { | ||||
|             u'title': u'31.10.2013 - Odcinek 2', | ||||
|             u'description': u'31.10.2013 - Odcinek 2', | ||||
|         }, | ||||
|         u'skip': u'Download has to use same server IP as extraction. Therefore, a good (load-balancing) DNS resolver will make the download fail.' | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         json_url = 'http://www.tvp.pl/pub/stat/videofileinfo?video_id=%s' % video_id | ||||
|         json_params = self._download_webpage( | ||||
|             json_url, video_id, u"Downloading video metadata") | ||||
|  | ||||
|         params = json.loads(json_params) | ||||
|         self.report_extraction(video_id) | ||||
|         video_url = params['video_url'] | ||||
|  | ||||
|         title = self._og_search_title(webpage, fatal=True) | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'ext': 'wmv', | ||||
|             'url': video_url, | ||||
|             'description': self._og_search_description(webpage), | ||||
|             'thumbnail': self._og_search_thumbnail(webpage), | ||||
|         } | ||||
| @@ -5,7 +5,7 @@ import datetime | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     determine_ext, | ||||
|     compat_HTTPError, | ||||
|     ExtractorError, | ||||
| ) | ||||
|  | ||||
| @@ -16,26 +16,22 @@ class VevoIE(InfoExtractor): | ||||
|     (currently used by MTVIE) | ||||
|     """ | ||||
|     _VALID_URL = r'((http://www.vevo.com/watch/.*?/.*?/)|(vevo:))(?P<id>.*?)(\?|$)' | ||||
|     _TEST = { | ||||
|     _TESTS = [{ | ||||
|         u'url': u'http://www.vevo.com/watch/hurts/somebody-to-die-for/GB1101300280', | ||||
|         u'file': u'GB1101300280.mp4', | ||||
|         u"md5": u"06bea460acb744eab74a9d7dcb4bfd61", | ||||
|         u'info_dict': { | ||||
|             u"upload_date": u"20130624", | ||||
|             u"uploader": u"Hurts", | ||||
|             u"title": u"Somebody to Die For", | ||||
|             u'duration': 230, | ||||
|             u"duration": 230, | ||||
|             u"width": 1920, | ||||
|             u"height": 1080, | ||||
|         } | ||||
|     } | ||||
|     }] | ||||
|     _SMIL_BASE_URL = 'http://smil.lvl3.vevo.com/' | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         json_url = 'http://videoplayer.vevo.com/VideoService/AuthenticateVideo?isrc=%s' % video_id | ||||
|         info_json = self._download_webpage(json_url, video_id, u'Downloading json info') | ||||
|  | ||||
|         self.report_extraction(video_id) | ||||
|         video_info = json.loads(info_json)['video'] | ||||
|     def _formats_from_json(self, video_info): | ||||
|         last_version = {'version': -1} | ||||
|         for version in video_info['videoVersions']: | ||||
|             # These are the HTTP downloads, other types are for different manifests | ||||
| @@ -50,17 +46,75 @@ class VevoIE(InfoExtractor): | ||||
|         # Already sorted from worst to best quality | ||||
|         for rend in renditions.findall('rendition'): | ||||
|             attr = rend.attrib | ||||
|             f_url = attr['url'] | ||||
|             format_note = '%(videoCodec)s@%(videoBitrate)4sk, %(audioCodec)s@%(audioBitrate)3sk' % attr | ||||
|             formats.append({ | ||||
|                 'url': f_url, | ||||
|                 'ext': determine_ext(f_url), | ||||
|                 'url': attr['url'], | ||||
|                 'format_id': attr['name'], | ||||
|                 'format_note': format_note, | ||||
|                 'height': int(attr['frameheight']), | ||||
|                 'width': int(attr['frameWidth']), | ||||
|             }) | ||||
|         return formats | ||||
|  | ||||
|         date_epoch = int(self._search_regex( | ||||
|             r'/Date\((\d+)\)/', video_info['launchDate'], u'launch date'))/1000 | ||||
|         upload_date = datetime.datetime.fromtimestamp(date_epoch) | ||||
|     def _formats_from_smil(self, smil_xml): | ||||
|         formats = [] | ||||
|         smil_doc = xml.etree.ElementTree.fromstring(smil_xml.encode('utf-8')) | ||||
|         els = smil_doc.findall('.//{http://www.w3.org/2001/SMIL20/Language}video') | ||||
|         for el in els: | ||||
|             src = el.attrib['src'] | ||||
|             m = re.match(r'''(?xi) | ||||
|                 (?P<ext>[a-z0-9]+): | ||||
|                 (?P<path> | ||||
|                     [/a-z0-9]+     # The directory and main part of the URL | ||||
|                     _(?P<cbr>[0-9]+)k | ||||
|                     _(?P<width>[0-9]+)x(?P<height>[0-9]+) | ||||
|                     _(?P<vcodec>[a-z0-9]+) | ||||
|                     _(?P<vbr>[0-9]+) | ||||
|                     _(?P<acodec>[a-z0-9]+) | ||||
|                     _(?P<abr>[0-9]+) | ||||
|                     \.[a-z0-9]+  # File extension | ||||
|                 )''', src) | ||||
|             if not m: | ||||
|                 continue | ||||
|  | ||||
|             format_url = self._SMIL_BASE_URL + m.group('path') | ||||
|             formats.append({ | ||||
|                 'url': format_url, | ||||
|                 'format_id': u'SMIL_' + m.group('cbr'), | ||||
|                 'vcodec': m.group('vcodec'), | ||||
|                 'acodec': m.group('acodec'), | ||||
|                 'vbr': int(m.group('vbr')), | ||||
|                 'abr': int(m.group('abr')), | ||||
|                 'ext': m.group('ext'), | ||||
|                 'width': int(m.group('width')), | ||||
|                 'height': int(m.group('height')), | ||||
|             }) | ||||
|         return formats | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|  | ||||
|         json_url = 'http://videoplayer.vevo.com/VideoService/AuthenticateVideo?isrc=%s' % video_id | ||||
|         info_json = self._download_webpage(json_url, video_id, u'Downloading json info') | ||||
|         video_info = json.loads(info_json)['video'] | ||||
|  | ||||
|         formats = self._formats_from_json(video_info) | ||||
|         try: | ||||
|             smil_url = '%s/Video/V2/VFILE/%s/%sr.smil' % ( | ||||
|                 self._SMIL_BASE_URL, video_id, video_id.lower()) | ||||
|             smil_xml = self._download_webpage(smil_url, video_id, | ||||
|                                               u'Downloading SMIL info') | ||||
|             formats.extend(self._formats_from_smil(smil_xml)) | ||||
|         except ExtractorError as ee: | ||||
|             if not isinstance(ee.cause, compat_HTTPError): | ||||
|                 raise | ||||
|             self._downloader.report_warning( | ||||
|                 u'Cannot download SMIL information, falling back to JSON ..') | ||||
|  | ||||
|         timestamp_ms = int(self._search_regex( | ||||
|             r'/Date\((\d+)\)/', video_info['launchDate'], u'launch date')) | ||||
|         upload_date = datetime.datetime.fromtimestamp(timestamp_ms // 1000) | ||||
|         info = { | ||||
|             'id': video_id, | ||||
|             'title': video_info['title'], | ||||
| @@ -71,7 +125,4 @@ class VevoIE(InfoExtractor): | ||||
|             'duration': video_info['duration'], | ||||
|         } | ||||
|  | ||||
|         # TODO: Remove when #980 has been merged | ||||
|         info.update(formats[-1]) | ||||
|  | ||||
|         return info | ||||
|   | ||||
| @@ -8,7 +8,7 @@ from ..utils import ( | ||||
|  | ||||
|  | ||||
| class ViddlerIE(InfoExtractor): | ||||
|     _VALID_URL = r'(?P<domain>https?://(?:www\.)?viddler.com)/(?:v|embed|player)/(?P<id>[0-9]+)' | ||||
|     _VALID_URL = r'(?P<domain>https?://(?:www\.)?viddler.com)/(?:v|embed|player)/(?P<id>[a-z0-9]+)' | ||||
|     _TEST = { | ||||
|         u"url": u"http://www.viddler.com/v/43903784", | ||||
|         u'file': u'43903784.mp4', | ||||
|   | ||||
| @@ -16,7 +16,7 @@ class VideoDetectiveIE(InfoExtractor): | ||||
|         u'info_dict': { | ||||
|             u'title': u'KICK-ASS 2', | ||||
|             u'description': u'md5:65ba37ad619165afac7d432eaded6013', | ||||
|             u'duration': 138, | ||||
|             u'duration': 135, | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| # encoding: utf-8 | ||||
| import json | ||||
| import re | ||||
| import itertools | ||||
| @@ -10,6 +11,7 @@ from ..utils import ( | ||||
|     clean_html, | ||||
|     get_element_by_attribute, | ||||
|     ExtractorError, | ||||
|     RegexNotFoundError, | ||||
|     std_headers, | ||||
|     unsmuggle_url, | ||||
| ) | ||||
| @@ -18,12 +20,12 @@ class VimeoIE(InfoExtractor): | ||||
|     """Information extractor for vimeo.com.""" | ||||
|  | ||||
|     # _VALID_URL matches Vimeo URLs | ||||
|     _VALID_URL = r'(?P<proto>https?://)?(?:(?:www|player)\.)?vimeo(?P<pro>pro)?\.com/(?:(?:(?:groups|album)/[^/]+)|(?:.*?)/)?(?P<direct_link>play_redirect_hls\?clip_id=)?(?:videos?/)?(?P<id>[0-9]+)/?(?:[?].*)?$' | ||||
|     _VALID_URL = r'(?P<proto>https?://)?(?:(?:www|(?P<player>player))\.)?vimeo(?P<pro>pro)?\.com/(?:(?:(?:groups|album)/[^/]+)|(?:.*?)/)?(?P<direct_link>play_redirect_hls\?clip_id=)?(?:videos?/)?(?P<id>[0-9]+)/?(?:[?].*)?(?:#.*)?$' | ||||
|     _NETRC_MACHINE = 'vimeo' | ||||
|     IE_NAME = u'vimeo' | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             u'url': u'http://vimeo.com/56015672', | ||||
|             u'url': u'http://vimeo.com/56015672#at=0', | ||||
|             u'file': u'56015672.mp4', | ||||
|             u'md5': u'8879b6cc097e987f02484baf890129e5', | ||||
|             u'info_dict': { | ||||
| @@ -54,7 +56,22 @@ class VimeoIE(InfoExtractor): | ||||
|                 u'title': u'Kathy Sierra: Building the minimum Badass User, Business of Software', | ||||
|                 u'uploader': u'The BLN & Business of Software', | ||||
|             }, | ||||
|         } | ||||
|         }, | ||||
|         { | ||||
|             u'url': u'http://vimeo.com/68375962', | ||||
|             u'file': u'68375962.mp4', | ||||
|             u'md5': u'aaf896bdb7ddd6476df50007a0ac0ae7', | ||||
|             u'note': u'Video protected with password', | ||||
|             u'info_dict': { | ||||
|                 u'title': u'youtube-dl password protected test video', | ||||
|                 u'upload_date': u'20130614', | ||||
|                 u'uploader_id': u'user18948128', | ||||
|                 u'uploader': u'Jaime Marquínez Ferrándiz', | ||||
|             }, | ||||
|             u'params': { | ||||
|                 u'videopassword': u'youtube-dl', | ||||
|             }, | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     def _login(self): | ||||
| @@ -111,11 +128,9 @@ class VimeoIE(InfoExtractor): | ||||
|             raise ExtractorError(u'Invalid URL: %s' % url) | ||||
|  | ||||
|         video_id = mobj.group('id') | ||||
|         if not mobj.group('proto'): | ||||
|             url = 'https://' + url | ||||
|         elif mobj.group('pro'): | ||||
|         if mobj.group('pro') or mobj.group('player'): | ||||
|             url = 'http://player.vimeo.com/video/' + video_id | ||||
|         elif mobj.group('direct_link'): | ||||
|         else: | ||||
|             url = 'https://vimeo.com/' + video_id | ||||
|  | ||||
|         # Retrieve video webpage to extract further information | ||||
| @@ -129,18 +144,26 @@ class VimeoIE(InfoExtractor): | ||||
|  | ||||
|         # Extract the config JSON | ||||
|         try: | ||||
|             config = self._search_regex([r' = {config:({.+?}),assets:', r'c=({.+?);'], | ||||
|                 webpage, u'info section', flags=re.DOTALL) | ||||
|             config = json.loads(config) | ||||
|         except: | ||||
|             try: | ||||
|                 config_url = self._html_search_regex( | ||||
|                     r' data-config-url="(.+?)"', webpage, u'config URL') | ||||
|                 config_json = self._download_webpage(config_url, video_id) | ||||
|                 config = json.loads(config_json) | ||||
|             except RegexNotFoundError: | ||||
|                 # For pro videos or player.vimeo.com urls | ||||
|                 config = self._search_regex([r' = {config:({.+?}),assets:', r'c=({.+?);'], | ||||
|                     webpage, u'info section', flags=re.DOTALL) | ||||
|                 config = json.loads(config) | ||||
|         except Exception as e: | ||||
|             if re.search('The creator of this video has not given you permission to embed it on this domain.', webpage): | ||||
|                 raise ExtractorError(u'The author has restricted the access to this video, try with the "--referer" option') | ||||
|  | ||||
|             if re.search('If so please provide the correct password.', webpage): | ||||
|             if re.search('<form[^>]+?id="pw_form"', webpage) is not None: | ||||
|                 self._verify_video_password(url, video_id, webpage) | ||||
|                 return self._real_extract(url) | ||||
|             else: | ||||
|                 raise ExtractorError(u'Unable to extract info section') | ||||
|                 raise ExtractorError(u'Unable to extract info section', | ||||
|                                      cause=e) | ||||
|  | ||||
|         # Extract title | ||||
|         video_title = config["video"]["title"] | ||||
| @@ -179,47 +202,47 @@ class VimeoIE(InfoExtractor): | ||||
|  | ||||
|         # Vimeo specific: extract video codec and quality information | ||||
|         # First consider quality, then codecs, then take everything | ||||
|         # TODO bind to format param | ||||
|         codecs = [('h264', 'mp4'), ('vp8', 'flv'), ('vp6', 'flv')] | ||||
|         files = { 'hd': [], 'sd': [], 'other': []} | ||||
|         codecs = [('vp6', 'flv'), ('vp8', 'flv'), ('h264', 'mp4')] | ||||
|         files = {'hd': [], 'sd': [], 'other': []} | ||||
|         config_files = config["video"].get("files") or config["request"].get("files") | ||||
|         for codec_name, codec_extension in codecs: | ||||
|             if codec_name in config_files: | ||||
|                 if 'hd' in config_files[codec_name]: | ||||
|                     files['hd'].append((codec_name, codec_extension, 'hd')) | ||||
|                 elif 'sd' in config_files[codec_name]: | ||||
|                     files['sd'].append((codec_name, codec_extension, 'sd')) | ||||
|             for quality in config_files.get(codec_name, []): | ||||
|                 format_id = '-'.join((codec_name, quality)).lower() | ||||
|                 key = quality if quality in files else 'other' | ||||
|                 video_url = None | ||||
|                 if isinstance(config_files[codec_name], dict): | ||||
|                     file_info = config_files[codec_name][quality] | ||||
|                     video_url = file_info.get('url') | ||||
|                 else: | ||||
|                     files['other'].append((codec_name, codec_extension, config_files[codec_name][0])) | ||||
|                     file_info = {} | ||||
|                 if video_url is None: | ||||
|                     video_url = "http://player.vimeo.com/play_redirect?clip_id=%s&sig=%s&time=%s&quality=%s&codecs=%s&type=moogaloop_local&embed_location=" \ | ||||
|                         %(video_id, sig, timestamp, quality, codec_name.upper()) | ||||
|  | ||||
|         for quality in ('hd', 'sd', 'other'): | ||||
|             if len(files[quality]) > 0: | ||||
|                 video_quality = files[quality][0][2] | ||||
|                 video_codec = files[quality][0][0] | ||||
|                 video_extension = files[quality][0][1] | ||||
|                 self.to_screen(u'%s: Downloading %s file at %s quality' % (video_id, video_codec.upper(), video_quality)) | ||||
|                 break | ||||
|         else: | ||||
|                 files[key].append({ | ||||
|                     'ext': codec_extension, | ||||
|                     'url': video_url, | ||||
|                     'format_id': format_id, | ||||
|                     'width': file_info.get('width'), | ||||
|                     'height': file_info.get('height'), | ||||
|                 }) | ||||
|         formats = [] | ||||
|         for key in ('other', 'sd', 'hd'): | ||||
|             formats += files[key] | ||||
|         if len(formats) == 0: | ||||
|             raise ExtractorError(u'No known codec found') | ||||
|  | ||||
|         video_url = None | ||||
|         if isinstance(config_files[video_codec], dict): | ||||
|             video_url = config_files[video_codec][video_quality].get("url") | ||||
|         if video_url is None: | ||||
|             video_url = "http://player.vimeo.com/play_redirect?clip_id=%s&sig=%s&time=%s&quality=%s&codecs=%s&type=moogaloop_local&embed_location=" \ | ||||
|                         %(video_id, sig, timestamp, video_quality, video_codec.upper()) | ||||
|  | ||||
|         return [{ | ||||
|         return { | ||||
|             'id':       video_id, | ||||
|             'url':      video_url, | ||||
|             'uploader': video_uploader, | ||||
|             'uploader_id': video_uploader_id, | ||||
|             'upload_date':  video_upload_date, | ||||
|             'title':    video_title, | ||||
|             'ext':      video_extension, | ||||
|             'thumbnail':    video_thumbnail, | ||||
|             'description':  video_description, | ||||
|         }] | ||||
|             'formats': formats, | ||||
|             'webpage_url': url, | ||||
|         } | ||||
|  | ||||
|  | ||||
| class VimeoChannelIE(InfoExtractor): | ||||
|   | ||||
| @@ -27,7 +27,7 @@ class VineIE(InfoExtractor): | ||||
|         video_url = self._html_search_regex(r'<meta property="twitter:player:stream" content="(.+?)"', | ||||
|             webpage, u'video URL') | ||||
|  | ||||
|         uploader = self._html_search_regex(r'<div class="user">.*?<h2>(.+?)</h2>', | ||||
|         uploader = self._html_search_regex(r'<p class="username">(.*?)</p>', | ||||
|             webpage, u'uploader', fatal=False, flags=re.DOTALL) | ||||
|  | ||||
|         return [{ | ||||
|   | ||||
							
								
								
									
										45
									
								
								youtube_dl/extractor/vk.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								youtube_dl/extractor/vk.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| # encoding: utf-8 | ||||
| import re | ||||
| import json | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_str, | ||||
|     unescapeHTML, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class VKIE(InfoExtractor): | ||||
|     IE_NAME = u'vk.com' | ||||
|     _VALID_URL = r'https?://vk\.com/(?:videos.*?\?.*?z=)?video(?P<id>.*?)(?:\?|%2F|$)' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'url': u'http://vk.com/videos-77521?z=video-77521_162222515%2Fclub77521', | ||||
|         u'md5': u'0deae91935c54e00003c2a00646315f0', | ||||
|         u'info_dict': { | ||||
|             u'id': u'162222515', | ||||
|             u'ext': u'flv', | ||||
|             u'title': u'ProtivoGunz - Хуёвая песня', | ||||
|             u'uploader': u'Noize MC', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         info_url = 'http://vk.com/al_video.php?act=show&al=1&video=%s' % video_id | ||||
|         info_page = self._download_webpage(info_url, video_id) | ||||
|         m_yt = re.search(r'src="(http://www.youtube.com/.*?)"', info_page) | ||||
|         if m_yt is not None: | ||||
|             self.to_screen(u'Youtube video detected') | ||||
|             return self.url_result(m_yt.group(1), 'Youtube') | ||||
|         vars_json = self._search_regex(r'var vars = ({.*?});', info_page, u'vars') | ||||
|         vars = json.loads(vars_json) | ||||
|  | ||||
|         return { | ||||
|             'id': compat_str(vars['vid']), | ||||
|             'url': vars['url240'], | ||||
|             'title': unescapeHTML(vars['md_title']), | ||||
|             'thumbnail': vars['jpg'], | ||||
|             'uploader': vars['md_author'], | ||||
|         } | ||||
| @@ -13,6 +13,7 @@ class WeiboIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://video\.weibo\.com/v/weishipin/t_(?P<id>.+?)\.htm' | ||||
|  | ||||
|     _TEST = { | ||||
|         u'add_ie': ['Sina'], | ||||
|         u'url': u'http://video.weibo.com/v/weishipin/t_zjUw2kZ.htm', | ||||
|         u'file': u'98322879.flv', | ||||
|         u'info_dict': { | ||||
|   | ||||
| @@ -19,7 +19,8 @@ class XHamsterIE(InfoExtractor): | ||||
|         u'info_dict': { | ||||
|             u"upload_date": u"20121014",  | ||||
|             u"uploader_id": u"Ruseful2011",  | ||||
|             u"title": u"FemaleAgent Shy beauty takes the bait" | ||||
|             u"title": u"FemaleAgent Shy beauty takes the bait", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     }, | ||||
|     { | ||||
| @@ -27,28 +28,33 @@ class XHamsterIE(InfoExtractor): | ||||
|         u'file': u'2221348.flv', | ||||
|         u'md5': u'e767b9475de189320f691f49c679c4c7', | ||||
|         u'info_dict': { | ||||
|             u"upload_date": u"20130914",  | ||||
|             u"uploader_id": u"jojo747400",  | ||||
|             u"title": u"Britney Spears  Sexy Booty" | ||||
|             u"upload_date": u"20130914", | ||||
|             u"uploader_id": u"jojo747400", | ||||
|             u"title": u"Britney Spears  Sexy Booty", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self,url): | ||||
|         def extract_video_url(webpage): | ||||
|             mobj = re.search(r'\'srv\': \'(?P<server>[^\']*)\',\s*\'file\': \'(?P<file>[^\']+)\',', webpage) | ||||
|             if mobj is None: | ||||
|                 raise ExtractorError(u'Unable to extract media URL') | ||||
|             if len(mobj.group('server')) == 0: | ||||
|                 return compat_urllib_parse.unquote(mobj.group('file')) | ||||
|             else: | ||||
|                 return mobj.group('server')+'/key='+mobj.group('file') | ||||
|  | ||||
|         def is_hd(webpage): | ||||
|             return webpage.find('<div class=\'icon iconHD\'>') != -1 | ||||
|  | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|  | ||||
|         video_id = mobj.group('id') | ||||
|         seo = mobj.group('seo') | ||||
|         mrss_url = 'http://xhamster.com/movies/%s/%s.html?hd' % (video_id, seo) | ||||
|         mrss_url = 'http://xhamster.com/movies/%s/%s.html' % (video_id, seo) | ||||
|         webpage = self._download_webpage(mrss_url, video_id) | ||||
|  | ||||
|         mobj = re.search(r'\'srv\': \'(?P<server>[^\']*)\',\s*\'file\': \'(?P<file>[^\']+)\',', webpage) | ||||
|         if mobj is None: | ||||
|             raise ExtractorError(u'Unable to extract media URL') | ||||
|         if len(mobj.group('server')) == 0: | ||||
|             video_url = compat_urllib_parse.unquote(mobj.group('file')) | ||||
|         else: | ||||
|             video_url = mobj.group('server')+'/key='+mobj.group('file') | ||||
|  | ||||
|         video_title = self._html_search_regex(r'<title>(?P<title>.+?) - xHamster\.com</title>', | ||||
|             webpage, u'title') | ||||
|  | ||||
| @@ -72,13 +78,34 @@ class XHamsterIE(InfoExtractor): | ||||
|         video_thumbnail = self._search_regex(r'\'image\':\'(?P<thumbnail>[^\']+)\'', | ||||
|             webpage, u'thumbnail', fatal=False) | ||||
|  | ||||
|         return [{ | ||||
|             'id':       video_id, | ||||
|             'url':      video_url, | ||||
|             'ext':      determine_ext(video_url), | ||||
|             'title':    video_title, | ||||
|         age_limit = self._rta_search(webpage) | ||||
|  | ||||
|         video_url = extract_video_url(webpage) | ||||
|         hd = is_hd(webpage) | ||||
|         formats = [{ | ||||
|             'url': video_url, | ||||
|             'ext': determine_ext(video_url), | ||||
|             'format': 'hd' if hd else 'sd', | ||||
|             'format_id': 'hd' if hd else 'sd', | ||||
|         }] | ||||
|         if not hd: | ||||
|             webpage = self._download_webpage(mrss_url+'?hd', video_id) | ||||
|             if is_hd(webpage): | ||||
|                 video_url = extract_video_url(webpage) | ||||
|                 formats.append({ | ||||
|                     'url': video_url, | ||||
|                     'ext': determine_ext(video_url), | ||||
|                     'format': 'hd', | ||||
|                     'format_id': 'hd', | ||||
|                 }) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': video_title, | ||||
|             'formats': formats, | ||||
|             'description': video_description, | ||||
|             'upload_date': video_upload_date, | ||||
|             'uploader_id': video_uploader_id, | ||||
|             'thumbnail': video_thumbnail | ||||
|         }] | ||||
|             'thumbnail': video_thumbnail, | ||||
|             'age_limit': age_limit, | ||||
|         } | ||||
|   | ||||
| @@ -9,7 +9,7 @@ from ..utils import ( | ||||
|  | ||||
|  | ||||
| class XNXXIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://)?video\.xnxx\.com/video([0-9]+)/(.*)' | ||||
|     _VALID_URL = r'^(?:https?://)?(?:video|www)\.xnxx\.com/video([0-9]+)/(.*)' | ||||
|     VIDEO_URL_RE = r'flv_url=(.*?)&' | ||||
|     VIDEO_TITLE_RE = r'<title>(.*?)\s+-\s+XNXX.COM' | ||||
|     VIDEO_THUMB_RE = r'url_bigthumb=(.*?)&' | ||||
| @@ -18,7 +18,8 @@ class XNXXIE(InfoExtractor): | ||||
|         u'file': u'1135332.flv', | ||||
|         u'md5': u'0831677e2b4761795f68d417e0b7b445', | ||||
|         u'info_dict': { | ||||
|             u"title": u"lida \u00bb Naked Funny Actress  (5)" | ||||
|             u"title": u"lida \u00bb Naked Funny Actress  (5)", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -50,4 +51,5 @@ class XNXXIE(InfoExtractor): | ||||
|             'ext': 'flv', | ||||
|             'thumbnail': video_thumbnail, | ||||
|             'description': None, | ||||
|             'age_limit': 18, | ||||
|         }] | ||||
|   | ||||
							
								
								
									
										55
									
								
								youtube_dl/extractor/xtube.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								youtube_dl/extractor/xtube.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| import os | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     compat_urllib_parse_urlparse, | ||||
|     compat_urllib_request, | ||||
|     compat_urllib_parse, | ||||
| ) | ||||
|  | ||||
| class XTubeIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?(?P<url>xtube\.com/watch\.php\?v=(?P<videoid>[^/?&]+))' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.xtube.com/watch.php?v=kVTUy_G222_', | ||||
|         u'file': u'kVTUy_G222_.mp4', | ||||
|         u'md5': u'092fbdd3cbe292c920ef6fc6a8a9cdab', | ||||
|         u'info_dict': { | ||||
|             u"title": u"strange erotica", | ||||
|             u"description": u"surreal gay themed erotica...almost an ET kind of thing", | ||||
|             u"uploader": u"greenshowers", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('videoid') | ||||
|         url = 'http://www.' + mobj.group('url') | ||||
|  | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('Cookie', 'age_verified=1') | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         video_title = self._html_search_regex(r'<div class="p_5px[^>]*>([^<]+)', webpage, u'title') | ||||
|         video_uploader = self._html_search_regex(r'so_s\.addVariable\("owner_u", "([^"]+)', webpage, u'uploader', fatal=False) | ||||
|         video_description = self._html_search_regex(r'<p class="video_description">([^<]+)', webpage, u'description', default=None) | ||||
|         video_url= self._html_search_regex(r'var videoMp4 = "([^"]+)', webpage, u'video_url').replace('\\/', '/') | ||||
|         path = compat_urllib_parse_urlparse(video_url).path | ||||
|         extension = os.path.splitext(path)[1][1:] | ||||
|         format = path.split('/')[5].split('_')[:2] | ||||
|         format[0] += 'p' | ||||
|         format[1] += 'k' | ||||
|         format = "-".join(format) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': video_title, | ||||
|             'uploader': video_uploader, | ||||
|             'description': video_description, | ||||
|             'url': video_url, | ||||
|             'ext': extension, | ||||
|             'format': format, | ||||
|             'format_id': format, | ||||
|             'age_limit': 18, | ||||
|         } | ||||
| @@ -13,7 +13,8 @@ class XVideosIE(InfoExtractor): | ||||
|         u'file': u'939581.flv', | ||||
|         u'md5': u'1d0c835822f0a71a7bf011855db929d0', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Funny Porns By >>>>S<<<<<< -1" | ||||
|             u"title": u"Funny Porns By >>>>S<<<<<< -1", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -46,6 +47,7 @@ class XVideosIE(InfoExtractor): | ||||
|             'ext': 'flv', | ||||
|             'thumbnail': video_thumbnail, | ||||
|             'description': None, | ||||
|             'age_limit': 18, | ||||
|         } | ||||
|  | ||||
|         return [info] | ||||
|   | ||||
| @@ -132,7 +132,7 @@ class YahooSearchIE(SearchInfoExtractor): | ||||
|                 mobj = re.search(r'(?P<url>screen\.yahoo\.com/.*?-\d*?\.html)"', r) | ||||
|                 e = self.url_result('http://' + mobj.group('url'), 'Yahoo') | ||||
|                 res['entries'].append(e) | ||||
|             if (pagenum * 30 +i >= n) or (m[u'last'] >= (m[u'total'] -1 )): | ||||
|             if (pagenum * 30 +i >= n) or (m[u'last'] >= (m[u'total'] -1)): | ||||
|                 break | ||||
|  | ||||
|         return res | ||||
|   | ||||
| @@ -13,7 +13,8 @@ class YouJizzIE(InfoExtractor): | ||||
|         u'file': u'2189178.flv', | ||||
|         u'md5': u'07e15fa469ba384c7693fd246905547c', | ||||
|         u'info_dict': { | ||||
|             u"title": u"Zeichentrick 1" | ||||
|             u"title": u"Zeichentrick 1", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -25,6 +26,8 @@ class YouJizzIE(InfoExtractor): | ||||
|         # Get webpage content | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         age_limit = self._rta_search(webpage) | ||||
|  | ||||
|         # Get the video title | ||||
|         video_title = self._html_search_regex(r'<title>(?P<title>.*)</title>', | ||||
|             webpage, u'title').strip() | ||||
| @@ -60,6 +63,7 @@ class YouJizzIE(InfoExtractor): | ||||
|                 'title': video_title, | ||||
|                 'ext': 'flv', | ||||
|                 'format': 'flv', | ||||
|                 'player_url': embed_page_url} | ||||
|                 'player_url': embed_page_url, | ||||
|                 'age_limit': age_limit} | ||||
|  | ||||
|         return [info] | ||||
|   | ||||
| @@ -18,7 +18,7 @@ class YoukuIE(InfoExtractor): | ||||
|         u"url": u"http://v.youku.com/v_show/id_XNDgyMDQ2NTQw.html", | ||||
|         u"file": u"XNDgyMDQ2NTQw_part00.flv", | ||||
|         u"md5": u"ffe3f2e435663dc2d1eea34faeff5b5b", | ||||
|         u"params": { u"test": False }, | ||||
|         u"params": {u"test": False}, | ||||
|         u"info_dict": { | ||||
|             u"title": u"youtube-dl test video \"'/\\ä↭𝕐" | ||||
|         } | ||||
| @@ -37,8 +37,8 @@ class YoukuIE(InfoExtractor): | ||||
|         source = list("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/\:._-1234567890") | ||||
|         seed = float(seed) | ||||
|         for i in range(len(source)): | ||||
|             seed  =  (seed * 211 + 30031 ) % 65536 | ||||
|             index  =  math.floor(seed / 65536 * len(source) ) | ||||
|             seed  =  (seed * 211 + 30031) % 65536 | ||||
|             index  =  math.floor(seed / 65536 * len(source)) | ||||
|             mixed.append(source[int(index)]) | ||||
|             source.remove(source[int(index)]) | ||||
|         #return ''.join(mixed) | ||||
|   | ||||
| @@ -17,7 +17,7 @@ from ..aes import ( | ||||
| ) | ||||
|  | ||||
| class YouPornIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://)?(?:\w+\.)?youporn\.com/watch/(?P<videoid>[0-9]+)/(?P<title>[^/]+)' | ||||
|     _VALID_URL = r'^(?:https?://)?(?:www\.)?(?P<url>youporn\.com/watch/(?P<videoid>[0-9]+)/(?P<title>[^/]+))' | ||||
|     _TEST = { | ||||
|         u'url': u'http://www.youporn.com/watch/505835/sex-ed-is-it-safe-to-masturbate-daily/', | ||||
|         u'file': u'505835.mp4', | ||||
| @@ -26,27 +26,15 @@ class YouPornIE(InfoExtractor): | ||||
|             u"upload_date": u"20101221",  | ||||
|             u"description": u"Love & Sex Answers: http://bit.ly/DanAndJenn -- Is It Unhealthy To Masturbate Daily?",  | ||||
|             u"uploader": u"Ask Dan And Jennifer",  | ||||
|             u"title": u"Sex Ed: Is It Safe To Masturbate Daily?" | ||||
|             u"title": u"Sex Ed: Is It Safe To Masturbate Daily?", | ||||
|             u"age_limit": 18, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _print_formats(self, formats): | ||||
|         """Print all available formats""" | ||||
|         print(u'Available formats:') | ||||
|         print(u'ext\t\tformat') | ||||
|         print(u'---------------------------------') | ||||
|         for format in formats: | ||||
|             print(u'%s\t\t%s'  % (format['ext'], format['format'])) | ||||
|  | ||||
|     def _specific(self, req_format, formats): | ||||
|         for x in formats: | ||||
|             if x["format"] == req_format: | ||||
|                 return x | ||||
|         return None | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('videoid') | ||||
|         url = 'http://www.' + mobj.group('url') | ||||
|  | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('Cookie', 'age_verified=1') | ||||
| @@ -70,27 +58,22 @@ class YouPornIE(InfoExtractor): | ||||
|         except KeyError: | ||||
|             raise ExtractorError('Missing JSON parameter: ' + sys.exc_info()[1]) | ||||
|  | ||||
|         # Get all of the formats available | ||||
|         # Get all of the links from the page | ||||
|         DOWNLOAD_LIST_RE = r'(?s)<ul class="downloadList">(?P<download_list>.*?)</ul>' | ||||
|         download_list_html = self._search_regex(DOWNLOAD_LIST_RE, | ||||
|             webpage, u'download list').strip() | ||||
|  | ||||
|         # Get all of the links from the page | ||||
|         LINK_RE = r'(?s)<a href="(?P<url>[^"]+)">' | ||||
|         LINK_RE = r'<a href="([^"]+)">' | ||||
|         links = re.findall(LINK_RE, download_list_html) | ||||
|          | ||||
|         # Get link of hd video if available | ||||
|         mobj = re.search(r'var encryptedQuality720URL = \'(?P<encrypted_video_url>[a-zA-Z0-9+/]+={0,2})\';', webpage) | ||||
|         if mobj != None: | ||||
|             encrypted_video_url = mobj.group(u'encrypted_video_url') | ||||
|             video_url = aes_decrypt_text(encrypted_video_url, video_title, 32).decode('utf-8') | ||||
|             links = [video_url] + links | ||||
|  | ||||
|         # Get all encrypted links | ||||
|         encrypted_links = re.findall(r'var encryptedQuality[0-9]{3}URL = \'([a-zA-Z0-9+/]+={0,2})\';', webpage) | ||||
|         for encrypted_link in encrypted_links: | ||||
|             link = aes_decrypt_text(encrypted_link, video_title, 32).decode('utf-8') | ||||
|             links.append(link) | ||||
|          | ||||
|         if not links: | ||||
|             raise ExtractorError(u'ERROR: no known formats available for video') | ||||
|  | ||||
|         self.to_screen(u'Links found: %d' % len(links)) | ||||
|  | ||||
|         formats = [] | ||||
|         for link in links: | ||||
|  | ||||
| @@ -98,43 +81,36 @@ class YouPornIE(InfoExtractor): | ||||
|             # http://cdn1.download.youporn.phncdn.com/201210/31/8004515/480p_370k_8004515/YouPorn%20-%20Nubile%20Films%20The%20Pillow%20Fight.mp4?nvb=20121113051249&nva=20121114051249&ir=1200&sr=1200&hash=014b882080310e95fb6a0 | ||||
|             # A path looks like this: | ||||
|             # /201210/31/8004515/480p_370k_8004515/YouPorn%20-%20Nubile%20Films%20The%20Pillow%20Fight.mp4 | ||||
|             video_url = unescapeHTML( link ) | ||||
|             path = compat_urllib_parse_urlparse( video_url ).path | ||||
|             extension = os.path.splitext( path )[1][1:] | ||||
|             video_url = unescapeHTML(link) | ||||
|             path = compat_urllib_parse_urlparse(video_url).path | ||||
|             extension = os.path.splitext(path)[1][1:] | ||||
|             format = path.split('/')[4].split('_')[:2] | ||||
|  | ||||
|             # size = format[0] | ||||
|             # bitrate = format[1] | ||||
|             format = "-".join( format ) | ||||
|             format = "-".join(format) | ||||
|             # title = u'%s-%s-%s' % (video_title, size, bitrate) | ||||
|  | ||||
|             formats.append({ | ||||
|                 'id': video_id, | ||||
|                 'url': video_url, | ||||
|                 'uploader': video_uploader, | ||||
|                 'upload_date': upload_date, | ||||
|                 'title': video_title, | ||||
|                 'ext': extension, | ||||
|                 'format': format, | ||||
|                 'thumbnail': thumbnail, | ||||
|                 'description': video_description, | ||||
|                 'age_limit': age_limit, | ||||
|                 'format_id': format, | ||||
|             }) | ||||
|  | ||||
|         if self._downloader.params.get('listformats', None): | ||||
|             self._print_formats(formats) | ||||
|             return | ||||
|  | ||||
|         req_format = self._downloader.params.get('format', 'best') | ||||
|         self.to_screen(u'Format: %s' % req_format) | ||||
|  | ||||
|         if req_format is None or req_format == 'best': | ||||
|             return [formats[0]] | ||||
|         elif req_format == 'worst': | ||||
|             return [formats[-1]] | ||||
|         elif req_format in ('-1', 'all'): | ||||
|             return formats | ||||
|         else: | ||||
|             format = self._specific( req_format, formats ) | ||||
|             if format is None: | ||||
|                 raise ExtractorError(u'Requested format not available') | ||||
|             return [format] | ||||
|         # Sort and remove doubles | ||||
|         formats.sort(key=lambda format: list(map(lambda s: s.zfill(6), format['format'].split('-')))) | ||||
|         for i in range(len(formats)-1,0,-1): | ||||
|             if formats[i]['format_id'] == formats[i-1]['format_id']: | ||||
|                 del formats[i] | ||||
|          | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'uploader': video_uploader, | ||||
|             'upload_date': upload_date, | ||||
|             'title': video_title, | ||||
|             'thumbnail': thumbnail, | ||||
|             'description': video_description, | ||||
|             'age_limit': age_limit, | ||||
|             'formats': formats, | ||||
|         } | ||||
|   | ||||
| @@ -74,14 +74,8 @@ class YoutubeBaseInfoExtractor(InfoExtractor): | ||||
|             self._downloader.report_warning(u'unable to fetch login page: %s' % compat_str(err)) | ||||
|             return False | ||||
|  | ||||
|         galx = None | ||||
|         dsh = None | ||||
|         match = re.search(re.compile(r'<input.+?name="GALX".+?value="(.+?)"', re.DOTALL), login_page) | ||||
|         if match: | ||||
|           galx = match.group(1) | ||||
|         match = re.search(re.compile(r'<input.+?name="dsh".+?value="(.+?)"', re.DOTALL), login_page) | ||||
|         if match: | ||||
|           dsh = match.group(1) | ||||
|         galx = self._search_regex(r'(?s)<input.+?name="GALX".+?value="(.+?)"', | ||||
|                                   login_page, u'Login GALX parameter') | ||||
|  | ||||
|         # Log in | ||||
|         login_form_strs = { | ||||
| @@ -95,7 +89,6 @@ class YoutubeBaseInfoExtractor(InfoExtractor): | ||||
|                 u'checkConnection': u'', | ||||
|                 u'checkedDomains': u'youtube', | ||||
|                 u'dnConn': u'', | ||||
|                 u'dsh': dsh, | ||||
|                 u'pstMsg': u'0', | ||||
|                 u'rmShown': u'1', | ||||
|                 u'secTok': u'', | ||||
| @@ -146,10 +139,10 @@ class YoutubeBaseInfoExtractor(InfoExtractor): | ||||
|  | ||||
| class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|     IE_DESC = u'YouTube.com' | ||||
|     _VALID_URL = r"""^ | ||||
|     _VALID_URL = r"""(?x)^ | ||||
|                      ( | ||||
|                          (?:https?://)?                                       # http(s):// (optional) | ||||
|                          (?:(?:(?:(?:\w+\.)?youtube(?:-nocookie)?\.com/| | ||||
|                          (?:https?://|//)?                                    # http(s):// or protocol-independent URL (optional) | ||||
|                          (?:(?:(?:(?:\w+\.)?[yY][oO][uU][tT][uU][bB][eE](?:-nocookie)?\.com/| | ||||
|                             tube\.majestyc\.net/| | ||||
|                             youtube\.googleapis\.com/)                        # the various hostnames, with wildcard subdomains | ||||
|                          (?:.*?\#/)?                                          # handle anchor (#/) redirect urls | ||||
| @@ -236,11 +229,13 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|         '136': 'mp4', | ||||
|         '137': 'mp4', | ||||
|         '138': 'mp4', | ||||
|         '139': 'mp4', | ||||
|         '140': 'mp4', | ||||
|         '141': 'mp4', | ||||
|         '160': 'mp4', | ||||
|  | ||||
|         # Dash mp4 audio | ||||
|         '139': 'm4a', | ||||
|         '140': 'm4a', | ||||
|         '141': 'm4a', | ||||
|  | ||||
|         # Dash webm | ||||
|         '171': 'webm', | ||||
|         '172': 'webm', | ||||
| @@ -344,18 +339,6 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|                 u"description": u"test chars:  \"'/\\ä↭𝕐\n\nThis is a test video for youtube-dl.\n\nFor more information, contact phihag@phihag.de ." | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             u"url":  u"http://www.youtube.com/watch?v=1ltcDfZMA3U", | ||||
|             u"file":  u"1ltcDfZMA3U.flv", | ||||
|             u"note": u"Test VEVO video (#897)", | ||||
|             u"info_dict": { | ||||
|                 u"upload_date": u"20070518", | ||||
|                 u"title": u"Maps - It Will Find You", | ||||
|                 u"description": u"Music video by Maps performing It Will Find You.", | ||||
|                 u"uploader": u"MuteUSA", | ||||
|                 u"uploader_id": u"MuteUSA" | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             u"url":  u"http://www.youtube.com/watch?v=UxxajLWwzqY", | ||||
|             u"file":  u"UxxajLWwzqY.mp4", | ||||
| @@ -380,6 +363,18 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|                 u"uploader_id": u"justintimberlakeVEVO" | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             u"url":  u"//www.YouTube.com/watch?v=yZIXLfi8CZQ", | ||||
|             u"file":  u"yZIXLfi8CZQ.mp4", | ||||
|             u"note": u"Embed-only video (#1746)", | ||||
|             u"info_dict": { | ||||
|                 u"upload_date": u"20120608", | ||||
|                 u"title": u"Principal Sexually Assaults A Teacher - Episode 117 - 8th June 2012", | ||||
|                 u"description": u"md5:09b78bd971f1e3e289601dfba15ca4f7", | ||||
|                 u"uploader": u"SET India", | ||||
|                 u"uploader_id": u"setindia" | ||||
|             } | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|  | ||||
| @@ -387,7 +382,7 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|     def suitable(cls, url): | ||||
|         """Receives a URL and returns True if suitable for this IE.""" | ||||
|         if YoutubePlaylistIE.suitable(url): return False | ||||
|         return re.match(cls._VALID_URL, url, re.VERBOSE) is not None | ||||
|         return re.match(cls._VALID_URL, url) is not None | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         super(YoutubeIE, self).__init__(*args, **kwargs) | ||||
| @@ -1036,6 +1031,8 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|         """Turn the encrypted s field into a working signature""" | ||||
|  | ||||
|         if player_url is not None: | ||||
|             if player_url.startswith(u'//'): | ||||
|                 player_url = u'https:' + player_url | ||||
|             try: | ||||
|                 player_id = (player_url, len(s)) | ||||
|                 if player_id not in self._player_cache: | ||||
| @@ -1099,7 +1096,7 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|         else: | ||||
|             raise ExtractorError(u'Unable to decrypt signature, key length %d not supported; retrying might work' % (len(s))) | ||||
|  | ||||
|     def _get_available_subtitles(self, video_id): | ||||
|     def _get_available_subtitles(self, video_id, webpage): | ||||
|         try: | ||||
|             sub_list = self._download_webpage( | ||||
|                 'http://video.google.com/timedtext?hl=en&type=list&v=%s' % video_id, | ||||
| @@ -1115,8 +1112,8 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|             params = compat_urllib_parse.urlencode({ | ||||
|                 'lang': lang, | ||||
|                 'v': video_id, | ||||
|                 'fmt': self._downloader.params.get('subtitlesformat'), | ||||
|                 'name': l[0], | ||||
|                 'fmt': self._downloader.params.get('subtitlesformat', 'srt'), | ||||
|                 'name': l[0].encode('utf-8'), | ||||
|             }) | ||||
|             url = u'http://www.youtube.com/api/timedtext?' + params | ||||
|             sub_lang_list[lang] = url | ||||
| @@ -1128,7 +1125,7 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|     def _get_available_automatic_caption(self, video_id, webpage): | ||||
|         """We need the webpage for getting the captions url, pass it as an | ||||
|            argument to speed up the process.""" | ||||
|         sub_format = self._downloader.params.get('subtitlesformat') | ||||
|         sub_format = self._downloader.params.get('subtitlesformat', 'srt') | ||||
|         self.to_screen(u'%s: Looking for automatic captions' % video_id) | ||||
|         mobj = re.search(r';ytplayer.config = ({.*?});', webpage) | ||||
|         err_msg = u'Couldn\'t find automatic captions for %s' % video_id | ||||
| @@ -1150,7 +1147,7 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|             list_page = self._download_webpage(list_url, video_id) | ||||
|             caption_list = xml.etree.ElementTree.fromstring(list_page.encode('utf-8')) | ||||
|             original_lang_node = caption_list.find('track') | ||||
|             if not original_lang_node or original_lang_node.attrib.get('kind') != 'asr' : | ||||
|             if original_lang_node is None or original_lang_node.attrib.get('kind') != 'asr' : | ||||
|                 self._downloader.report_warning(u'Video doesn\'t have automatic captions') | ||||
|                 return {} | ||||
|             original_lang = original_lang_node.attrib['lang_code'] | ||||
| @@ -1287,7 +1284,7 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|             # We simulate the access to the video from www.youtube.com/v/{video_id} | ||||
|             # this can be viewed without login into Youtube | ||||
|             data = compat_urllib_parse.urlencode({'video_id': video_id, | ||||
|                                                   'el': 'embedded', | ||||
|                                                   'el': 'player_embedded', | ||||
|                                                   'gl': 'US', | ||||
|                                                   'hl': 'en', | ||||
|                                                   'eurl': 'https://youtube.googleapis.com/v/' + video_id, | ||||
| @@ -1316,6 +1313,11 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|             else: | ||||
|                 raise ExtractorError(u'"token" parameter not in video info for unknown reason') | ||||
|  | ||||
|         if 'view_count' in video_info: | ||||
|             view_count = int(video_info['view_count'][0]) | ||||
|         else: | ||||
|             view_count = None | ||||
|  | ||||
|         # Check for "rental" videos | ||||
|         if 'ypc_video_rental_bar_text' in video_info and 'author' not in video_info: | ||||
|             raise ExtractorError(u'"rental" videos not supported') | ||||
| @@ -1403,32 +1405,29 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|             # this signatures are encrypted | ||||
|             if 'url_encoded_fmt_stream_map' not in args: | ||||
|                 raise ValueError(u'No stream_map present')  # caught below | ||||
|             m_s = re.search(r'[&,]s=', args['url_encoded_fmt_stream_map']) | ||||
|             re_signature = re.compile(r'[&,]s=') | ||||
|             m_s = re_signature.search(args['url_encoded_fmt_stream_map']) | ||||
|             if m_s is not None: | ||||
|                 self.to_screen(u'%s: Encrypted signatures detected.' % video_id) | ||||
|                 video_info['url_encoded_fmt_stream_map'] = [args['url_encoded_fmt_stream_map']] | ||||
|             m_s = re.search(r'[&,]s=', args.get('adaptive_fmts', u'')) | ||||
|             m_s = re_signature.search(args.get('adaptive_fmts', u'')) | ||||
|             if m_s is not None: | ||||
|                 if 'url_encoded_fmt_stream_map' in video_info: | ||||
|                     video_info['url_encoded_fmt_stream_map'][0] += ',' + args['adaptive_fmts'] | ||||
|                 if 'adaptive_fmts' in video_info: | ||||
|                     video_info['adaptive_fmts'][0] += ',' + args['adaptive_fmts'] | ||||
|                 else: | ||||
|                     video_info['url_encoded_fmt_stream_map'] = [args['adaptive_fmts']] | ||||
|             elif 'adaptive_fmts' in video_info: | ||||
|                 if 'url_encoded_fmt_stream_map' in video_info: | ||||
|                     video_info['url_encoded_fmt_stream_map'][0] += ',' + video_info['adaptive_fmts'][0] | ||||
|                 else: | ||||
|                     video_info['url_encoded_fmt_stream_map'] = video_info['adaptive_fmts'] | ||||
|                     video_info['adaptive_fmts'] = [args['adaptive_fmts']] | ||||
|         except ValueError: | ||||
|             pass | ||||
|  | ||||
|         if 'conn' in video_info and video_info['conn'][0].startswith('rtmp'): | ||||
|             self.report_rtmp_download() | ||||
|             video_url_list = [(None, video_info['conn'][0])] | ||||
|         elif 'url_encoded_fmt_stream_map' in video_info and len(video_info['url_encoded_fmt_stream_map']) >= 1: | ||||
|             if 'rtmpe%3Dyes' in video_info['url_encoded_fmt_stream_map'][0]: | ||||
|         elif len(video_info.get('url_encoded_fmt_stream_map', [])) >= 1 or len(video_info.get('adaptive_fmts', [])) >= 1: | ||||
|             encoded_url_map = video_info.get('url_encoded_fmt_stream_map', [''])[0] + ',' + video_info.get('adaptive_fmts',[''])[0] | ||||
|             if 'rtmpe%3Dyes' in encoded_url_map: | ||||
|                 raise ExtractorError('rtmpe downloads are not supported, see https://github.com/rg3/youtube-dl/issues/343 for more information.', expected=True) | ||||
|             url_map = {} | ||||
|             for url_data_str in video_info['url_encoded_fmt_stream_map'][0].split(','): | ||||
|             for url_data_str in encoded_url_map.split(','): | ||||
|                 url_data = compat_parse_qs(url_data_str) | ||||
|                 if 'itag' in url_data and 'url' in url_data: | ||||
|                     url = url_data['url'][0] | ||||
| @@ -1481,13 +1480,13 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|             raise ExtractorError(u'no conn, hlsvp or url_encoded_fmt_stream_map information found in video info') | ||||
|  | ||||
|         results = [] | ||||
|         for format_param, video_real_url in video_url_list: | ||||
|         for itag, video_real_url in video_url_list: | ||||
|             # Extension | ||||
|             video_extension = self._video_extensions.get(format_param, 'flv') | ||||
|             video_extension = self._video_extensions.get(itag, 'flv') | ||||
|  | ||||
|             video_format = '{0} - {1}{2}'.format(format_param if format_param else video_extension, | ||||
|                                               self._video_dimensions.get(format_param, '???'), | ||||
|                                               ' ('+self._special_itags[format_param]+')' if format_param in self._special_itags else '') | ||||
|             video_format = '{0} - {1}{2}'.format(itag if itag else video_extension, | ||||
|                                               self._video_dimensions.get(itag, '???'), | ||||
|                                               ' ('+self._special_itags[itag]+')' if itag in self._special_itags else '') | ||||
|  | ||||
|             results.append({ | ||||
|                 'id':       video_id, | ||||
| @@ -1498,13 +1497,16 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|                 'title':    video_title, | ||||
|                 'ext':      video_extension, | ||||
|                 'format':   video_format, | ||||
|                 'format_id': itag, | ||||
|                 'thumbnail':    video_thumbnail, | ||||
|                 'description':  video_description, | ||||
|                 'player_url':   player_url, | ||||
|                 'subtitles':    video_subtitles, | ||||
|                 'duration':     video_duration, | ||||
|                 'age_limit':    18 if age_gate else 0, | ||||
|                 'annotations':  video_annotations | ||||
|                 'annotations':  video_annotations, | ||||
|                 'webpage_url': 'https://www.youtube.com/watch?v=%s' % video_id, | ||||
|                 'view_count': view_count, | ||||
|             }) | ||||
|         return results | ||||
|  | ||||
| @@ -1590,7 +1592,6 @@ class YoutubePlaylistIE(InfoExtractor): | ||||
| class YoutubeChannelIE(InfoExtractor): | ||||
|     IE_DESC = u'YouTube.com channels' | ||||
|     _VALID_URL = r"^(?:https?://)?(?:youtu\.be|(?:\w+\.)?youtube(?:-nocookie)?\.com)/channel/([0-9A-Za-z_-]+)" | ||||
|     _TEMPLATE_URL = 'http://www.youtube.com/channel/%s/videos?sort=da&flow=list&view=0&page=%s&gl=US&hl=en' | ||||
|     _MORE_PAGES_INDICATOR = 'yt-uix-load-more' | ||||
|     _MORE_PAGES_URL = 'http://www.youtube.com/c4_browse_ajax?action_load_more_videos=1&flow=list&paging=%s&view=0&sort=da&channel_id=%s' | ||||
|     IE_NAME = u'youtube:channel' | ||||
| @@ -1611,29 +1612,30 @@ class YoutubeChannelIE(InfoExtractor): | ||||
|         # Download channel page | ||||
|         channel_id = mobj.group(1) | ||||
|         video_ids = [] | ||||
|         pagenum = 1 | ||||
|         url = 'https://www.youtube.com/channel/%s/videos' % channel_id | ||||
|         channel_page = self._download_webpage(url, channel_id) | ||||
|         if re.search(r'channel-header-autogenerated-label', channel_page) is not None: | ||||
|             autogenerated = True | ||||
|         else: | ||||
|             autogenerated = False | ||||
|  | ||||
|         url = self._TEMPLATE_URL % (channel_id, pagenum) | ||||
|         page = self._download_webpage(url, channel_id, | ||||
|                                       u'Downloading page #%s' % pagenum) | ||||
|  | ||||
|         # Extract video identifiers | ||||
|         ids_in_page = self.extract_videos_from_page(page) | ||||
|         video_ids.extend(ids_in_page) | ||||
|  | ||||
|         # Download any subsequent channel pages using the json-based channel_ajax query | ||||
|         if self._MORE_PAGES_INDICATOR in page: | ||||
|         if autogenerated: | ||||
|             # The videos are contained in a single page | ||||
|             # the ajax pages can't be used, they are empty | ||||
|             video_ids = self.extract_videos_from_page(channel_page) | ||||
|         else: | ||||
|             # Download all channel pages using the json-based channel_ajax query | ||||
|             for pagenum in itertools.count(1): | ||||
|                 url = self._MORE_PAGES_URL % (pagenum, channel_id) | ||||
|                 page = self._download_webpage(url, channel_id, | ||||
|                                               u'Downloading page #%s' % pagenum) | ||||
|  | ||||
|      | ||||
|                 page = json.loads(page) | ||||
|  | ||||
|      | ||||
|                 ids_in_page = self.extract_videos_from_page(page['content_html']) | ||||
|                 video_ids.extend(ids_in_page) | ||||
|  | ||||
|                 if self._MORE_PAGES_INDICATOR  not in page['load_more_widget_html']: | ||||
|      | ||||
|                 if self._MORE_PAGES_INDICATOR not in page['load_more_widget_html']: | ||||
|                     break | ||||
|  | ||||
|         self._downloader.to_screen(u'[youtube] Channel %s: Found %i videos' % (channel_id, len(video_ids))) | ||||
| @@ -1750,6 +1752,10 @@ class YoutubeSearchIE(SearchInfoExtractor): | ||||
|         videos = [self.url_result('http://www.youtube.com/watch?v=%s' % id, 'Youtube') for id in video_ids] | ||||
|         return self.playlist_result(videos, query) | ||||
|  | ||||
| class YoutubeSearchDateIE(YoutubeSearchIE): | ||||
|     _API_URL = 'https://gdata.youtube.com/feeds/api/videos?q=%s&start-index=%i&max-results=50&v=2&alt=jsonc&orderby=published' | ||||
|     _SEARCH_KEY = 'ytsearchdate' | ||||
|     IE_DESC = u'YouTube.com searches, newest videos first' | ||||
|  | ||||
| class YoutubeShowIE(InfoExtractor): | ||||
|     IE_DESC = u'YouTube.com (multi-season) shows' | ||||
|   | ||||
| @@ -2,11 +2,15 @@ import io | ||||
| import json | ||||
| import traceback | ||||
| import hashlib | ||||
| import os | ||||
| import subprocess | ||||
| import sys | ||||
| from zipimport import zipimporter | ||||
|  | ||||
| from .utils import * | ||||
| from .utils import ( | ||||
|     compat_str, | ||||
|     compat_urllib_request, | ||||
| ) | ||||
| from .version import __version__ | ||||
|  | ||||
| def rsa_verify(message, signature, key): | ||||
|   | ||||
| @@ -572,6 +572,11 @@ class ExtractorError(Exception): | ||||
|         return u''.join(traceback.format_tb(self.traceback)) | ||||
|  | ||||
|  | ||||
| class RegexNotFoundError(ExtractorError): | ||||
|     """Error when a regex didn't match""" | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class DownloadError(Exception): | ||||
|     """Download Error exception. | ||||
|  | ||||
|   | ||||
| @@ -1,2 +1,2 @@ | ||||
|  | ||||
| __version__ = '2013.10.18.1' | ||||
| __version__ = '2013.11.19' | ||||
|   | ||||
		Reference in New Issue
	
	Block a user