From e8cd8c4bd832446f1971215b9fedc4531555dc1a Mon Sep 17 00:00:00 2001 From: =?utf8?q?Rog=C3=A9rio=20Brito?= Date: Sun, 19 Jan 2014 02:38:23 -0200 Subject: [PATCH 1/1] Imported Upstream version 2014.01.17.2 --- README.md | 60 +- README.txt | 65 +- devscripts/bash-completion.in | 2 +- devscripts/check-porn.py | 44 +- devscripts/gh-pages/update-feed.py | 92 ++- devscripts/make_readme.py | 10 +- devscripts/release.sh | 2 + setup.py | 5 +- test/test_YoutubeDL.py | 91 ++- test/test_all_urls.py | 2 + test/test_download.py | 4 +- test/test_playlists.py | 84 +- test/test_subtitles.py | 8 +- test/test_unicode_literals.py | 47 ++ test/test_utils.py | 8 + test/test_youtube_signature.py | 6 - youtube-dl | Bin 313559 -> 333421 bytes youtube-dl.1 | 74 +- youtube-dl.bash-completion | 4 +- youtube_dl/FileDownloader.py | 728 +----------------- youtube_dl/YoutubeDL.py | 361 +++++---- youtube_dl/__init__.py | 59 +- youtube_dl/downloader/__init__.py | 23 + youtube_dl/downloader/common.py | 317 ++++++++ youtube_dl/downloader/hls.py | 44 ++ youtube_dl/downloader/http.py | 186 +++++ youtube_dl/downloader/mplayer.py | 40 + youtube_dl/downloader/rtmp.py | 178 +++++ youtube_dl/extractor/__init__.py | 21 +- youtube_dl/extractor/academicearth.py | 3 +- youtube_dl/extractor/appletrailers.py | 65 +- youtube_dl/extractor/archiveorg.py | 32 +- youtube_dl/extractor/arte.py | 46 +- youtube_dl/extractor/auengine.py | 17 +- youtube_dl/extractor/bambuser.py | 30 +- youtube_dl/extractor/bandcamp.py | 117 +-- youtube_dl/extractor/blinkx.py | 41 +- youtube_dl/extractor/bliptv.py | 162 ++-- youtube_dl/extractor/bloomberg.py | 5 +- youtube_dl/extractor/brightcove.py | 83 +- youtube_dl/extractor/c56.py | 40 +- youtube_dl/extractor/channel9.py | 20 +- youtube_dl/extractor/cmt.py | 19 + youtube_dl/extractor/cnn.py | 87 ++- youtube_dl/extractor/collegehumor.py | 106 ++- youtube_dl/extractor/comedycentral.py | 4 +- youtube_dl/extractor/common.py | 97 ++- youtube_dl/extractor/condenast.py | 74 +- youtube_dl/extractor/cspan.py | 65 +- youtube_dl/extractor/defense.py | 11 +- youtube_dl/extractor/dreisat.py | 15 +- youtube_dl/extractor/everyonesmixtape.py | 69 ++ youtube_dl/extractor/flickr.py | 22 +- youtube_dl/extractor/franceinter.py | 38 + youtube_dl/extractor/francetv.py | 26 + youtube_dl/extractor/gamespot.py | 14 +- youtube_dl/extractor/generic.py | 164 ++-- youtube_dl/extractor/imdb.py | 54 +- youtube_dl/extractor/ina.py | 2 +- youtube_dl/extractor/internetvideoarchive.py | 13 +- youtube_dl/extractor/ivi.py | 18 +- youtube_dl/extractor/jpopsukitv.py | 73 ++ youtube_dl/extractor/kankan.py | 40 +- youtube_dl/extractor/khanacademy.py | 71 ++ youtube_dl/extractor/lynda.py | 201 +++++ youtube_dl/extractor/macgamestore.py | 43 ++ youtube_dl/extractor/mdr.py | 3 +- youtube_dl/extractor/metacritic.py | 23 +- youtube_dl/extractor/mit.py | 17 +- youtube_dl/extractor/mixcloud.py | 36 +- youtube_dl/extractor/mpora.py | 66 ++ youtube_dl/extractor/mtv.py | 2 +- youtube_dl/extractor/myvideo.py | 6 +- youtube_dl/extractor/novamov.py | 62 ++ youtube_dl/extractor/nowvideo.py | 2 +- youtube_dl/extractor/orf.py | 120 ++- youtube_dl/extractor/pornhd.py | 2 +- youtube_dl/extractor/pornhub.py | 38 +- youtube_dl/extractor/redtube.py | 2 +- youtube_dl/extractor/rtlnow.py | 2 +- youtube_dl/extractor/smotri.py | 2 +- youtube_dl/extractor/soundcloud.py | 107 +-- youtube_dl/extractor/spankwire.py | 38 +- youtube_dl/extractor/spiegel.py | 3 +- youtube_dl/extractor/teamcoco.py | 57 +- youtube_dl/extractor/ted.py | 17 +- youtube_dl/extractor/theplatform.py | 14 +- youtube_dl/extractor/veehd.py | 39 +- youtube_dl/extractor/veoh.py | 46 +- youtube_dl/extractor/vimeo.py | 161 ++-- youtube_dl/extractor/wistia.py | 4 +- youtube_dl/extractor/yahoo.py | 12 +- youtube_dl/extractor/youporn.py | 38 +- youtube_dl/extractor/youtube.py | 335 +++----- youtube_dl/extractor/zdf.py | 36 +- youtube_dl/postprocessor/__init__.py | 18 + youtube_dl/postprocessor/common.py | 49 ++ .../ffmpeg.py} | 115 ++- youtube_dl/postprocessor/xattrpp.py | 108 +++ youtube_dl/utils.py | 91 ++- youtube_dl/version.py | 2 +- 101 files changed, 3860 insertions(+), 2365 deletions(-) create mode 100644 test/test_unicode_literals.py create mode 100644 youtube_dl/downloader/__init__.py create mode 100644 youtube_dl/downloader/common.py create mode 100644 youtube_dl/downloader/hls.py create mode 100644 youtube_dl/downloader/http.py create mode 100644 youtube_dl/downloader/mplayer.py create mode 100644 youtube_dl/downloader/rtmp.py create mode 100644 youtube_dl/extractor/cmt.py create mode 100644 youtube_dl/extractor/everyonesmixtape.py create mode 100644 youtube_dl/extractor/franceinter.py create mode 100644 youtube_dl/extractor/jpopsukitv.py create mode 100644 youtube_dl/extractor/khanacademy.py create mode 100644 youtube_dl/extractor/lynda.py create mode 100644 youtube_dl/extractor/macgamestore.py create mode 100644 youtube_dl/extractor/mpora.py create mode 100644 youtube_dl/extractor/novamov.py create mode 100644 youtube_dl/postprocessor/__init__.py create mode 100644 youtube_dl/postprocessor/common.py rename youtube_dl/{PostProcessor.py => postprocessor/ffmpeg.py} (80%) create mode 100644 youtube_dl/postprocessor/xattrpp.py diff --git a/README.md b/README.md index caed948..cf0bb7b 100644 --- a/README.md +++ b/README.md @@ -34,12 +34,16 @@ which means you can modify it, redistribute it or use it however you like. empty string (--proxy "") for direct connection --no-check-certificate Suppress HTTPS certificate validation. --cache-dir DIR Location in the filesystem where youtube-dl can - store downloaded information permanently. By + store some downloaded information permanently. By default $XDG_CACHE_HOME/youtube-dl or ~/.cache - /youtube-dl . + /youtube-dl . At the moment, only YouTube player + files (for videos with obfuscated signatures) are + cached, but that may change. --no-cache-dir Disable filesystem caching + --socket-timeout None Time to wait before giving up, in seconds --bidi-workaround Work around terminals that lack bidirectional - text support. Requires fribidi executable in PATH + text support. Requires bidiv or fribidi + executable in PATH ## Video Selection: --playlist-start NUMBER playlist video to start at (default is 1) @@ -54,8 +58,10 @@ which means you can modify it, redistribute it or use it however you like. --max-filesize SIZE Do not download any videos larger than SIZE (e.g. 50k or 44.6m) --date DATE download only videos uploaded in this date - --datebefore DATE download only videos uploaded before this date - --dateafter DATE download only videos uploaded after this date + --datebefore DATE download only videos uploaded on or before this + date (i.e. inclusive) + --dateafter DATE download only videos uploaded on or after this + date (i.e. inclusive) --min-views COUNT Do not download any videos with less than COUNT views --max-views COUNT Do not download any videos with more than COUNT @@ -87,13 +93,13 @@ which means you can modify it, redistribute it or use it however you like. different, %(autonumber)s to get an automatically incremented number, %(ext)s for the filename extension, %(format)s for the format description - (like "22 - 1280x720" or "HD"),%(format_id)s for + (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 + 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. Use - to output to stdout. Can also be used to download to a different directory, for example with -o '/my/d @@ -105,7 +111,7 @@ which means you can modify it, redistribute it or use it however you like. avoid "&" and spaces in filenames -a, --batch-file FILE file containing URLs to download ('-' for stdin) --load-info FILE json file containing the video information - (created with the "--write-json" option + (created with the "--write-json" option) -w, --no-overwrites do not overwrite files -c, --continue force resume of partially downloaded files. By default, youtube-dl will resume downloads if @@ -139,7 +145,7 @@ which means you can modify it, redistribute it or use it however you like. --no-progress do not print progress bar --console-title display progress in console titlebar -v, --verbose print various debugging information - --dump-intermediate-pages print downloaded pages to debug problems(very + --dump-intermediate-pages print downloaded pages to debug problems (very verbose) --write-pages Write downloaded intermediary pages to files in the current directory to debug problems @@ -152,8 +158,7 @@ which means you can modify it, redistribute it or use it however you like. --prefer-free-formats prefer free video formats unless a specific one is requested --max-quality FORMAT highest quality format to download - -F, --list-formats list all available formats (currently youtube - only) + -F, --list-formats list all available formats ## Subtitle Options: --write-sub write subtitle file @@ -171,7 +176,7 @@ which means you can modify it, redistribute it or use it however you like. -u, --username USERNAME account username -p, --password PASSWORD account password -n, --netrc use .netrc authentication data - --video-password PASSWORD video password (vimeo only) + --video-password PASSWORD video password (vimeo, smotri) ## Post-processing Options: -x, --extract-audio convert video files to audio-only files (requires @@ -189,7 +194,13 @@ which means you can modify it, redistribute it or use it however you like. processed files are overwritten by default --embed-subs embed subtitles in the video (only for mp4 videos) - --add-metadata add metadata to the files + --add-metadata write metadata to the video file + --xattrs write metadata to the video file's xattrs (using + dublin core and xdg standards) + --prefer-avconv Prefer avconv over ffmpeg for running the + postprocessors (default) + --prefer-ffmpeg Prefer ffmpeg over avconv for running the + postprocessors # CONFIGURATION @@ -228,9 +239,12 @@ Videos can be filtered by their upload date using the options `--date`, `--dateb Examples: - $ youtube-dl --dateafter now-6months #will only download the videos uploaded in the last 6 months - $ youtube-dl --date 19700101 #will only download the videos uploaded in January 1, 1970 - $ youtube-dl --dateafter 20000101 --datebefore 20100101 #will only download the videos uploaded between 2000 and 2010 + $ # Download only the videos uploaded in the last 6 months + $ youtube-dl --dateafter now-6months + $ # Download only the videos uploaded on January 1, 1970 + $ youtube-dl --date 19700101 + $ # will only download the videos uploaded in the 200x decade + $ youtube-dl --dateafter 20000101 --datebefore 20091231 # FAQ @@ -309,7 +323,7 @@ Site support requests must contain an example URL. An example URL is a URL you m ### Are you using the latest version? -Before reporting any issue, type youtube-dl -U. This should report that you're up-to-date. Ábout 20% of the reports we receive are already fixed, but people are using outdated versions. This goes for feature requests as well. +Before reporting any issue, type youtube-dl -U. This should report that you're up-to-date. About 20% of the reports we receive are already fixed, but people are using outdated versions. This goes for feature requests as well. ### Is the issue already documented? @@ -334,3 +348,7 @@ In particular, every site support request issue should only pertain to services ### Is anyone going to need the feature? Only post features that you (or an incapicated friend you can personally talk to) require. Do not post features because they seem like a good idea. If they are really useful, they will be requested by someone who requires them. + +### Is your question about youtube-dl? + +It may sound strange, but some bug reports we receive are completely unrelated to youtube-dl and relate to a different or even the reporter's own application. Please make sure that you are actually using youtube-dl. If you are using a UI for youtube-dl, report the bug to the maintainer of the actual application providing the UI. On the other hand, if your UI for youtube-dl fails in some way you believe is related to youtube-dl, by all means, go ahead and report the bug. diff --git a/README.txt b/README.txt index 3645b8c..69cab28 100644 --- a/README.txt +++ b/README.txt @@ -41,12 +41,16 @@ OPTIONS empty string (--proxy "") for direct connection --no-check-certificate Suppress HTTPS certificate validation. --cache-dir DIR Location in the filesystem where youtube-dl can - store downloaded information permanently. By + store some downloaded information permanently. By default $XDG_CACHE_HOME/youtube-dl or ~/.cache - /youtube-dl . + /youtube-dl . At the moment, only YouTube player + files (for videos with obfuscated signatures) are + cached, but that may change. --no-cache-dir Disable filesystem caching + --socket-timeout None Time to wait before giving up, in seconds --bidi-workaround Work around terminals that lack bidirectional - text support. Requires fribidi executable in PATH + text support. Requires bidiv or fribidi + executable in PATH Video Selection: ---------------- @@ -63,8 +67,10 @@ Video Selection: --max-filesize SIZE Do not download any videos larger than SIZE (e.g. 50k or 44.6m) --date DATE download only videos uploaded in this date - --datebefore DATE download only videos uploaded before this date - --dateafter DATE download only videos uploaded after this date + --datebefore DATE download only videos uploaded on or before this + date (i.e. inclusive) + --dateafter DATE download only videos uploaded on or after this + date (i.e. inclusive) --min-views COUNT Do not download any videos with less than COUNT views --max-views COUNT Do not download any videos with more than COUNT @@ -100,13 +106,13 @@ Filesystem Options: different, %(autonumber)s to get an automatically incremented number, %(ext)s for the filename extension, %(format)s for the format description - (like "22 - 1280x720" or "HD"),%(format_id)s for + (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 + 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. Use - to output to stdout. Can also be used to download to a different directory, for example with -o '/my/d @@ -118,7 +124,7 @@ Filesystem Options: avoid "&" and spaces in filenames -a, --batch-file FILE file containing URLs to download ('-' for stdin) --load-info FILE json file containing the video information - (created with the "--write-json" option + (created with the "--write-json" option) -w, --no-overwrites do not overwrite files -c, --continue force resume of partially downloaded files. By default, youtube-dl will resume downloads if @@ -154,7 +160,7 @@ Verbosity / Simulation Options: --no-progress do not print progress bar --console-title display progress in console titlebar -v, --verbose print various debugging information - --dump-intermediate-pages print downloaded pages to debug problems(very + --dump-intermediate-pages print downloaded pages to debug problems (very verbose) --write-pages Write downloaded intermediary pages to files in the current directory to debug problems @@ -169,8 +175,7 @@ Video Format Options: --prefer-free-formats prefer free video formats unless a specific one is requested --max-quality FORMAT highest quality format to download - -F, --list-formats list all available formats (currently youtube - only) + -F, --list-formats list all available formats Subtitle Options: ----------------- @@ -192,7 +197,7 @@ Authentication Options: -u, --username USERNAME account username -p, --password PASSWORD account password -n, --netrc use .netrc authentication data - --video-password PASSWORD video password (vimeo only) + --video-password PASSWORD video password (vimeo, smotri) Post-processing Options: ------------------------ @@ -212,7 +217,13 @@ Post-processing Options: processed files are overwritten by default --embed-subs embed subtitles in the video (only for mp4 videos) - --add-metadata add metadata to the files + --add-metadata write metadata to the video file + --xattrs write metadata to the video file's xattrs (using + dublin core and xdg standards) + --prefer-avconv Prefer avconv over ffmpeg for running the + postprocessors (default) + --prefer-ffmpeg Prefer ffmpeg over avconv for running the + postprocessors CONFIGURATION ============= @@ -277,9 +288,11 @@ Videos can be filtered by their upload date using the options --date, Examples: - $ youtube-dl --dateafter now-6months #will only download the videos uploaded in the last 6 months - $ youtube-dl --date 19700101 #will only download the videos uploaded in January 1, 1970 - $ youtube-dl --dateafter 20000101 --datebefore 20100101 #will only download the videos uploaded between 2000 and 2010 +$ # Download only the videos uploaded in the last 6 months $ youtube-dl +--dateafter now-6months $ # Download only the videos uploaded on January +1, 1970 $ youtube-dl --date 19700101 $ # will only download the videos +uploaded in the 200x decade $ youtube-dl --dateafter 20000101 +--datebefore 20091231 FAQ === @@ -413,7 +426,7 @@ a video service (e.g. http://www.youtube.com/ ) is not an example URL. Are you using the latest version? Before reporting any issue, type youtube-dl -U. This should report that -you're up-to-date. Ábout 20% of the reports we receive are already +you're up-to-date. About 20% of the reports we receive are already fixed, but people are using outdated versions. This goes for feature requests as well. @@ -478,3 +491,13 @@ Only post features that you (or an incapicated friend you can personally talk to) require. Do not post features because they seem like a good idea. If they are really useful, they will be requested by someone who requires them. + +Is your question about youtube-dl? + +It may sound strange, but some bug reports we receive are completely +unrelated to youtube-dl and relate to a different or even the reporter's +own application. Please make sure that you are actually using +youtube-dl. If you are using a UI for youtube-dl, report the bug to the +maintainer of the actual application providing the UI. On the other +hand, if your UI for youtube-dl fails in some way you believe is related +to youtube-dl, by all means, go ahead and report the bug. diff --git a/devscripts/bash-completion.in b/devscripts/bash-completion.in index 3af87a3..28bd237 100644 --- a/devscripts/bash-completion.in +++ b/devscripts/bash-completion.in @@ -6,7 +6,7 @@ __youtube_dl() prev="${COMP_WORDS[COMP_CWORD-1]}" opts="{{flags}}" keywords=":ytfavorites :ytrecommended :ytsubscriptions :ytwatchlater :ythistory" - fileopts="-a|--batch-file|--download-archive|--cookies" + fileopts="-a|--batch-file|--download-archive|--cookies|--load-info" diropts="--cache-dir" if [[ ${prev} =~ ${fileopts} ]]; then diff --git a/devscripts/check-porn.py b/devscripts/check-porn.py index 63401fe..86aa37b 100644 --- a/devscripts/check-porn.py +++ b/devscripts/check-porn.py @@ -3,6 +3,9 @@ """ This script employs a VERY basic heuristic ('porn' in webpage.lower()) to check if we are not 'age_limit' tagging some porn site + +A second approach implemented relies on a list of porn domains, to activate it +pass the list filename as the only argument """ # Allow direct execution @@ -11,25 +14,42 @@ 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_parse_urlparse from youtube_dl.utils import compat_urllib_request +if len(sys.argv) > 1: + METHOD = 'LIST' + LIST = open(sys.argv[1]).read().decode('utf8').strip() +else: + METHOD = 'EURISTIC' + for test in get_testcases(): - try: - webpage = compat_urllib_request.urlopen(test['url'], timeout=10).read() - except: - print('\nFail: {0}'.format(test['name'])) - continue + if METHOD == 'EURISTIC': + try: + webpage = compat_urllib_request.urlopen(test['url'], timeout=10).read() + except: + print('\nFail: {0}'.format(test['name'])) + continue + + webpage = webpage.decode('utf8', 'replace') + + RESULT = 'porn' in webpage.lower() + + elif METHOD == 'LIST': + domain = compat_urllib_parse_urlparse(test['url']).netloc + if not domain: + print('\nFail: {0}'.format(test['name'])) + continue + domain = '.'.join(domain.split('.')[-2:]) - webpage = webpage.decode('utf8', 'replace') + RESULT = ('.' + domain + '\n' in LIST or '\n' + domain + '\n' in LIST) - 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): + if RESULT 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): + elif not RESULT 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: diff --git a/devscripts/gh-pages/update-feed.py b/devscripts/gh-pages/update-feed.py index 16571a9..0ba15ae 100755 --- a/devscripts/gh-pages/update-feed.py +++ b/devscripts/gh-pages/update-feed.py @@ -1,56 +1,76 @@ #!/usr/bin/env python3 import datetime - +import io +import json import textwrap -import json -atom_template=textwrap.dedent("""\ - - - youtube-dl releases - youtube-dl-updates-feed - @TIMESTAMP@ - @ENTRIES@ - """) - -entry_template=textwrap.dedent(""" - - youtube-dl-@VERSION@ - New version @VERSION@ - - -
- Downloads available at https://yt-dl.org/downloads/@VERSION@/ -
-
- - The youtube-dl maintainers - - @TIMESTAMP@ -
- """) +atom_template = textwrap.dedent("""\ + + + + youtube-dl releases + https://yt-dl.org/feed/youtube-dl-updates-feed + @TIMESTAMP@ + @ENTRIES@ + """) -now = datetime.datetime.now() -now_iso = now.isoformat() +entry_template = textwrap.dedent(""" + + https://yt-dl.org/feed/youtube-dl-updates-feed/youtube-dl-@VERSION@ + New version @VERSION@ + + +
+ Downloads available at https://yt-dl.org/downloads/@VERSION@/ +
+
+ + The youtube-dl maintainers + + @TIMESTAMP@ +
+ """) -atom_template = atom_template.replace('@TIMESTAMP@',now_iso) +now = datetime.datetime.now() +now_iso = now.isoformat() + 'Z' -entries=[] +atom_template = atom_template.replace('@TIMESTAMP@', now_iso) versions_info = json.load(open('update/versions.json')) versions = list(versions_info['versions'].keys()) versions.sort() +entries = [] for v in versions: - entry = entry_template.replace('@TIMESTAMP@',v.replace('.','-')) - entry = entry.replace('@VERSION@',v) - entries.append(entry) + fields = v.split('.') + year, month, day = map(int, fields[:3]) + faked = 0 + patchlevel = 0 + while True: + try: + datetime.date(year, month, day) + except ValueError: + day -= 1 + faked += 1 + assert day > 0 + continue + break + if len(fields) >= 4: + try: + patchlevel = int(fields[3]) + except ValueError: + patchlevel = 1 + timestamp = '%04d-%02d-%02dT00:%02d:%02dZ' % (year, month, day, faked, patchlevel) + + entry = entry_template.replace('@TIMESTAMP@', timestamp) + entry = entry.replace('@VERSION@', v) + entries.append(entry) entries_str = textwrap.indent(''.join(entries), '\t') atom_template = atom_template.replace('@ENTRIES@', entries_str) -with open('update/releases.atom','w',encoding='utf-8') as atom_file: - atom_file.write(atom_template) +with io.open('update/releases.atom', 'w', encoding='utf-8') as atom_file: + atom_file.write(atom_template) diff --git a/devscripts/make_readme.py b/devscripts/make_readme.py index 7f2ea31..cae1fa4 100755 --- a/devscripts/make_readme.py +++ b/devscripts/make_readme.py @@ -1,20 +1,24 @@ +import io import sys import re README_FILE = 'README.md' helptext = sys.stdin.read() -with open(README_FILE) as f: +if isinstance(helptext, bytes): + helptext = helptext.decode('utf-8') + +with io.open(README_FILE, encoding='utf-8') as f: oldreadme = f.read() header = oldreadme[:oldreadme.index('# OPTIONS')] footer = oldreadme[oldreadme.index('# CONFIGURATION'):] -options = helptext[helptext.index(' General Options:')+19:] +options = helptext[helptext.index(' General Options:') + 19:] options = re.sub(r'^ (\w.+)$', r'## \1', options, flags=re.M) options = '# OPTIONS\n' + options + '\n' -with open(README_FILE, 'w') as f: +with io.open(README_FILE, 'w', encoding='utf-8') as f: f.write(header) f.write(options) f.write(footer) diff --git a/devscripts/release.sh b/devscripts/release.sh index 2766174..323acf8 100755 --- a/devscripts/release.sh +++ b/devscripts/release.sh @@ -24,6 +24,8 @@ if [ -z "$1" ]; then echo "ERROR: specify version number like this: $0 1994.09.0 version="$1" if [ ! -z "`git tag | grep "$version"`" ]; then echo 'ERROR: version already present'; exit 1; fi if [ ! -z "`git status --porcelain | grep -v CHANGELOG`" ]; then echo 'ERROR: the working directory is not clean; commit or stash changes'; exit 1; fi +useless_files=$(find youtube_dl -type f -not -name '*.py') +if [ ! -z "$useless_files" ]; then echo "ERROR: Non-.py files in youtube_dl: $useless_files"; exit 1; fi if [ ! -f "updates_key.pem" ]; then echo 'ERROR: updates_key.pem missing'; exit 1; fi /bin/echo -e "\n### First of all, testing..." diff --git a/setup.py b/setup.py index 8e24fe6..1f45159 100644 --- a/setup.py +++ b/setup.py @@ -71,7 +71,10 @@ setup( author_email='ytdl@yt-dl.org', maintainer='Philipp Hagemeister', maintainer_email='phihag@phihag.de', - packages=['youtube_dl', 'youtube_dl.extractor'], + packages=[ + 'youtube_dl', + 'youtube_dl.extractor', 'youtube_dl.downloader', + 'youtube_dl.postprocessor'], # Provokes warning on most systems (why?!) # test_suite = 'nose.collector', diff --git a/test/test_YoutubeDL.py b/test/test_YoutubeDL.py index 3100c36..01de10e 100644 --- a/test/test_YoutubeDL.py +++ b/test/test_YoutubeDL.py @@ -8,6 +8,7 @@ sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) from test.helper import FakeYDL from youtube_dl import YoutubeDL +from youtube_dl.extractor import YoutubeIE class YDL(FakeYDL): @@ -33,6 +34,8 @@ class TestFormatSelection(unittest.TestCase): {u'ext': u'mp4', u'height': 460}, ] info_dict = {u'formats': formats, u'extractor': u'test'} + yie = YoutubeIE(ydl) + yie._sort_formats(info_dict['formats']) ydl.process_ie_result(info_dict) downloaded = ydl.downloaded_info_dicts[0] self.assertEqual(downloaded[u'ext'], u'webm') @@ -45,28 +48,46 @@ class TestFormatSelection(unittest.TestCase): {u'ext': u'mp4', u'height': 1080}, ] info_dict[u'formats'] = formats + yie = YoutubeIE(ydl) + yie._sort_formats(info_dict['formats']) ydl.process_ie_result(info_dict) downloaded = ydl.downloaded_info_dicts[0] self.assertEqual(downloaded[u'ext'], u'mp4') - # No prefer_free_formats => keep original formats order + # No prefer_free_formats => prefer mp4 and flv for greater compatibilty ydl = YDL() ydl.params['prefer_free_formats'] = False formats = [ {u'ext': u'webm', u'height': 720}, + {u'ext': u'mp4', u'height': 720}, {u'ext': u'flv', u'height': 720}, ] info_dict[u'formats'] = formats + yie = YoutubeIE(ydl) + yie._sort_formats(info_dict['formats']) + ydl.process_ie_result(info_dict) + downloaded = ydl.downloaded_info_dicts[0] + self.assertEqual(downloaded[u'ext'], u'mp4') + + ydl = YDL() + ydl.params['prefer_free_formats'] = False + formats = [ + {u'ext': u'flv', u'height': 720}, + {u'ext': u'webm', u'height': 720}, + ] + info_dict[u'formats'] = formats + yie = YoutubeIE(ydl) + yie._sort_formats(info_dict['formats']) ydl.process_ie_result(info_dict) downloaded = ydl.downloaded_info_dicts[0] self.assertEqual(downloaded[u'ext'], u'flv') def test_format_limit(self): formats = [ - {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'}, + {u'format_id': u'meh', u'url': u'http://example.com/meh', 'preference': 1}, + {u'format_id': u'good', u'url': u'http://example.com/good', 'preference': 2}, + {u'format_id': u'great', u'url': u'http://example.com/great', 'preference': 3}, + {u'format_id': u'excellent', u'url': u'http://example.com/exc', 'preference': 4}, ] info_dict = { u'formats': formats, u'extractor': u'test', 'id': 'testvid'} @@ -78,12 +99,12 @@ class TestFormatSelection(unittest.TestCase): ydl = YDL({'format_limit': 'good'}) assert ydl.params['format_limit'] == 'good' - ydl.process_ie_result(info_dict) + ydl.process_ie_result(info_dict.copy()) downloaded = ydl.downloaded_info_dicts[0] self.assertEqual(downloaded[u'format_id'], u'good') ydl = YDL({'format_limit': 'great', 'format': 'all'}) - ydl.process_ie_result(info_dict) + ydl.process_ie_result(info_dict.copy()) self.assertEqual(ydl.downloaded_info_dicts[0][u'format_id'], u'meh') self.assertEqual(ydl.downloaded_info_dicts[1][u'format_id'], u'good') self.assertEqual(ydl.downloaded_info_dicts[2][u'format_id'], u'great') @@ -91,44 +112,80 @@ class TestFormatSelection(unittest.TestCase): ydl = YDL() ydl.params['format_limit'] = 'excellent' - ydl.process_ie_result(info_dict) + ydl.process_ie_result(info_dict.copy()) 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'}, + {u'format_id': u'35', u'ext': u'mp4', 'preference': 1}, + {u'format_id': u'45', u'ext': u'webm', 'preference': 2}, + {u'format_id': u'47', u'ext': u'webm', 'preference': 3}, + {u'format_id': u'2', u'ext': u'flv', 'preference': 4}, ] info_dict = {u'formats': formats, u'extractor': u'test'} ydl = YDL({'format': u'20/47'}) - ydl.process_ie_result(info_dict) + ydl.process_ie_result(info_dict.copy()) 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) + ydl.process_ie_result(info_dict.copy()) downloaded = ydl.downloaded_info_dicts[0] self.assertEqual(downloaded['format_id'], u'35') ydl = YDL() - ydl.process_ie_result(info_dict) + ydl.process_ie_result(info_dict.copy()) 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) + ydl.process_ie_result(info_dict.copy()) 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) + ydl.process_ie_result(info_dict.copy()) downloaded = ydl.downloaded_info_dicts[0] self.assertEqual(downloaded['format_id'], u'35') + def test_youtube_format_selection(self): + order = [ + '38', '37', '46', '22', '45', '35', '44', '18', '34', '43', '6', '5', '36', '17', '13', + # Apple HTTP Live Streaming + '96', '95', '94', '93', '92', '132', '151', + # 3D + '85', '84', '102', '83', '101', '82', '100', + # Dash video + '138', '137', '248', '136', '247', '135', '246', + '245', '244', '134', '243', '133', '242', '160', + # Dash audio + '141', '172', '140', '139', '171', + ] + + for f1id, f2id in zip(order, order[1:]): + f1 = YoutubeIE._formats[f1id].copy() + f1['format_id'] = f1id + f2 = YoutubeIE._formats[f2id].copy() + f2['format_id'] = f2id + + info_dict = {'formats': [f1, f2], 'extractor': 'youtube'} + ydl = YDL() + yie = YoutubeIE(ydl) + yie._sort_formats(info_dict['formats']) + ydl.process_ie_result(info_dict) + downloaded = ydl.downloaded_info_dicts[0] + self.assertEqual(downloaded['format_id'], f1id) + + info_dict = {'formats': [f2, f1], 'extractor': 'youtube'} + ydl = YDL() + yie = YoutubeIE(ydl) + yie._sort_formats(info_dict['formats']) + ydl.process_ie_result(info_dict) + downloaded = ydl.downloaded_info_dicts[0] + self.assertEqual(downloaded['format_id'], f1id) + def test_add_extra_info(self): test_dict = { 'extractor': 'Foo', diff --git a/test/test_all_urls.py b/test/test_all_urls.py index bd77b7c..75547f4 100644 --- a/test/test_all_urls.py +++ b/test/test_all_urls.py @@ -113,6 +113,8 @@ class TestAllURLsMatching(unittest.TestCase): def test_vimeo_matching(self): self.assertMatch('http://vimeo.com/channels/tributes', ['vimeo:channel']) self.assertMatch('http://vimeo.com/user7108434', ['vimeo:user']) + self.assertMatch('http://vimeo.com/user7108434/videos', ['vimeo:user']) + self.assertMatch('https://vimeo.com/user21297594/review/75524534/3c257a1b5d', ['vimeo:review']) # https://github.com/rg3/youtube-dl/issues/1930 def test_soundcloud_not_matching_sets(self): diff --git a/test/test_download.py b/test/test_download.py index dd5818d..0d925ae 100644 --- a/test/test_download.py +++ b/test/test_download.py @@ -90,7 +90,7 @@ def generator(test_case): def _hook(status): if status['status'] == 'finished': finished_hook_called.add(status['filename']) - ydl.fd.add_progress_hook(_hook) + ydl.add_progress_hook(_hook) def get_tc_filename(tc): return tc.get('file') or ydl.prepare_filename(tc.get('info_dict', {})) @@ -148,7 +148,7 @@ def generator(test_case): for key, value in info_dict.items() if value and key in ('title', 'description', 'uploader', 'upload_date', 'uploader_id', 'location')) if not all(key in tc.get('info_dict', {}).keys() for key in test_info_dict.keys()): - sys.stderr.write(u'\n"info_dict": ' + json.dumps(test_info_dict, ensure_ascii=False, indent=2) + u'\n') + sys.stderr.write(u'\n"info_dict": ' + json.dumps(test_info_dict, ensure_ascii=False, indent=4) + u'\n') # Check for the presence of mandatory fields for key in ('id', 'url', 'title', 'ext'): diff --git a/test/test_playlists.py b/test/test_playlists.py index 1b7b4e3..5eeba09 100644 --- a/test/test_playlists.py +++ b/test/test_playlists.py @@ -1,6 +1,7 @@ #!/usr/bin/env python # encoding: utf-8 +from __future__ import unicode_literals # Allow direct execution import os @@ -28,7 +29,10 @@ from youtube_dl.extractor import ( BandcampAlbumIE, SmotriCommunityIE, SmotriUserIE, - IviCompilationIE + IviCompilationIE, + ImdbListIE, + KhanAcademyIE, + EveryonesMixtapeIE, ) @@ -42,7 +46,7 @@ class TestPlaylists(unittest.TestCase): ie = DailymotionPlaylistIE(dl) result = ie.extract('http://www.dailymotion.com/playlist/xv4bw_nqtv_sport/1#video=xl8v3q') self.assertIsPlaylist(result) - self.assertEqual(result['title'], u'SPORT') + self.assertEqual(result['title'], 'SPORT') self.assertTrue(len(result['entries']) > 20) def test_dailymotion_user(self): @@ -50,7 +54,7 @@ class TestPlaylists(unittest.TestCase): ie = DailymotionUserIE(dl) result = ie.extract('http://www.dailymotion.com/user/generation-quoi/') self.assertIsPlaylist(result) - self.assertEqual(result['title'], u'Génération Quoi') + self.assertEqual(result['title'], 'Génération Quoi') self.assertTrue(len(result['entries']) >= 26) def test_vimeo_channel(self): @@ -58,7 +62,7 @@ class TestPlaylists(unittest.TestCase): ie = VimeoChannelIE(dl) result = ie.extract('http://vimeo.com/channels/tributes') self.assertIsPlaylist(result) - self.assertEqual(result['title'], u'Vimeo Tributes') + self.assertEqual(result['title'], 'Vimeo Tributes') self.assertTrue(len(result['entries']) > 24) def test_vimeo_user(self): @@ -66,7 +70,7 @@ class TestPlaylists(unittest.TestCase): ie = VimeoUserIE(dl) result = ie.extract('http://vimeo.com/nkistudio/videos') self.assertIsPlaylist(result) - self.assertEqual(result['title'], u'Nki') + self.assertEqual(result['title'], 'Nki') self.assertTrue(len(result['entries']) > 65) def test_vimeo_album(self): @@ -74,7 +78,7 @@ class TestPlaylists(unittest.TestCase): ie = VimeoAlbumIE(dl) result = ie.extract('http://vimeo.com/album/2632481') self.assertIsPlaylist(result) - self.assertEqual(result['title'], u'Staff Favorites: November 2013') + self.assertEqual(result['title'], 'Staff Favorites: November 2013') self.assertTrue(len(result['entries']) > 12) def test_vimeo_groups(self): @@ -82,7 +86,7 @@ class TestPlaylists(unittest.TestCase): ie = VimeoGroupsIE(dl) result = ie.extract('http://vimeo.com/groups/rolexawards') self.assertIsPlaylist(result) - self.assertEqual(result['title'], u'Rolex Awards for Enterprise') + self.assertEqual(result['title'], 'Rolex Awards for Enterprise') self.assertTrue(len(result['entries']) > 72) def test_ustream_channel(self): @@ -90,7 +94,7 @@ class TestPlaylists(unittest.TestCase): ie = UstreamChannelIE(dl) result = ie.extract('http://www.ustream.tv/channel/young-americans-for-liberty') self.assertIsPlaylist(result) - self.assertEqual(result['id'], u'5124905') + self.assertEqual(result['id'], '5124905') self.assertTrue(len(result['entries']) >= 11) def test_soundcloud_set(self): @@ -98,7 +102,7 @@ class TestPlaylists(unittest.TestCase): 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.assertEqual(result['title'], 'The Royal Concept EP') self.assertTrue(len(result['entries']) >= 6) def test_soundcloud_user(self): @@ -106,7 +110,7 @@ class TestPlaylists(unittest.TestCase): ie = SoundcloudUserIE(dl) result = ie.extract('https://soundcloud.com/the-concept-band') self.assertIsPlaylist(result) - self.assertEqual(result['id'], u'9615865') + self.assertEqual(result['id'], '9615865') self.assertTrue(len(result['entries']) >= 12) def test_livestream_event(self): @@ -114,7 +118,7 @@ class TestPlaylists(unittest.TestCase): ie = LivestreamIE(dl) result = ie.extract('http://new.livestream.com/tedx/cityenglish') self.assertIsPlaylist(result) - self.assertEqual(result['title'], u'TEDCity2.0 (English)') + self.assertEqual(result['title'], 'TEDCity2.0 (English)') self.assertTrue(len(result['entries']) >= 4) def test_nhl_videocenter(self): @@ -122,8 +126,8 @@ class TestPlaylists(unittest.TestCase): ie = NHLVideocenterIE(dl) result = ie.extract('http://video.canucks.nhl.com/videocenter/console?catid=999') self.assertIsPlaylist(result) - self.assertEqual(result['id'], u'999') - self.assertEqual(result['title'], u'Highlights') + self.assertEqual(result['id'], '999') + self.assertEqual(result['title'], 'Highlights') self.assertEqual(len(result['entries']), 12) def test_bambuser_channel(self): @@ -131,7 +135,7 @@ class TestPlaylists(unittest.TestCase): ie = BambuserChannelIE(dl) result = ie.extract('http://bambuser.com/channel/pixelversity') self.assertIsPlaylist(result) - self.assertEqual(result['title'], u'pixelversity') + self.assertEqual(result['title'], 'pixelversity') self.assertTrue(len(result['entries']) >= 60) def test_bandcamp_album(self): @@ -139,7 +143,7 @@ class TestPlaylists(unittest.TestCase): ie = BandcampAlbumIE(dl) result = ie.extract('http://mpallante.bandcamp.com/album/nightmare-night-ep') self.assertIsPlaylist(result) - self.assertEqual(result['title'], u'Nightmare Night EP') + self.assertEqual(result['title'], 'Nightmare Night EP') self.assertTrue(len(result['entries']) >= 4) def test_smotri_community(self): @@ -147,8 +151,8 @@ class TestPlaylists(unittest.TestCase): ie = SmotriCommunityIE(dl) result = ie.extract('http://smotri.com/community/video/kommuna') self.assertIsPlaylist(result) - self.assertEqual(result['id'], u'kommuna') - self.assertEqual(result['title'], u'КПРФ') + self.assertEqual(result['id'], 'kommuna') + self.assertEqual(result['title'], 'КПРФ') self.assertTrue(len(result['entries']) >= 4) def test_smotri_user(self): @@ -156,17 +160,17 @@ class TestPlaylists(unittest.TestCase): ie = SmotriUserIE(dl) result = ie.extract('http://smotri.com/user/inspector') self.assertIsPlaylist(result) - self.assertEqual(result['id'], u'inspector') - self.assertEqual(result['title'], u'Inspector') + self.assertEqual(result['id'], 'inspector') + self.assertEqual(result['title'], 'Inspector') self.assertTrue(len(result['entries']) >= 9) def test_AcademicEarthCourse(self): dl = FakeYDL() ie = AcademicEarthCourseIE(dl) - result = ie.extract(u'http://academicearth.org/courses/building-dynamic-websites/') + result = ie.extract('http://academicearth.org/courses/building-dynamic-websites/') self.assertIsPlaylist(result) - self.assertEqual(result['id'], u'building-dynamic-websites') - self.assertEqual(result['title'], u'Building Dynamic Websites') + self.assertEqual(result['id'], 'building-dynamic-websites') + self.assertEqual(result['title'], 'Building Dynamic Websites') self.assertEqual(result['description'], u"Today's websites are increasingly dynamic. Pages are no longer static HTML files but instead generated by scripts and database calls. User interfaces are more seamless, with technologies like Ajax replacing traditional page reloads. This course teaches students how to build dynamic websites with Ajax and with Linux, Apache, MySQL, and PHP (LAMP), one of today's most popular frameworks. Students learn how to set up domain names with DNS, how to structure pages with XHTML and CSS, how to program in JavaScript and PHP, how to configure Apache and MySQL, how to design and query databases with SQL, how to use Ajax with both XML and JSON, and how to build mashups. The course explores issues of security, scalability, and cross-browser support and also discusses enterprise-level deployments of websites, including third-party hosting, virtualization, colocation in data centers, firewalling, and load-balancing.") self.assertEqual(len(result['entries']), 10) @@ -175,8 +179,8 @@ class TestPlaylists(unittest.TestCase): ie = IviCompilationIE(dl) result = ie.extract('http://www.ivi.ru/watch/dezhurnyi_angel') self.assertIsPlaylist(result) - self.assertEqual(result['id'], u'dezhurnyi_angel') - self.assertEqual(result['title'], u'Дежурный ангел (2010 - 2012)') + self.assertEqual(result['id'], 'dezhurnyi_angel') + self.assertEqual(result['title'], 'Дежурный ангел (2010 - 2012)') self.assertTrue(len(result['entries']) >= 36) def test_ivi_compilation_season(self): @@ -184,9 +188,37 @@ class TestPlaylists(unittest.TestCase): ie = IviCompilationIE(dl) result = ie.extract('http://www.ivi.ru/watch/dezhurnyi_angel/season2') self.assertIsPlaylist(result) - self.assertEqual(result['id'], u'dezhurnyi_angel/season2') - self.assertEqual(result['title'], u'Дежурный ангел (2010 - 2012) 2 сезон') + self.assertEqual(result['id'], 'dezhurnyi_angel/season2') + self.assertEqual(result['title'], 'Дежурный ангел (2010 - 2012) 2 сезон') self.assertTrue(len(result['entries']) >= 20) + + def test_imdb_list(self): + dl = FakeYDL() + ie = ImdbListIE(dl) + result = ie.extract('http://www.imdb.com/list/sMjedvGDd8U') + self.assertIsPlaylist(result) + self.assertEqual(result['id'], 'sMjedvGDd8U') + self.assertEqual(result['title'], 'Animated and Family Films') + self.assertTrue(len(result['entries']) >= 48) + + def test_khanacademy_topic(self): + dl = FakeYDL() + ie = KhanAcademyIE(dl) + result = ie.extract('https://www.khanacademy.org/math/applied-math/cryptography') + self.assertIsPlaylist(result) + self.assertEqual(result['id'], 'cryptography') + self.assertEqual(result['title'], 'Journey into cryptography') + self.assertEqual(result['description'], 'How have humans protected their secret messages through history? What has changed today?') + self.assertTrue(len(result['entries']) >= 3) + + def test_EveryonesMixtape(self): + dl = FakeYDL() + ie = EveryonesMixtapeIE(dl) + result = ie.extract('http://everyonesmixtape.com/#/mix/m7m0jJAbMQi') + self.assertIsPlaylist(result) + self.assertEqual(result['id'], 'm7m0jJAbMQi') + self.assertEqual(result['title'], 'Driving') + self.assertEqual(len(result['entries']), 24) if __name__ == '__main__': diff --git a/test/test_subtitles.py b/test/test_subtitles.py index 23a6531..1e4e62f 100644 --- a/test/test_subtitles.py +++ b/test/test_subtitles.py @@ -36,10 +36,6 @@ 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() @@ -171,13 +167,13 @@ class TestTedSubtitles(BaseTestSubtitles): def test_subtitles(self): self.DL.params['writesubtitles'] = True subtitles = self.getSubtitles() - self.assertEqual(md5(subtitles['en']), '2154f31ff9b9f89a0aa671537559c21d') + self.assertEqual(md5(subtitles['en']), '4262c1665ff928a2dada178f62cb8d14') def test_subtitles_lang(self): self.DL.params['writesubtitles'] = True self.DL.params['subtitleslangs'] = ['fr'] subtitles = self.getSubtitles() - self.assertEqual(md5(subtitles['fr']), '7616cbc6df20ec2c1204083c83871cf6') + self.assertEqual(md5(subtitles['fr']), '66a63f7f42c97a50f8c0e90bc7797bb5') def test_allsubtitles(self): self.DL.params['writesubtitles'] = True diff --git a/test/test_unicode_literals.py b/test/test_unicode_literals.py new file mode 100644 index 0000000..a4ba7ba --- /dev/null +++ b/test/test_unicode_literals.py @@ -0,0 +1,47 @@ +from __future__ import unicode_literals + +import io +import os +import re +import unittest + +rootDir = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) + +IGNORED_FILES = [ + 'setup.py', # http://bugs.python.org/issue13943 +] + + +class TestUnicodeLiterals(unittest.TestCase): + def test_all_files(self): + print('Skipping this test (not yet fully implemented)') + return + + for dirpath, _, filenames in os.walk(rootDir): + for basename in filenames: + if not basename.endswith('.py'): + continue + if basename in IGNORED_FILES: + continue + + fn = os.path.join(dirpath, basename) + with io.open(fn, encoding='utf-8') as inf: + code = inf.read() + + if "'" not in code and '"' not in code: + continue + imps = 'from __future__ import unicode_literals' + self.assertTrue( + imps in code, + ' %s missing in %s' % (imps, fn)) + + m = re.search(r'(?<=\s)u[\'"](?!\)|,|$)', code) + if m is not None: + self.assertTrue( + m is None, + 'u present in %s, around %s' % ( + fn, code[m.start() - 10:m.end() + 10])) + + +if __name__ == '__main__': + unittest.main() diff --git a/test/test_utils.py b/test/test_utils.py index e5778cd..bee355e 100644 --- a/test/test_utils.py +++ b/test/test_utils.py @@ -18,6 +18,7 @@ from youtube_dl.utils import ( find_xpath_attr, get_meta_content, orderedSet, + parse_duration, sanitize_filename, shell_quote, smuggle_url, @@ -192,5 +193,12 @@ class TestUtil(unittest.TestCase): url_basename(u'http://media.w3.org/2010/05/sintel/trailer.mp4'), u'trailer.mp4') + def test_parse_duration(self): + self.assertEqual(parse_duration(None), None) + self.assertEqual(parse_duration('1'), 1) + self.assertEqual(parse_duration('1337:12'), 80232) + self.assertEqual(parse_duration('9:12:43'), 33163) + self.assertEqual(parse_duration('x:y'), None) + if __name__ == '__main__': unittest.main() diff --git a/test/test_youtube_signature.py b/test/test_youtube_signature.py index 0567006..a3fc530 100644 --- a/test/test_youtube_signature.py +++ b/test/test_youtube_signature.py @@ -27,12 +27,6 @@ _TESTS = [ 85, u'3456789a0cdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRS[UVWXYZ!"#$%&\'()*+,-./:;<=>?@', ), - ( - u'https://s.ytimg.com/yts/swfbin/watch_as3-vflg5GhxU.swf', - u'swf', - 82, - u':/.-,+*)=\'&%$#"!ZYX0VUTSRQPONMLKJIHGFEDCBAzyxw>utsrqponmlkjihgfedcba987654321' - ), ] diff --git a/youtube-dl b/youtube-dl index 2e3e8a9b6cf644d8b656750bfe3386298e6609d3..1e06aaad529038d248e799b88bb24d209941183a 100755 GIT binary patch delta 188543 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�$lk2;Jf{y>+H}>dYXh2^Pu)kXn_b}d16uunV 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

o`8VB$R5=iZb|@X4czW+J_{} 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

Lz;s++`}pd<##X8j?sQj%36yc6GjiOdGGE> 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

  • p&15wbbpD{G!r3NVKi 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 diff --git a/youtube-dl.1 b/youtube-dl.1 index c99538c..ce1c4f9 100644 --- a/youtube-dl.1 +++ b/youtube-dl.1 @@ -1,4 +1,4 @@ -.TH YOUTUBE\-DL 1 "" +.TH "YOUTUBE\-DL" "1" "" "" "" .SH NAME .PP youtube\-dl \- download videos from youtube.com or other video platforms @@ -38,12 +38,16 @@ redistribute it or use it however you like. \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ empty\ string\ (\-\-proxy\ "")\ for\ direct\ connection \-\-no\-check\-certificate\ \ \ \ \ Suppress\ HTTPS\ certificate\ validation. \-\-cache\-dir\ DIR\ \ \ \ \ \ \ \ \ \ \ \ Location\ in\ the\ filesystem\ where\ youtube\-dl\ can -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ store\ downloaded\ information\ permanently.\ By +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ store\ some\ downloaded\ information\ permanently.\ By \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ default\ $XDG_CACHE_HOME/youtube\-dl\ or\ ~/.cache -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ /youtube\-dl\ . +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ /youtube\-dl\ .\ At\ the\ moment,\ only\ YouTube\ player +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ files\ (for\ videos\ with\ obfuscated\ signatures)\ are +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ cached,\ but\ that\ may\ change. \-\-no\-cache\-dir\ \ \ \ \ \ \ \ \ \ \ \ \ Disable\ filesystem\ caching +\-\-socket\-timeout\ None\ \ \ \ \ \ Time\ to\ wait\ before\ giving\ up,\ in\ seconds \-\-bidi\-workaround\ \ \ \ \ \ \ \ \ \ Work\ around\ terminals\ that\ lack\ bidirectional -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ text\ support.\ Requires\ fribidi\ executable\ in\ PATH +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ text\ support.\ Requires\ bidiv\ or\ fribidi +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ executable\ in\ PATH \f[] .fi .SS Video Selection: @@ -62,8 +66,10 @@ redistribute it or use it however you like. \-\-max\-filesize\ SIZE\ \ \ \ \ \ \ \ Do\ not\ download\ any\ videos\ larger\ than\ SIZE\ (e.g. \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ 50k\ or\ 44.6m) \-\-date\ DATE\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ download\ only\ videos\ uploaded\ in\ this\ date -\-\-datebefore\ DATE\ \ \ \ \ \ \ \ \ \ download\ only\ videos\ uploaded\ before\ this\ date -\-\-dateafter\ DATE\ \ \ \ \ \ \ \ \ \ \ download\ only\ videos\ uploaded\ after\ this\ date +\-\-datebefore\ DATE\ \ \ \ \ \ \ \ \ \ download\ only\ videos\ uploaded\ on\ or\ before\ this +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ date\ (i.e.\ inclusive) +\-\-dateafter\ DATE\ \ \ \ \ \ \ \ \ \ \ download\ only\ videos\ uploaded\ on\ or\ after\ this +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ date\ (i.e.\ inclusive) \-\-min\-views\ COUNT\ \ \ \ \ \ \ \ \ \ Do\ not\ download\ any\ videos\ with\ less\ than\ COUNT \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ views \-\-max\-views\ COUNT\ \ \ \ \ \ \ \ \ \ Do\ not\ download\ any\ videos\ with\ more\ than\ COUNT @@ -103,13 +109,13 @@ redistribute it or use it however you like. \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ different,\ %(autonumber)s\ to\ get\ an\ automatically \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ incremented\ number,\ %(ext)s\ for\ the\ filename \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ extension,\ %(format)s\ for\ the\ format\ description -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ (like\ "22\ \-\ 1280x720"\ or\ "HD"),%(format_id)s\ for +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ (like\ "22\ \-\ 1280x720"\ or\ "HD"),\ %(format_id)s\ for \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ the\ unique\ id\ of\ the\ format\ (like\ Youtube\[aq]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 +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ 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.\ Use\ \-\ to \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ output\ to\ stdout.\ Can\ also\ be\ used\ to\ download\ to \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ a\ different\ directory,\ for\ example\ with\ \-o\ \[aq]/my/d @@ -121,7 +127,7 @@ redistribute it or use it however you like. \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ avoid\ "&"\ and\ spaces\ in\ filenames \-a,\ \-\-batch\-file\ FILE\ \ \ \ \ \ file\ containing\ URLs\ to\ download\ (\[aq]\-\[aq]\ for\ stdin) \-\-load\-info\ FILE\ \ \ \ \ \ \ \ \ \ \ json\ file\ containing\ the\ video\ information -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ (created\ with\ the\ "\-\-write\-json"\ option +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ (created\ with\ the\ "\-\-write\-json"\ option) \-w,\ \-\-no\-overwrites\ \ \ \ \ \ \ \ do\ not\ overwrite\ files \-c,\ \-\-continue\ \ \ \ \ \ \ \ \ \ \ \ \ force\ resume\ of\ partially\ downloaded\ files.\ By \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ default,\ youtube\-dl\ will\ resume\ downloads\ if @@ -159,7 +165,7 @@ redistribute it or use it however you like. \-\-no\-progress\ \ \ \ \ \ \ \ \ \ \ \ \ \ do\ not\ print\ progress\ bar \-\-console\-title\ \ \ \ \ \ \ \ \ \ \ \ display\ progress\ in\ console\ titlebar \-v,\ \-\-verbose\ \ \ \ \ \ \ \ \ \ \ \ \ \ print\ various\ debugging\ information -\-\-dump\-intermediate\-pages\ \ print\ downloaded\ pages\ to\ debug\ problems(very +\-\-dump\-intermediate\-pages\ \ print\ downloaded\ pages\ to\ debug\ problems\ (very \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ verbose) \-\-write\-pages\ \ \ \ \ \ \ \ \ \ \ \ \ \ Write\ downloaded\ intermediary\ pages\ to\ files\ in \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ the\ current\ directory\ to\ debug\ problems @@ -176,8 +182,7 @@ redistribute it or use it however you like. \-\-prefer\-free\-formats\ \ \ \ \ \ prefer\ free\ video\ formats\ unless\ a\ specific\ one \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ is\ requested \-\-max\-quality\ FORMAT\ \ \ \ \ \ \ highest\ quality\ format\ to\ download -\-F,\ \-\-list\-formats\ \ \ \ \ \ \ \ \ list\ all\ available\ formats\ (currently\ youtube -\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ only) +\-F,\ \-\-list\-formats\ \ \ \ \ \ \ \ \ list\ all\ available\ formats \f[] .fi .SS Subtitle Options: @@ -203,7 +208,7 @@ redistribute it or use it however you like. \-u,\ \-\-username\ USERNAME\ \ \ \ account\ username \-p,\ \-\-password\ PASSWORD\ \ \ \ account\ password \-n,\ \-\-netrc\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ use\ .netrc\ authentication\ data -\-\-video\-password\ PASSWORD\ \ video\ password\ (vimeo\ only) +\-\-video\-password\ PASSWORD\ \ video\ password\ (vimeo,\ smotri) \f[] .fi .SS Post\-processing Options: @@ -225,7 +230,13 @@ redistribute it or use it however you like. \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ processed\ files\ are\ overwritten\ by\ default \-\-embed\-subs\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ embed\ subtitles\ in\ the\ video\ (only\ for\ mp4 \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ videos) -\-\-add\-metadata\ \ \ \ \ \ \ \ \ \ \ \ \ add\ metadata\ to\ the\ files +\-\-add\-metadata\ \ \ \ \ \ \ \ \ \ \ \ \ write\ metadata\ to\ the\ video\ file +\-\-xattrs\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ write\ metadata\ to\ the\ video\ file\[aq]s\ xattrs\ (using +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ dublin\ core\ and\ xdg\ standards) +\-\-prefer\-avconv\ \ \ \ \ \ \ \ \ \ \ \ Prefer\ avconv\ over\ ffmpeg\ for\ running\ the +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ postprocessors\ (default) +\-\-prefer\-ffmpeg\ \ \ \ \ \ \ \ \ \ \ \ Prefer\ ffmpeg\ over\ avconv\ for\ running\ the +\ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ postprocessors \f[] .fi .SH CONFIGURATION @@ -306,14 +317,12 @@ Relative dates: Dates in the format \f[C](now|today)[+\-][0\-9](day|week|month|year)(s)?\f[] .PP Examples: -.IP -.nf -\f[C] -$\ youtube\-dl\ \-\-dateafter\ now\-6months\ #will\ only\ download\ the\ videos\ uploaded\ in\ the\ last\ 6\ months -$\ youtube\-dl\ \-\-date\ 19700101\ #will\ only\ download\ the\ videos\ uploaded\ in\ January\ 1,\ 1970 -$\ youtube\-dl\ \-\-dateafter\ 20000101\ \-\-datebefore\ 20100101\ #will\ only\ download\ the\ videos\ uploaded\ between\ 2000\ and\ 2010 -\f[] -.fi +.PP +$ # Download only the videos uploaded in the last 6 months $ youtube\-dl +\-\-dateafter now\-6months $ # Download only the videos uploaded on +January 1, 1970 $ youtube\-dl \-\-date 19700101 $ # will only download +the videos uploaded in the 200x decade $ youtube\-dl \-\-dateafter +20000101 \-\-datebefore 20091231 .SH FAQ .SS Can you please put the \-b option back? .PP @@ -451,7 +460,7 @@ http://www.youtube.com/ ) is \f[I]not\f[] an example URL. .PP Before reporting any issue, type youtube\-dl \-U. This should report that you\[aq]re up\-to\-date. -Ábout 20% of the reports we receive are already fixed, but people are +About 20% of the reports we receive are already fixed, but people are using outdated versions. This goes for feature requests as well. .SS Is the issue already documented? @@ -521,3 +530,14 @@ talk to) require. Do not post features because they seem like a good idea. If they are really useful, they will be requested by someone who requires them. +.SS Is your question about youtube\-dl? +.PP +It may sound strange, but some bug reports we receive are completely +unrelated to youtube\-dl and relate to a different or even the +reporter\[aq]s own application. +Please make sure that you are actually using youtube\-dl. +If you are using a UI for youtube\-dl, report the bug to the maintainer +of the actual application providing the UI. +On the other hand, if your UI for youtube\-dl fails in some way you +believe is related to youtube\-dl, by all means, go ahead and report the +bug. diff --git a/youtube-dl.bash-completion b/youtube-dl.bash-completion index 0cb7e92..6346285 100644 --- a/youtube-dl.bash-completion +++ b/youtube-dl.bash-completion @@ -4,9 +4,9 @@ __youtube_dl() COMPREPLY=() cur="${COMP_WORDS[COMP_CWORD]}" prev="${COMP_WORDS[COMP_CWORD-1]}" - opts="--help --version --update --ignore-errors --abort-on-error --dump-user-agent --user-agent --referer --list-extractors --extractor-descriptions --proxy --no-check-certificate --cache-dir --no-cache-dir --socket-timeout --bidi-workaround --playlist-start --playlist-end --match-title --reject-title --max-downloads --min-filesize --max-filesize --date --datebefore --dateafter --min-views --max-views --no-playlist --age-limit --download-archive --rate-limit --retries --buffer-size --no-resize-buffer --test --title --id --literal --auto-number --output --autonumber-size --restrict-filenames --batch-file --load-info --no-overwrites --continue --no-continue --cookies --no-part --no-mtime --write-description --write-info-json --write-annotations --write-thumbnail --quiet --simulate --skip-download --get-url --get-title --get-id --get-thumbnail --get-description --get-duration --get-filename --get-format --dump-json --newline --no-progress --console-title --verbose --dump-intermediate-pages --write-pages --youtube-print-sig-code --format --all-formats --prefer-free-formats --max-quality --list-formats --write-sub --write-auto-sub --all-subs --list-subs --sub-format --sub-lang --username --password --netrc --video-password --extract-audio --audio-format --audio-quality --recode-video --keep-video --no-post-overwrites --embed-subs --add-metadata" + opts="--help --version --update --ignore-errors --abort-on-error --dump-user-agent --user-agent --referer --list-extractors --extractor-descriptions --proxy --no-check-certificate --cache-dir --no-cache-dir --socket-timeout --bidi-workaround --playlist-start --playlist-end --match-title --reject-title --max-downloads --min-filesize --max-filesize --date --datebefore --dateafter --min-views --max-views --no-playlist --age-limit --download-archive --rate-limit --retries --buffer-size --no-resize-buffer --test --title --id --literal --auto-number --output --autonumber-size --restrict-filenames --batch-file --load-info --no-overwrites --continue --no-continue --cookies --no-part --no-mtime --write-description --write-info-json --write-annotations --write-thumbnail --quiet --simulate --skip-download --get-url --get-title --get-id --get-thumbnail --get-description --get-duration --get-filename --get-format --dump-json --newline --no-progress --console-title --verbose --dump-intermediate-pages --write-pages --youtube-print-sig-code --print-traffic --format --all-formats --prefer-free-formats --max-quality --list-formats --write-sub --write-auto-sub --all-subs --list-subs --sub-format --sub-lang --username --password --netrc --video-password --extract-audio --audio-format --audio-quality --recode-video --keep-video --no-post-overwrites --embed-subs --add-metadata --xattrs --prefer-avconv --prefer-ffmpeg" keywords=":ytfavorites :ytrecommended :ytsubscriptions :ytwatchlater :ythistory" - fileopts="-a|--batch-file|--download-archive|--cookies" + fileopts="-a|--batch-file|--download-archive|--cookies|--load-info" diropts="--cache-dir" if [[ ${prev} =~ ${fileopts} ]]; then diff --git a/youtube_dl/FileDownloader.py b/youtube_dl/FileDownloader.py index 4712493..5c8e676 100644 --- a/youtube_dl/FileDownloader.py +++ b/youtube_dl/FileDownloader.py @@ -1,724 +1,12 @@ -import os -import re -import subprocess -import sys -import time - -from .utils import ( - compat_urllib_error, - compat_urllib_request, - ContentTooShortError, - determine_ext, - encodeFilename, - format_bytes, - sanitize_open, - timeconvert, -) - - -class FileDownloader(object): - """File Downloader class. - - File downloader objects are the ones responsible of downloading the - actual video file and writing it to disk. - - File downloaders accept a lot of parameters. In order not to saturate - the object constructor with arguments, it receives a dictionary of - options instead. - - Available options: - - verbose: Print additional info to stdout. - quiet: Do not print messages to stdout. - ratelimit: Download speed limit, in bytes/sec. - retries: Number of times to retry for HTTP error 5xx - buffersize: Size of download buffer in bytes. - noresizebuffer: Do not automatically resize the download buffer. - continuedl: Try to continue downloads if possible. - noprogress: Do not print the progress bar. - logtostderr: Log messages to stderr instead of stdout. - consoletitle: Display progress in console window's titlebar. - nopart: Do not use temporary .part files. - updatetime: Use the Last-modified header to set output file timestamps. - test: Download only first bytes to test the downloader. - min_filesize: Skip files smaller than this size - max_filesize: Skip files larger than this size - """ - - params = None - - def __init__(self, ydl, params): - """Create a FileDownloader object with the given options.""" - self.ydl = ydl - self._progress_hooks = [] - self.params = params - - @staticmethod - def format_seconds(seconds): - (mins, secs) = divmod(seconds, 60) - (hours, mins) = divmod(mins, 60) - if hours > 99: - return '--:--:--' - if hours == 0: - return '%02d:%02d' % (mins, secs) - else: - return '%02d:%02d:%02d' % (hours, mins, secs) - - @staticmethod - def calc_percent(byte_counter, data_len): - if data_len is None: - return None - return float(byte_counter) / float(data_len) * 100.0 - - @staticmethod - def format_percent(percent): - if percent is None: - return '---.-%' - return '%6s' % ('%3.1f%%' % percent) - - @staticmethod - def calc_eta(start, now, total, current): - if total is None: - return None - dif = now - start - if current == 0 or dif < 0.001: # One millisecond - return None - rate = float(current) / dif - return int((float(total) - float(current)) / rate) - - @staticmethod - def format_eta(eta): - if eta is None: - return '--:--' - return FileDownloader.format_seconds(eta) - - @staticmethod - def calc_speed(start, now, bytes): - dif = now - start - if bytes == 0 or dif < 0.001: # One millisecond - return None - return float(bytes) / dif - - @staticmethod - def format_speed(speed): - if speed is None: - return '%10s' % '---b/s' - return '%10s' % ('%s/s' % format_bytes(speed)) - - @staticmethod - def best_block_size(elapsed_time, bytes): - new_min = max(bytes / 2.0, 1.0) - new_max = min(max(bytes * 2.0, 1.0), 4194304) # Do not surpass 4 MB - if elapsed_time < 0.001: - return int(new_max) - rate = bytes / elapsed_time - if rate > new_max: - return int(new_max) - if rate < new_min: - return int(new_min) - return int(rate) - - @staticmethod - def parse_bytes(bytestr): - """Parse a string indicating a byte quantity into an integer.""" - matchobj = re.match(r'(?i)^(\d+(?:\.\d+)?)([kMGTPEZY]?)$', bytestr) - if matchobj is None: - return None - number = float(matchobj.group(1)) - multiplier = 1024.0 ** 'bkmgtpezy'.index(matchobj.group(2).lower()) - return int(round(number * multiplier)) - - def to_screen(self, *args, **kargs): - self.ydl.to_screen(*args, **kargs) - - def to_stderr(self, message): - self.ydl.to_screen(message) - - def to_console_title(self, message): - self.ydl.to_console_title(message) - - def trouble(self, *args, **kargs): - self.ydl.trouble(*args, **kargs) - - def report_warning(self, *args, **kargs): - self.ydl.report_warning(*args, **kargs) - - def report_error(self, *args, **kargs): - self.ydl.report_error(*args, **kargs) - - 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 temp_name(self, filename): - """Returns a temporary filename for the given filename.""" - if self.params.get('nopart', False) or filename == u'-' or \ - (os.path.exists(encodeFilename(filename)) and not os.path.isfile(encodeFilename(filename))): - return filename - return filename + u'.part' - - def undo_temp_name(self, filename): - if filename.endswith(u'.part'): - return filename[:-len(u'.part')] - return filename - - def try_rename(self, old_filename, new_filename): - try: - if old_filename == new_filename: - return - os.rename(encodeFilename(old_filename), encodeFilename(new_filename)) - except (IOError, OSError): - self.report_error(u'unable to rename file') - - def try_utime(self, filename, last_modified_hdr): - """Try to set the last-modified time of the given file.""" - if last_modified_hdr is None: - return - if not os.path.isfile(encodeFilename(filename)): - return - timestr = last_modified_hdr - if timestr is None: - return - filetime = timeconvert(timestr) - if filetime is None: - return filetime - # Ignore obviously invalid dates - if filetime == 0: - return - try: - os.utime(filename, (time.time(), filetime)) - except: - pass - return filetime - - def report_destination(self, filename): - """Report destination filename.""" - self.to_screen(u'[download] Destination: ' + filename) - - def _report_progress_status(self, msg, is_last_line=False): - fullmsg = u'[download] ' + msg - if self.params.get('progress_with_newline', False): - self.to_screen(fullmsg) - else: - if os.name == 'nt': - prev_len = getattr(self, '_report_progress_prev_line_length', - 0) - if prev_len > len(fullmsg): - fullmsg += u' ' * (prev_len - len(fullmsg)) - self._report_progress_prev_line_length = len(fullmsg) - clear_line = u'\r' - else: - clear_line = (u'\r\x1b[K' if sys.stderr.isatty() else u'\r') - self.to_screen(clear_line + fullmsg, skip_eol=not is_last_line) - self.to_console_title(u'youtube-dl ' + msg) - - def report_progress(self, percent, data_len_str, speed, eta): - """Report download progress.""" - if self.params.get('noprogress', False): - return - 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) - - msg = (u'%s of %s at %s ETA %s' % - (percent_str, data_len_str, speed_str, eta_str)) - self._report_progress_status(msg) - - def report_progress_live_stream(self, downloaded_data_len, speed, elapsed): - if self.params.get('noprogress', False): - return - downloaded_str = format_bytes(downloaded_data_len) - speed_str = self.format_speed(speed) - elapsed_str = FileDownloader.format_seconds(elapsed) - msg = u'%s at %s (%s)' % (downloaded_str, speed_str, elapsed_str) - self._report_progress_status(msg) - - def report_finish(self, data_len_str, tot_time): - """Report download finished.""" - if self.params.get('noprogress', False): - self.to_screen(u'[download] Download completed') - else: - self._report_progress_status( - (u'100%% of %s in %s' % - (data_len_str, self.format_seconds(tot_time))), - is_last_line=True) - - def report_resuming_byte(self, resume_len): - """Report attempt to resume at given byte.""" - self.to_screen(u'[download] Resuming download at byte %s' % resume_len) - - def report_retry(self, count, retries): - """Report retry in case of HTTP error 5xx""" - self.to_screen(u'[download] Got server HTTP error. Retrying (attempt %d of %d)...' % (count, retries)) - - def report_file_already_downloaded(self, file_name): - """Report file has already been fully downloaded.""" - try: - self.to_screen(u'[download] %s has already been downloaded' % file_name) - except UnicodeEncodeError: - self.to_screen(u'[download] The file has already been downloaded') - - def report_unable_to_resume(self): - """Report it was impossible to resume download.""" - self.to_screen(u'[download] Unable to resume') - - def _download_with_rtmpdump(self, filename, url, player_url, page_url, play_path, tc_url, live, conn): - def run_rtmpdump(args): - start = time.time() - resume_percent = None - resume_downloaded_data_len = None - proc = subprocess.Popen(args, stderr=subprocess.PIPE) - cursor_in_new_line = True - proc_stderr_closed = False - while not proc_stderr_closed: - # read line from stderr - line = u'' - while True: - char = proc.stderr.read(1) - if not char: - proc_stderr_closed = True - break - if char in [b'\r', b'\n']: - break - line += char.decode('ascii', 'replace') - if not line: - # proc_stderr_closed is True - continue - mobj = re.search(r'([0-9]+\.[0-9]{3}) kB / [0-9]+\.[0-9]{2} sec \(([0-9]{1,2}\.[0-9])%\)', line) - if mobj: - downloaded_data_len = int(float(mobj.group(1))*1024) - percent = float(mobj.group(2)) - if not resume_percent: - resume_percent = percent - resume_downloaded_data_len = downloaded_data_len - eta = self.calc_eta(start, time.time(), 100-resume_percent, percent-resume_percent) - speed = self.calc_speed(start, time.time(), downloaded_data_len-resume_downloaded_data_len) - data_len = None - if percent > 0: - data_len = int(downloaded_data_len * 100 / percent) - data_len_str = u'~' + format_bytes(data_len) - self.report_progress(percent, data_len_str, speed, eta) - cursor_in_new_line = False - self._hook_progress({ - 'downloaded_bytes': downloaded_data_len, - 'total_bytes': data_len, - 'tmpfilename': tmpfilename, - 'filename': filename, - 'status': 'downloading', - 'eta': eta, - 'speed': speed, - }) - else: - # no percent for live streams - mobj = re.search(r'([0-9]+\.[0-9]{3}) kB / [0-9]+\.[0-9]{2} sec', line) - if mobj: - downloaded_data_len = int(float(mobj.group(1))*1024) - time_now = time.time() - speed = self.calc_speed(start, time_now, downloaded_data_len) - self.report_progress_live_stream(downloaded_data_len, speed, time_now - start) - cursor_in_new_line = False - self._hook_progress({ - 'downloaded_bytes': downloaded_data_len, - 'tmpfilename': tmpfilename, - 'filename': filename, - 'status': 'downloading', - 'speed': speed, - }) - elif self.params.get('verbose', False): - if not cursor_in_new_line: - self.to_screen(u'') - cursor_in_new_line = True - self.to_screen(u'[rtmpdump] '+line) - proc.wait() - if not cursor_in_new_line: - self.to_screen(u'') - return proc.returncode - - self.report_destination(filename) - tmpfilename = self.temp_name(filename) - test = self.params.get('test', False) - - # Check for rtmpdump first - try: - subprocess.call(['rtmpdump', '-h'], stdout=(open(os.path.devnull, 'w')), stderr=subprocess.STDOUT) - except (OSError, IOError): - self.report_error(u'RTMP download detected but "rtmpdump" could not be run') - return False - - # Download using rtmpdump. rtmpdump returns exit code 2 when - # the connection was interrumpted and resuming appears to be - # possible. This is part of rtmpdump's normal usage, AFAIK. - basic_args = ['rtmpdump', '--verbose', '-r', url, '-o', tmpfilename] - if player_url is not None: - basic_args += ['--swfVfy', player_url] - if page_url is not None: - basic_args += ['--pageUrl', page_url] - if play_path is not None: - basic_args += ['--playpath', play_path] - if tc_url is not None: - basic_args += ['--tcUrl', url] - if test: - basic_args += ['--stop', '1'] - if live: - basic_args += ['--live'] - if conn: - basic_args += ['--conn', conn] - args = basic_args + [[], ['--resume', '--skip', '1']][self.params.get('continuedl', False)] - - if sys.platform == 'win32' and sys.version_info < (3, 0): - # Windows subprocess module does not actually support Unicode - # on Python 2.x - # See http://stackoverflow.com/a/9951851/35070 - subprocess_encoding = sys.getfilesystemencoding() - args = [a.encode(subprocess_encoding, 'ignore') for a in args] - else: - subprocess_encoding = None - - if self.params.get('verbose', False): - if subprocess_encoding: - str_args = [ - a.decode(subprocess_encoding) if isinstance(a, bytes) else a - for a in args] - else: - str_args = args - try: - import pipes - shell_quote = lambda args: ' '.join(map(pipes.quote, str_args)) - except ImportError: - shell_quote = repr - self.to_screen(u'[debug] rtmpdump command line: ' + shell_quote(str_args)) - - retval = run_rtmpdump(args) - - while (retval == 2 or retval == 1) and not test: - prevsize = os.path.getsize(encodeFilename(tmpfilename)) - self.to_screen(u'[rtmpdump] %s bytes' % prevsize) - time.sleep(5.0) # This seems to be needed - retval = run_rtmpdump(basic_args + ['-e'] + [[], ['-k', '1']][retval == 1]) - cursize = os.path.getsize(encodeFilename(tmpfilename)) - if prevsize == cursize and retval == 1: - break - # Some rtmp streams seem abort after ~ 99.8%. Don't complain for those - if prevsize == cursize and retval == 2 and cursize > 1024: - self.to_screen(u'[rtmpdump] Could not download the whole video. This can happen for some advertisements.') - retval = 0 - break - if retval == 0 or (test and retval == 2): - fsize = os.path.getsize(encodeFilename(tmpfilename)) - self.to_screen(u'[rtmpdump] %s bytes' % fsize) - self.try_rename(tmpfilename, filename) - self._hook_progress({ - 'downloaded_bytes': fsize, - 'total_bytes': fsize, - 'filename': filename, - 'status': 'finished', - }) - return True - else: - self.to_stderr(u"\n") - self.report_error(u'rtmpdump exited with code %d' % retval) - return False - - def _download_with_mplayer(self, filename, url): - self.report_destination(filename) - tmpfilename = self.temp_name(filename) - - args = ['mplayer', '-really-quiet', '-vo', 'null', '-vc', 'dummy', '-dumpstream', '-dumpfile', tmpfilename, url] - # Check for mplayer first - try: - subprocess.call(['mplayer', '-h'], stdout=(open(os.path.devnull, 'w')), stderr=subprocess.STDOUT) - except (OSError, IOError): - self.report_error(u'MMS or RTSP download detected but "%s" could not be run' % args[0] ) - return False - - # Download using mplayer. - retval = subprocess.call(args) - if retval == 0: - fsize = os.path.getsize(encodeFilename(tmpfilename)) - self.to_screen(u'\r[%s] %s bytes' % (args[0], fsize)) - self.try_rename(tmpfilename, filename) - self._hook_progress({ - 'downloaded_bytes': fsize, - 'total_bytes': fsize, - 'filename': filename, - 'status': 'finished', - }) - return True - else: - self.to_stderr(u"\n") - self.report_error(u'mplayer exited with code %d' % retval) - return False - - def _download_m3u8_with_ffmpeg(self, filename, url): - self.report_destination(filename) - tmpfilename = self.temp_name(filename) - - args = ['-y', '-i', url, '-f', 'mp4', '-c', 'copy', - '-bsf:a', 'aac_adtstoasc', tmpfilename] - - 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)) - self.try_rename(tmpfilename, filename) - self._hook_progress({ - 'downloaded_bytes': fsize, - 'total_bytes': fsize, - 'filename': filename, - 'status': 'finished', - }) - return True - else: - self.to_stderr(u"\n") - self.report_error(u'ffmpeg exited with code %d' % retval) - return False +# Legacy file for backwards compatibility, use youtube_dl.downloader instead! +from .downloader import FileDownloader as RealFileDownloader +from .downloader import get_suitable_downloader +# This class reproduces the old behaviour of FileDownloader +class FileDownloader(RealFileDownloader): def _do_download(self, filename, info_dict): - url = info_dict['url'] - - # Check file already present - if self.params.get('continuedl', False) and os.path.isfile(encodeFilename(filename)) and not self.params.get('nopart', False): - self.report_file_already_downloaded(filename) - self._hook_progress({ - 'filename': filename, - 'status': 'finished', - 'total_bytes': os.path.getsize(encodeFilename(filename)), - }) - return True - - # Attempt to download using rtmpdump - if url.startswith('rtmp'): - return self._download_with_rtmpdump(filename, url, - 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('rtmp_live', False), - info_dict.get('rtmp_conn', None)) - - # Attempt to download using mplayer - if url.startswith('mms') or url.startswith('rtsp'): - return self._download_with_mplayer(filename, url) - - # m3u8 manifest are downloaded with ffmpeg - if determine_ext(url) == u'm3u8': - return self._download_m3u8_with_ffmpeg(filename, url) - - tmpfilename = self.temp_name(filename) - stream = None - - # Do not include the Accept-Encoding header - headers = {'Youtubedl-no-compression': 'True'} - if 'user_agent' in info_dict: - headers['Youtubedl-user-agent'] = info_dict['user_agent'] - basic_request = compat_urllib_request.Request(url, None, headers) - request = compat_urllib_request.Request(url, None, headers) - - if self.params.get('test', False): - request.add_header('Range','bytes=0-10240') - - # Establish possible resume length - if os.path.isfile(encodeFilename(tmpfilename)): - resume_len = os.path.getsize(encodeFilename(tmpfilename)) - else: - resume_len = 0 - - open_mode = 'wb' - if resume_len != 0: - if self.params.get('continuedl', False): - self.report_resuming_byte(resume_len) - request.add_header('Range','bytes=%d-' % resume_len) - open_mode = 'ab' - else: - resume_len = 0 - - count = 0 - retries = self.params.get('retries', 0) - while count <= retries: - # Establish connection - try: - if count == 0 and 'urlhandle' in info_dict: - data = info_dict['urlhandle'] - data = compat_urllib_request.urlopen(request) - break - except (compat_urllib_error.HTTPError, ) as err: - if (err.code < 500 or err.code >= 600) and err.code != 416: - # Unexpected HTTP error - raise - elif err.code == 416: - # Unable to resume (requested range not satisfiable) - try: - # Open the connection again without the range header - data = compat_urllib_request.urlopen(basic_request) - content_length = data.info()['Content-Length'] - except (compat_urllib_error.HTTPError, ) as err: - if err.code < 500 or err.code >= 600: - raise - else: - # Examine the reported length - if (content_length is not None and - (resume_len - 100 < int(content_length) < resume_len + 100)): - # The file had already been fully downloaded. - # Explanation to the above condition: in issue #175 it was revealed that - # YouTube sometimes adds or removes a few bytes from the end of the file, - # changing the file size slightly and causing problems for some users. So - # I decided to implement a suggested change and consider the file - # completely downloaded if the file size differs less than 100 bytes from - # the one in the hard drive. - self.report_file_already_downloaded(filename) - self.try_rename(tmpfilename, filename) - self._hook_progress({ - 'filename': filename, - 'status': 'finished', - }) - return True - else: - # The length does not match, we start the download over - self.report_unable_to_resume() - open_mode = 'wb' - break - # Retry - count += 1 - if count <= retries: - self.report_retry(count, retries) - - if count > retries: - self.report_error(u'giving up after %s retries' % retries) - return False - - data_len = data.info().get('Content-length', None) - if data_len is not None: - data_len = int(data_len) + resume_len - min_data_len = self.params.get("min_filesize", None) - max_data_len = self.params.get("max_filesize", None) - if min_data_len is not None and data_len < min_data_len: - self.to_screen(u'\r[download] File is smaller than min-filesize (%s bytes < %s bytes). Aborting.' % (data_len, min_data_len)) - return False - if max_data_len is not None and data_len > max_data_len: - self.to_screen(u'\r[download] File is larger than max-filesize (%s bytes > %s bytes). Aborting.' % (data_len, max_data_len)) - return False - - data_len_str = format_bytes(data_len) - byte_counter = 0 + resume_len - block_size = self.params.get('buffersize', 1024) - start = time.time() - while True: - # Download and write - before = time.time() - data_block = data.read(block_size) - after = time.time() - if len(data_block) == 0: - break - byte_counter += len(data_block) - - # Open file just in time - if stream is None: - try: - (stream, tmpfilename) = sanitize_open(tmpfilename, open_mode) - assert stream is not None - filename = self.undo_temp_name(tmpfilename) - self.report_destination(filename) - except (OSError, IOError) as err: - self.report_error(u'unable to open for writing: %s' % str(err)) - return False - try: - stream.write(data_block) - except (IOError, OSError) as err: - self.to_stderr(u"\n") - self.report_error(u'unable to write data: %s' % str(err)) - return False - if not self.params.get('noresizebuffer', False): - block_size = self.best_block_size(after - before, len(data_block)) - - # Progress message - speed = self.calc_speed(start, time.time(), byte_counter - resume_len) - if data_len is 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._hook_progress({ - 'downloaded_bytes': byte_counter, - 'total_bytes': data_len, - 'tmpfilename': tmpfilename, - 'filename': filename, - 'status': 'downloading', - 'eta': eta, - 'speed': speed, - }) - - # Apply rate limit - self.slow_down(start, byte_counter - resume_len) - - if stream is None: - self.to_stderr(u"\n") - self.report_error(u'Did not get any data blocks') - return False - stream.close() - self.report_finish(data_len_str, (time.time() - start)) - if data_len is not None and byte_counter != data_len: - raise ContentTooShortError(byte_counter, int(data_len)) - self.try_rename(tmpfilename, filename) - - # Update file modification time - if self.params.get('updatetime', True): - info_dict['filetime'] = self.try_utime(filename, data.info().get('last-modified', None)) - - self._hook_progress({ - 'downloaded_bytes': byte_counter, - 'total_bytes': byte_counter, - 'filename': filename, - 'status': 'finished', - }) - - return True - - def _hook_progress(self, status): + real_fd = get_suitable_downloader(info_dict)(self.ydl, self.params) for ph in self._progress_hooks: - ph(status) - - def add_progress_hook(self, ph): - """ ph gets called on download progress, with a dictionary with the entries - * filename: The final filename - * status: One of "downloading" and "finished" - - It can also have some of the following entries: - - * downloaded_bytes: Bytes on disks - * total_bytes: Total bytes, None if unknown - * tmpfilename: The filename we're currently writing to - * eta: The estimated time in seconds, None if unknown - * speed: The download speed in bytes/second, None if unknown - - Hooks are guaranteed to be called at least once (with status "finished") - if the download is successful. - """ - self._progress_hooks.append(ph) + real_fd.add_progress_hook(ph) + return real_fd.download(filename, info_dict) diff --git a/youtube_dl/YoutubeDL.py b/youtube_dl/YoutubeDL.py index 2a078ad..d40314e 100644 --- a/youtube_dl/YoutubeDL.py +++ b/youtube_dl/YoutubeDL.py @@ -1,7 +1,7 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- -from __future__ import absolute_import +from __future__ import absolute_import, unicode_literals import collections import errno @@ -51,9 +51,11 @@ from .utils import ( write_json_file, write_string, YoutubeDLHandler, + prepend_extension, ) from .extractor import get_info_extractor, gen_extractors -from .FileDownloader import FileDownloader +from .downloader import get_suitable_downloader +from .postprocessor import FFmpegMergerPP from .version import __version__ @@ -148,11 +150,16 @@ class YoutubeDL(object): socket_timeout: Time to wait for unresponsive hosts, in seconds bidi_workaround: Work around buggy terminals without bidirectional text support, using fridibi + debug_printtraffic:Print out sent and received HTTP traffic The following parameters are not used by YoutubeDL itself, they are used by the FileDownloader: nopart, updatetime, buffersize, ratelimit, min_filesize, max_filesize, test, noresizebuffer, retries, continuedl, noprogress, consoletitle + + The following options are used by the post processors: + prefer_ffmpeg: If True, use ffmpeg instead of avconv if both are available, + otherwise prefer avconv. """ params = None @@ -164,6 +171,8 @@ class YoutubeDL(object): def __init__(self, params=None): """Create a FileDownloader object with the given options.""" + if params is None: + params = {} self._ies = [] self._ies_instances = {} self._pps = [] @@ -172,7 +181,7 @@ class YoutubeDL(object): self._num_downloads = 0 self._screen_file = [sys.stdout, sys.stderr][params.get('logtostderr', False)] self._err_file = sys.stderr - self.params = {} if params is None else params + self.params = params if params.get('bidi_workaround', False): try: @@ -183,15 +192,21 @@ class YoutubeDL(object): width_args = [] else: width_args = ['-w', str(width)] - self._fribidi = subprocess.Popen( - ['fribidi', '-c', 'UTF-8'] + width_args, + sp_kwargs = dict( stdin=subprocess.PIPE, stdout=slave, stderr=self._err_file) - self._fribidi_channel = os.fdopen(master, 'rb') + try: + self._output_process = subprocess.Popen( + ['bidiv'] + width_args, **sp_kwargs + ) + except OSError: + self._output_process = subprocess.Popen( + ['fribidi', '-c', 'UTF-8'] + width_args, **sp_kwargs) + self._output_channel = os.fdopen(master, 'rb') except OSError as ose: if ose.errno == 2: - self.report_warning(u'Could not find fribidi executable, ignoring --bidi-workaround . Make sure that fribidi is an executable file in one of the directories in your $PATH.') + self.report_warning('Could not find fribidi executable, ignoring --bidi-workaround . Make sure that fribidi is an executable file in one of the directories in your $PATH.') else: raise @@ -200,15 +215,13 @@ class YoutubeDL(object): and not params['restrictfilenames']): # On Python 3, the Unicode filesystem API will throw errors (#1474) self.report_warning( - u'Assuming --restrict-filenames since file system encoding ' - u'cannot encode all charactes. ' - u'Set the LC_ALL environment variable to fix this.') + 'Assuming --restrict-filenames since file system encoding ' + 'cannot encode all charactes. ' + 'Set the LC_ALL environment variable to fix this.') self.params['restrictfilenames'] = True - self.fd = FileDownloader(self, self.params) - if '%(stitle)s' in self.params.get('outtmpl', ''): - self.report_warning(u'%(stitle)s is deprecated. Use the %(title)s and the --restrict-filenames flag(which also secures %(uploader)s et al) instead.') + self.report_warning('%(stitle)s is deprecated. Use the %(title)s and the --restrict-filenames flag(which also secures %(uploader)s et al) instead.') self._setup_opener() @@ -242,17 +255,22 @@ class YoutubeDL(object): self._pps.append(pp) pp.set_downloader(self) + def add_progress_hook(self, ph): + """Add the progress hook (currently only for the file downloader)""" + self._progress_hooks.append(ph) + def _bidi_workaround(self, message): - if not hasattr(self, '_fribidi_channel'): + if not hasattr(self, '_output_channel'): return message - assert type(message) == type(u'') - line_count = message.count(u'\n') + 1 - self._fribidi.stdin.write((message + u'\n').encode('utf-8')) - self._fribidi.stdin.flush() - res = u''.join(self._fribidi_channel.readline().decode('utf-8') + assert hasattr(self, '_output_process') + assert type(message) == type('') + line_count = message.count('\n') + 1 + self._output_process.stdin.write((message + '\n').encode('utf-8')) + self._output_process.stdin.flush() + res = ''.join(self._output_channel.readline().decode('utf-8') for _ in range(line_count)) - return res[:-len(u'\n')] + return res[:-len('\n')] def to_screen(self, message, skip_eol=False): """Print message to stdout if not in quiet mode.""" @@ -264,19 +282,19 @@ class YoutubeDL(object): self.params['logger'].debug(message) elif not check_quiet or not self.params.get('quiet', False): message = self._bidi_workaround(message) - terminator = [u'\n', u''][skip_eol] + terminator = ['\n', ''][skip_eol] output = message + terminator write_string(output, self._screen_file) def to_stderr(self, message): """Print message to stderr.""" - assert type(message) == type(u'') + assert type(message) == type('') if self.params.get('logger'): self.params['logger'].error(message) else: message = self._bidi_workaround(message) - output = message + u'\n' + output = message + '\n' write_string(output, self._err_file) def to_console_title(self, message): @@ -287,21 +305,21 @@ class YoutubeDL(object): # 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) + write_string('\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) + write_string('\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) + write_string('\033[23;0t', self._screen_file) def __enter__(self): self.save_console_title() @@ -327,13 +345,13 @@ class YoutubeDL(object): if self.params.get('verbose'): if tb is None: if sys.exc_info()[0]: # if .trouble has been called from an except block - tb = u'' + tb = '' if hasattr(sys.exc_info()[1], 'exc_info') and sys.exc_info()[1].exc_info[0]: - tb += u''.join(traceback.format_exception(*sys.exc_info()[1].exc_info)) + tb += ''.join(traceback.format_exception(*sys.exc_info()[1].exc_info)) tb += compat_str(traceback.format_exc()) else: tb_data = traceback.format_list(traceback.extract_stack()) - tb = u''.join(tb_data) + tb = ''.join(tb_data) self.to_stderr(tb) if not self.params.get('ignoreerrors', False): if sys.exc_info()[0] and hasattr(sys.exc_info()[1], 'exc_info') and sys.exc_info()[1].exc_info[0]: @@ -349,10 +367,10 @@ class YoutubeDL(object): If stderr is a tty file the 'WARNING:' will be colored ''' if self._err_file.isatty() and os.name != 'nt': - _msg_header = u'\033[0;33mWARNING:\033[0m' + _msg_header = '\033[0;33mWARNING:\033[0m' else: - _msg_header = u'WARNING:' - warning_message = u'%s %s' % (_msg_header, message) + _msg_header = 'WARNING:' + warning_message = '%s %s' % (_msg_header, message) self.to_stderr(warning_message) def report_error(self, message, tb=None): @@ -361,18 +379,18 @@ class YoutubeDL(object): in red if stderr is a tty file. ''' if self._err_file.isatty() and os.name != 'nt': - _msg_header = u'\033[0;31mERROR:\033[0m' + _msg_header = '\033[0;31mERROR:\033[0m' else: - _msg_header = u'ERROR:' - error_message = u'%s %s' % (_msg_header, message) + _msg_header = 'ERROR:' + error_message = '%s %s' % (_msg_header, message) self.trouble(error_message, tb) def report_file_already_downloaded(self, file_name): """Report file has already been fully downloaded.""" try: - self.to_screen(u'[download] %s has already been downloaded' % file_name) + self.to_screen('[download] %s has already been downloaded' % file_name) except UnicodeEncodeError: - self.to_screen(u'[download] The file has already been downloaded') + self.to_screen('[download] The file has already been downloaded') def increment_downloads(self): """Increment the ordinal that assigns a number to each file.""" @@ -387,61 +405,61 @@ class YoutubeDL(object): autonumber_size = self.params.get('autonumber_size') if autonumber_size is None: autonumber_size = 5 - autonumber_templ = u'%0' + str(autonumber_size) + u'd' + autonumber_templ = '%0' + str(autonumber_size) + 'd' template_dict['autonumber'] = autonumber_templ % self._num_downloads if template_dict.get('playlist_index') is not None: - template_dict['playlist_index'] = u'%05d' % template_dict['playlist_index'] + template_dict['playlist_index'] = '%05d' % template_dict['playlist_index'] sanitize = lambda k, v: sanitize_filename( compat_str(v), restricted=self.params.get('restrictfilenames'), - is_id=(k == u'id')) + is_id=(k == 'id')) template_dict = dict((k, sanitize(k, v)) for k, v in template_dict.items() if v is not None) - template_dict = collections.defaultdict(lambda: u'NA', template_dict) + template_dict = collections.defaultdict(lambda: 'NA', template_dict) tmpl = os.path.expanduser(self.params['outtmpl']) filename = tmpl % template_dict return filename except ValueError as err: - self.report_error(u'Error in output template: ' + str(err) + u' (encoding: ' + repr(preferredencoding()) + ')') + self.report_error('Error in output template: ' + str(err) + ' (encoding: ' + repr(preferredencoding()) + ')') return None def _match_entry(self, info_dict): """ Returns None iff the file should be downloaded """ - video_title = info_dict.get('title', info_dict.get('id', u'video')) + video_title = info_dict.get('title', info_dict.get('id', 'video')) if 'title' in info_dict: # This can happen when we're just evaluating the playlist title = info_dict['title'] matchtitle = self.params.get('matchtitle', False) if matchtitle: if not re.search(matchtitle, title, re.IGNORECASE): - return u'"' + title + '" title did not match pattern "' + matchtitle + '"' + return '"' + title + '" title did not match pattern "' + matchtitle + '"' rejecttitle = self.params.get('rejecttitle', False) if rejecttitle: if re.search(rejecttitle, title, re.IGNORECASE): - return u'"' + title + '" title matched reject pattern "' + rejecttitle + '"' + return '"' + title + '" title matched reject pattern "' + rejecttitle + '"' date = info_dict.get('upload_date', None) if date is not None: dateRange = self.params.get('daterange', DateRange()) if date not in dateRange: - return u'%s upload date is not in range %s' % (date_from_str(date).isoformat(), dateRange) + return '%s upload date is not in range %s' % (date_from_str(date).isoformat(), dateRange) view_count = info_dict.get('view_count', None) if view_count is not None: min_views = self.params.get('min_views') if min_views is not None and view_count < min_views: - return u'Skipping %s, because it has not reached minimum view count (%d/%d)' % (video_title, view_count, min_views) + return 'Skipping %s, because it has not reached minimum view count (%d/%d)' % (video_title, view_count, min_views) max_views = self.params.get('max_views') if max_views is not None and view_count > max_views: - return u'Skipping %s, because it has exceeded the maximum view count (%d/%d)' % (video_title, view_count, max_views) + return 'Skipping %s, because it has exceeded the maximum view count (%d/%d)' % (video_title, view_count, max_views) age_limit = self.params.get('age_limit') if age_limit is not None: if age_limit < info_dict.get('age_limit', 0): - return u'Skipping "' + title + '" because it is age restricted' + return 'Skipping "' + title + '" because it is age restricted' if self.in_download_archive(info_dict): - return u'%s has already been recorded in archive' % video_title + return '%s has already been recorded in archive' % video_title return None @staticmethod @@ -468,8 +486,8 @@ class YoutubeDL(object): continue if not ie.working(): - self.report_warning(u'The program functionality for this site has been marked as broken, ' - u'and will probably not work.') + self.report_warning('The program functionality for this site has been marked as broken, ' + 'and will probably not work.') try: ie_result = ie.extract(url) @@ -502,7 +520,7 @@ class YoutubeDL(object): else: raise else: - self.report_error(u'no suitable InfoExtractor: %s' % url) + self.report_error('no suitable InfoExtractor: %s' % url) def process_ie_result(self, ie_result, download=True, extra_info={}): """ @@ -533,7 +551,7 @@ class YoutubeDL(object): def make_result(embedded_info): new_result = ie_result.copy() for f in ('_type', 'url', 'ext', 'player_url', 'formats', - 'entries', 'urlhandle', 'ie_key', 'duration', + 'entries', 'ie_key', 'duration', 'subtitles', 'annotations', 'format', 'thumbnail', 'thumbnails'): if f in new_result: @@ -553,7 +571,7 @@ class YoutubeDL(object): elif result_type == 'playlist': # 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('[download] Downloading playlist: %s' % playlist) playlist_results = [] @@ -568,11 +586,11 @@ class YoutubeDL(object): n_entries = len(entries) self.to_screen( - u"[%s] playlist '%s': Collected %d video ids (downloading %d of them)" % + "[%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)) + self.to_screen('[download] Downloading video #%s of %s' % (i, n_entries)) extra = { 'playlist': playlist, 'playlist_index': i + playliststart, @@ -584,7 +602,7 @@ class YoutubeDL(object): reason = self._match_entry(entry) if reason is not None: - self.to_screen(u'[download] ' + reason) + self.to_screen('[download] ' + reason) continue entry_result = self.process_ie_result(entry, @@ -617,7 +635,7 @@ class YoutubeDL(object): elif format_spec == 'worst': return available_formats[0] else: - extensions = [u'mp4', u'flv', u'webm', u'3gp'] + extensions = ['mp4', 'flv', 'webm', '3gp'] if format_spec in extensions: filter_f = lambda f: f['ext'] == format_spec else: @@ -636,7 +654,7 @@ class YoutubeDL(object): info_dict['playlist_index'] = None # This extractors handle format selection themselves - if info_dict['extractor'] in [u'youtube', u'Youku']: + if info_dict['extractor'] in ['Youku']: if download: self.process_info(info_dict) return info_dict @@ -653,33 +671,32 @@ class YoutubeDL(object): 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( + format['format'] = '{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 '', + note=' ({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) - return - format_limit = self.params.get('format_limit', None) if format_limit: formats = list(takewhile_inclusive( lambda f: f['format_id'] != format_limit, formats )) - if self.params.get('prefer_free_formats'): - def _free_formats_key(f): - try: - ext_ord = [u'flv', u'mp4', u'webm'].index(f['ext']) - except ValueError: - ext_ord = -1 - # We only compare the extension if they have the same height and width - return (f.get('height'), f.get('width'), ext_ord) - formats = sorted(formats, key=_free_formats_key) + + # TODO Central sorting goes here + + if formats[0] is not info_dict: + # only set the 'formats' fields if the original info_dict list them + # otherwise we end up with a circular reference, the first (and unique) + # element in the 'formats' field in info_dict is info_dict itself, + # wich can't be exported to json + info_dict['formats'] = formats + if self.params.get('listformats', None): + self.list_formats(info_dict) + return req_format = self.params.get('format', 'best') if req_format is None: @@ -689,21 +706,35 @@ class YoutubeDL(object): if req_format in ('-1', 'all'): formats_to_download = formats else: - # We can accept formats requestd in the format: 34/5/best, we pick + # We can accept formats requested 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: - selected_format = self.select_format(rf, formats) + if re.match(r'.+?\+.+?', rf) is not None: + # Two formats have been requested like '137+139' + format_1, format_2 = rf.split('+') + formats_info = (self.select_format(format_1, formats), + self.select_format(format_2, formats)) + if all(formats_info): + selected_format = { + 'requested_formats': formats_info, + 'format': rf, + 'ext': formats_info[0]['ext'], + } + else: + selected_format = None + else: + 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('requested format not available', expected=True) if download: if len(formats_to_download) > 1: - self.to_screen(u'[info] %s: downloading video in %s formats' % (info_dict['id'], len(formats_to_download))) + self.to_screen('[info] %s: downloading video in %s formats' % (info_dict['id'], len(formats_to_download))) for format in formats_to_download: new_info = dict(info_dict) new_info.update(format) @@ -721,7 +752,7 @@ class YoutubeDL(object): info_dict['fulltitle'] = info_dict['title'] if len(info_dict['title']) > 200: - info_dict['title'] = info_dict['title'][:197] + u'...' + info_dict['title'] = info_dict['title'][:197] + '...' # Keep for backwards compatibility info_dict['stitle'] = info_dict['title'] @@ -731,7 +762,7 @@ class YoutubeDL(object): reason = self._match_entry(info_dict) if reason is not None: - self.to_screen(u'[download] ' + reason) + self.to_screen('[download] ' + reason) return max_downloads = self.params.get('max_downloads') @@ -748,7 +779,7 @@ class YoutubeDL(object): self.to_stdout(info_dict['id']) if self.params.get('forceurl', False): # For RTMP URLs, also include the playpath - self.to_stdout(info_dict['url'] + info_dict.get('play_path', u'')) + self.to_stdout(info_dict['url'] + info_dict.get('play_path', '')) if self.params.get('forcethumbnail', False) and info_dict.get('thumbnail') is not None: self.to_stdout(info_dict['thumbnail']) if self.params.get('forcedescription', False) and info_dict.get('description') is not None: @@ -775,37 +806,37 @@ class YoutubeDL(object): if dn != '' and not os.path.exists(dn): os.makedirs(dn) except (OSError, IOError) as err: - self.report_error(u'unable to create directory ' + compat_str(err)) + self.report_error('unable to create directory ' + compat_str(err)) return if self.params.get('writedescription', False): - descfn = filename + u'.description' + descfn = filename + '.description' if self.params.get('nooverwrites', False) and os.path.exists(encodeFilename(descfn)): - self.to_screen(u'[info] Video description is already present') + self.to_screen('[info] Video description is already present') else: try: - self.to_screen(u'[info] Writing video description to: ' + descfn) + self.to_screen('[info] Writing video description to: ' + descfn) with io.open(encodeFilename(descfn), 'w', encoding='utf-8') as descfile: descfile.write(info_dict['description']) except (KeyError, TypeError): - self.report_warning(u'There\'s no description to write.') + self.report_warning('There\'s no description to write.') except (OSError, IOError): - self.report_error(u'Cannot write description file ' + descfn) + self.report_error('Cannot write description file ' + descfn) return if self.params.get('writeannotations', False): - annofn = filename + u'.annotations.xml' + annofn = filename + '.annotations.xml' if self.params.get('nooverwrites', False) and os.path.exists(encodeFilename(annofn)): - self.to_screen(u'[info] Video annotations are already present') + self.to_screen('[info] Video annotations are already present') else: try: - self.to_screen(u'[info] Writing video annotations to: ' + annofn) + self.to_screen('[info] Writing video annotations to: ' + 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.') + self.report_warning('There are no annotations to write.') except (OSError, IOError): - self.report_error(u'Cannot write annotations file: ' + annofn) + self.report_error('Cannot write annotations file: ' + annofn) return subtitles_are_requested = any([self.params.get('writesubtitles', False), @@ -823,46 +854,45 @@ class YoutubeDL(object): try: sub_filename = subtitles_filename(filename, sub_lang, sub_format) if self.params.get('nooverwrites', False) and os.path.exists(encodeFilename(sub_filename)): - self.to_screen(u'[info] Video subtitle %s.%s is already_present' % (sub_lang, sub_format)) + self.to_screen('[info] Video subtitle %s.%s is already_present' % (sub_lang, sub_format)) else: - self.to_screen(u'[info] Writing video subtitles to: ' + sub_filename) + self.to_screen('[info] Writing video subtitles to: ' + sub_filename) with io.open(encodeFilename(sub_filename), 'w', encoding='utf-8') as subfile: subfile.write(sub) except (OSError, IOError): - self.report_error(u'Cannot write subtitles file ' + descfn) + self.report_error('Cannot write subtitles file ' + descfn) return if self.params.get('writeinfojson', False): - infofn = os.path.splitext(filename)[0] + u'.info.json' + infofn = os.path.splitext(filename)[0] + '.info.json' if self.params.get('nooverwrites', False) and os.path.exists(encodeFilename(infofn)): - self.to_screen(u'[info] Video description metadata is already present') + self.to_screen('[info] Video description metadata is already present') else: - self.to_screen(u'[info] Writing video description metadata as JSON to: ' + infofn) + self.to_screen('[info] Writing video description metadata as JSON to: ' + infofn) try: - 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)) + write_json_file(info_dict, encodeFilename(infofn)) except (OSError, IOError): - self.report_error(u'Cannot write metadata to JSON file ' + infofn) + self.report_error('Cannot write metadata to JSON file ' + infofn) return if self.params.get('writethumbnail', False): if info_dict.get('thumbnail') is not None: - thumb_format = determine_ext(info_dict['thumbnail'], u'jpg') - thumb_filename = os.path.splitext(filename)[0] + u'.' + thumb_format + thumb_format = determine_ext(info_dict['thumbnail'], 'jpg') + thumb_filename = os.path.splitext(filename)[0] + '.' + thumb_format if self.params.get('nooverwrites', False) and os.path.exists(encodeFilename(thumb_filename)): - self.to_screen(u'[%s] %s: Thumbnail is already present' % + self.to_screen('[%s] %s: Thumbnail is already present' % (info_dict['extractor'], info_dict['id'])) else: - self.to_screen(u'[%s] %s: Downloading thumbnail ...' % + self.to_screen('[%s] %s: Downloading thumbnail ...' % (info_dict['extractor'], info_dict['id'])) try: uf = compat_urllib_request.urlopen(info_dict['thumbnail']) with open(thumb_filename, 'wb') as thumbf: shutil.copyfileobj(uf, thumbf) - self.to_screen(u'[%s] %s: Writing thumbnail to: %s' % + self.to_screen('[%s] %s: Writing thumbnail to: %s' % (info_dict['extractor'], info_dict['id'], thumb_filename)) except (compat_urllib_error.URLError, compat_http_client.HTTPException, socket.error) as err: - self.report_warning(u'Unable to download thumbnail "%s": %s' % + self.report_warning('Unable to download thumbnail "%s": %s' % (info_dict['thumbnail'], compat_str(err))) if not self.params.get('skip_download', False): @@ -870,21 +900,41 @@ class YoutubeDL(object): success = True else: try: - success = self.fd._do_download(filename, info_dict) + def dl(name, info): + fd = get_suitable_downloader(info)(self, self.params) + for ph in self._progress_hooks: + fd.add_progress_hook(ph) + return fd.download(name, info) + if info_dict.get('requested_formats') is not None: + downloaded = [] + success = True + for f in info_dict['requested_formats']: + new_info = dict(info_dict) + new_info.update(f) + fname = self.prepare_filename(new_info) + fname = prepend_extension(fname, 'f%s' % f['format_id']) + downloaded.append(fname) + partial_success = dl(fname, new_info) + success = success and partial_success + info_dict['__postprocessors'] = [FFmpegMergerPP(self)] + info_dict['__files_to_merge'] = downloaded + else: + # Just a single file + success = dl(filename, info_dict) except (compat_urllib_error.URLError, compat_http_client.HTTPException, socket.error) as err: - self.report_error(u'unable to download video data: %s' % str(err)) + self.report_error('unable to download video data: %s' % str(err)) return except (OSError, IOError) as err: raise UnavailableVideoError(err) except (ContentTooShortError, ) as err: - self.report_error(u'content too short (expected %s bytes and served %s)' % (err.expected, err.downloaded)) + self.report_error('content too short (expected %s bytes and served %s)' % (err.expected, err.downloaded)) return if success: try: self.post_process(filename, info_dict) except (PostProcessingError) as err: - self.report_error(u'postprocessing: %s' % str(err)) + self.report_error('postprocessing: %s' % str(err)) return self.record_download_archive(info_dict) @@ -901,9 +951,9 @@ class YoutubeDL(object): #It also downloads the videos self.extract_info(url) except UnavailableVideoError: - self.report_error(u'unable to download video') + self.report_error('unable to download video') except MaxDownloadsReached: - self.to_screen(u'[info] Maximum number of downloaded files reached.') + self.to_screen('[info] Maximum number of downloaded files reached.') raise return self._download_retcode @@ -916,7 +966,7 @@ class YoutubeDL(object): except DownloadError: webpage_url = info.get('webpage_url') if webpage_url is not None: - self.report_warning(u'The info failed to download, trying with "%s"' % webpage_url) + self.report_warning('The info failed to download, trying with "%s"' % webpage_url) return self.download([webpage_url]) else: raise @@ -927,7 +977,11 @@ class YoutubeDL(object): info = dict(ie_info) info['filepath'] = filename keep_video = None - for pp in self._pps: + pps_chain = [] + if ie_info.get('__postprocessors') is not None: + pps_chain.extend(ie_info['__postprocessors']) + pps_chain.extend(self._pps) + for pp in pps_chain: try: keep_video_wish, new_info = pp.run(info) if keep_video_wish is not None: @@ -940,10 +994,10 @@ class YoutubeDL(object): self.report_error(e.msg) if keep_video is False and not self.params.get('keepvideo', False): try: - self.to_screen(u'Deleting original file %s (pass -k to keep)' % filename) + self.to_screen('Deleting original file %s (pass -k to keep)' % filename) os.remove(encodeFilename(filename)) except (IOError, OSError): - self.report_warning(u'Unable to remove downloaded video file') + self.report_warning('Unable to remove downloaded video file') def _make_archive_id(self, info_dict): # Future-proof against any change in case @@ -954,7 +1008,7 @@ class YoutubeDL(object): extractor = info_dict.get('ie_key') # key in a playlist if extractor is None: return None # Incomplete video information - return extractor.lower() + u' ' + info_dict['id'] + return extractor.lower() + ' ' + info_dict['id'] def in_download_archive(self, info_dict): fn = self.params.get('download_archive') @@ -982,53 +1036,61 @@ class YoutubeDL(object): vid_id = self._make_archive_id(info_dict) assert vid_id with locked_file(fn, 'a', encoding='utf-8') as archive_file: - archive_file.write(vid_id + u'\n') + archive_file.write(vid_id + '\n') @staticmethod def format_resolution(format, default='unknown'): if format.get('vcodec') == 'none': return 'audio only' - if format.get('_resolution') is not None: - return format['_resolution'] + 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']) + res = '%sx%s' % (format['width'], format['height']) else: - res = u'%sp' % format['height'] + res = '%sp' % format['height'] + elif format.get('width') is not None: + res = '?x%d' % format['width'] else: res = default return res def list_formats(self, info_dict): def format_note(fdict): - res = u'' + res = '' + if fdict.get('ext') in ['f4f', 'f4m']: + res += '(unsupported) ' if fdict.get('format_note') is not None: - res += fdict['format_note'] + u' ' + res += fdict['format_note'] + ' ' + if fdict.get('tbr') is not None: + res += '%4dk ' % fdict['tbr'] if (fdict.get('vcodec') is not None and fdict.get('vcodec') != 'none'): - res += u'%-5s' % fdict['vcodec'] - elif fdict.get('vbr') is not None: - res += u'video' + res += '%-5s' % fdict['vcodec'] + if fdict.get('vbr') is not None: + res += '@' + elif fdict.get('vbr') is not None and fdict.get('abr') is not None: + res += 'video@' if fdict.get('vbr') is not None: - res += u'@%4dk' % fdict['vbr'] + res += '%4dk' % fdict['vbr'] if fdict.get('acodec') is not None: if res: - res += u', ' - res += u'%-5s' % fdict['acodec'] + res += ', ' + res += '%-5s' % fdict['acodec'] elif fdict.get('abr') is not None: if res: - res += u', ' + res += ', ' res += 'audio' if fdict.get('abr') is not None: - res += u'@%3dk' % fdict['abr'] + res += '@%3dk' % fdict['abr'] if fdict.get('filesize') is not None: if res: - res += u', ' + res += ', ' res += format_bytes(fdict['filesize']) return res def line(format, idlen=20): - return ((u'%-' + compat_str(idlen + 1) + u's%-10s%-12s%s') % ( + return (('%-' + compat_str(idlen + 1) + 's%-10s%-12s%s') % ( format['format_id'], format['ext'], self.format_resolution(format), @@ -1036,7 +1098,7 @@ class YoutubeDL(object): )) formats = info_dict.get('formats', [info_dict]) - idlen = max(len(u'format code'), + idlen = max(len('format code'), max(len(f['format_id']) for f in formats)) formats_s = [line(f, idlen) for f in formats] if len(formats) > 1: @@ -1044,10 +1106,10 @@ class YoutubeDL(object): 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'}, idlen=idlen) - self.to_screen(u'[info] Available formats for %s:\n%s\n%s' % - (info_dict['id'], header_line, u"\n".join(formats_s))) + 'format_id': 'format code', 'ext': 'extension', + 'resolution': 'resolution', 'format_note': 'note'}, idlen=idlen) + self.to_screen('[info] Available formats for %s:\n%s\n%s' % + (info_dict['id'], header_line, '\n'.join(formats_s))) def urlopen(self, req): """ Start an HTTP download """ @@ -1056,7 +1118,7 @@ class YoutubeDL(object): def print_debug_header(self): if not self.params.get('verbose'): return - write_string(u'[debug] youtube-dl version ' + __version__ + u'\n') + write_string('[debug] youtube-dl version ' + __version__ + '\n') try: sp = subprocess.Popen( ['git', 'rev-parse', '--short', 'HEAD'], @@ -1065,20 +1127,20 @@ class YoutubeDL(object): out, err = sp.communicate() out = out.decode().strip() if re.match('[0-9a-f]+', out): - write_string(u'[debug] Git HEAD: ' + out + u'\n') + write_string('[debug] Git HEAD: ' + out + '\n') except: try: sys.exc_clear() except: pass - write_string(u'[debug] Python version %s - %s' % - (platform.python_version(), platform_name()) + u'\n') + write_string('[debug] Python version %s - %s' % + (platform.python_version(), platform_name()) + '\n') proxy_map = {} for handler in self._opener.handlers: if hasattr(handler, 'proxies'): proxy_map.update(handler.proxies) - write_string(u'[debug] Proxy map: ' + compat_str(proxy_map) + u'\n') + write_string('[debug] Proxy map: ' + compat_str(proxy_map) + '\n') def _setup_opener(self): timeout_val = self.params.get('socket_timeout') @@ -1108,10 +1170,13 @@ class YoutubeDL(object): if 'http' in proxies and 'https' not in proxies: proxies['https'] = proxies['http'] proxy_handler = compat_urllib_request.ProxyHandler(proxies) + + debuglevel = 1 if self.params.get('debug_printtraffic') else 0 https_handler = make_HTTPS_handler( - self.params.get('nocheckcertificate', False)) + self.params.get('nocheckcertificate', False), debuglevel=debuglevel) + ydlh = YoutubeDLHandler(debuglevel=debuglevel) opener = compat_urllib_request.build_opener( - https_handler, proxy_handler, cookie_processor, YoutubeDLHandler()) + https_handler, proxy_handler, cookie_processor, ydlh) # Delete the default user-agent header, which would otherwise apply in # cases where our custom HTTP handler doesn't come into play # (See https://github.com/rg3/youtube-dl/issues/1309 for details) diff --git a/youtube_dl/__init__.py b/youtube_dl/__init__.py index 6343730..82b1ff4 100644 --- a/youtube_dl/__init__.py +++ b/youtube_dl/__init__.py @@ -38,12 +38,15 @@ __authors__ = ( 'Takuya Tsuchida', 'Sergey M.', 'Michael Orlitzky', + 'Chris Gahan', + 'Saimadhav Heblikar', ) __license__ = 'Public Domain' import codecs import getpass +import locale import optparse import os import random @@ -73,11 +76,12 @@ from .FileDownloader import ( from .extractor import gen_extractors from .version import __version__ from .YoutubeDL import YoutubeDL -from .PostProcessor import ( +from .postprocessor import ( FFmpegMetadataPP, FFmpegVideoConvertor, FFmpegExtractAudioPP, FFmpegEmbedSubtitlePP, + XAttrMetadataPP, ) @@ -185,16 +189,16 @@ def parseOpts(overrideArguments=None): 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(), 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 .') + help='Location in the filesystem where youtube-dl can store some downloaded information permanently. By default $XDG_CACHE_HOME/youtube-dl or ~/.cache/youtube-dl . At the moment, only YouTube player files (for videos with obfuscated signatures) are cached, but that may change.') general.add_option( '--no-cache-dir', action='store_const', const=None, dest='cachedir', help='Disable filesystem caching') general.add_option( '--socket-timeout', dest='socket_timeout', - type=float, default=None, help=optparse.SUPPRESS_HELP) + type=float, default=None, help=u'Time to wait before giving up, in seconds') general.add_option( '--bidi-workaround', dest='bidi_workaround', action='store_true', - help=u'Work around terminals that lack bidirectional text support. Requires fribidi executable in PATH') + help=u'Work around terminals that lack bidirectional text support. Requires bidiv or fribidi executable in PATH') selection.add_option( @@ -213,8 +217,12 @@ def parseOpts(overrideArguments=None): selection.add_option('--min-filesize', metavar='SIZE', dest='min_filesize', help="Do not download any videos smaller than SIZE (e.g. 50k or 44.6m)", default=None) selection.add_option('--max-filesize', metavar='SIZE', dest='max_filesize', help="Do not download any videos larger than SIZE (e.g. 50k or 44.6m)", default=None) selection.add_option('--date', metavar='DATE', dest='date', help='download only videos uploaded in this date', default=None) - selection.add_option('--datebefore', metavar='DATE', dest='datebefore', help='download only videos uploaded before this date', default=None) - selection.add_option('--dateafter', metavar='DATE', dest='dateafter', help='download only videos uploaded after this date', default=None) + selection.add_option( + '--datebefore', metavar='DATE', dest='datebefore', default=None, + help='download only videos uploaded on or before this date (i.e. inclusive)') + selection.add_option( + '--dateafter', metavar='DATE', dest='dateafter', default=None, + help='download only videos uploaded on or after this date (i.e. inclusive)') selection.add_option( '--min-views', metavar='COUNT', dest='min_views', default=None, type=int, @@ -239,7 +247,7 @@ def parseOpts(overrideArguments=None): authentication.add_option('-n', '--netrc', action='store_true', dest='usenetrc', help='use .netrc authentication data', default=False) authentication.add_option('--video-password', - dest='videopassword', metavar='PASSWORD', help='video password (vimeo only)') + dest='videopassword', metavar='PASSWORD', help='video password (vimeo, smotri)') video_format.add_option('-f', '--format', @@ -252,7 +260,7 @@ def parseOpts(overrideArguments=None): video_format.add_option('--max-quality', action='store', dest='format_limit', metavar='FORMAT', help='highest quality format to download') video_format.add_option('-F', '--list-formats', - action='store_true', dest='listformats', help='list all available formats (currently youtube only)') + action='store_true', dest='listformats', help='list all available formats') subtitles.add_option('--write-sub', '--write-srt', action='store_true', dest='writesubtitles', @@ -326,14 +334,16 @@ def parseOpts(overrideArguments=None): action='store_true', dest='verbose', help='print various debugging information', default=False) verbosity.add_option('--dump-intermediate-pages', action='store_true', dest='dump_intermediate_pages', default=False, - help='print downloaded pages to debug problems(very verbose)') + 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 intermediary pages to files in the current directory to debug problems') verbosity.add_option('--youtube-print-sig-code', action='store_true', dest='youtube_print_sig_code', default=False, help=optparse.SUPPRESS_HELP) - + verbosity.add_option('--print-traffic', + dest='debug_printtraffic', action='store_true', default=False, + help=optparse.SUPPRESS_HELP) filesystem.add_option('-t', '--title', action='store_true', dest='usetitle', help='use title in file name (default)', default=False) @@ -350,11 +360,11 @@ def parseOpts(overrideArguments=None): '%(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, ' - '%(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"),' + '%(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, ' + '%(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. ' 'Use - to output to stdout. Can also be used to download to a different directory, ' 'for example with -o \'/my/downloads/%(uploader)s/%(title)s-%(id)s.%(ext)s\' .')) @@ -368,7 +378,7 @@ def parseOpts(overrideArguments=None): dest='batchfile', metavar='FILE', help='file containing URLs to download (\'-\' for stdin)') filesystem.add_option('--load-info', dest='load_info_filename', metavar='FILE', - help='json file containing the video information (created with the "--write-json" option') + help='json file containing the video information (created with the "--write-json" option)') filesystem.add_option('-w', '--no-overwrites', action='store_true', dest='nooverwrites', help='do not overwrite files', default=False) filesystem.add_option('-c', '--continue', @@ -412,7 +422,13 @@ def parseOpts(overrideArguments=None): postproc.add_option('--embed-subs', action='store_true', dest='embedsubtitles', default=False, help='embed subtitles in the video (only for mp4 videos)') postproc.add_option('--add-metadata', action='store_true', dest='addmetadata', default=False, - help='add metadata to the files') + help='write metadata to the video file') + postproc.add_option('--xattrs', action='store_true', dest='xattrs', default=False, + help='write metadata to the video file\'s xattrs (using dublin core and xdg standards)') + postproc.add_option('--prefer-avconv', action='store_false', dest='prefer_ffmpeg', + help='Prefer avconv over ffmpeg for running the postprocessors (default)') + postproc.add_option('--prefer-ffmpeg', action='store_true', dest='prefer_ffmpeg', + help='Prefer ffmpeg over avconv for running the postprocessors') parser.add_option_group(general) @@ -473,6 +489,8 @@ def parseOpts(overrideArguments=None): write_string(u'[debug] System config: ' + repr(_hide_login_info(systemConf)) + '\n') write_string(u'[debug] User config: ' + repr(_hide_login_info(userConf)) + '\n') write_string(u'[debug] Command-line args: ' + repr(_hide_login_info(commandLineConf)) + '\n') + write_string(u'[debug] Encodings: locale %r, fs %r, out %r, pref: %r\n' % + (locale.getpreferredencoding(), sys.getfilesystemencoding(), sys.stdout.encoding, preferredencoding())) return parser, opts, args @@ -517,6 +535,8 @@ def _real_main(argv=None): sys.exit(u'ERROR: batch file could not be read') all_urls = batchurls + args all_urls = [url.strip() for url in all_urls] + _enc = preferredencoding() + all_urls = [url.decode(_enc, 'ignore') if isinstance(url, bytes) else url for url in all_urls] extractors = gen_extractors() @@ -546,7 +566,7 @@ def _real_main(argv=None): if opts.usenetrc and (opts.username is not None or opts.password is not None): parser.error(u'using .netrc conflicts with giving username/password') if opts.password is not None and opts.username is None: - parser.error(u' account username missing\n') + parser.error(u'account username missing\n') if opts.outtmpl is not None and (opts.usetitle or opts.autonumber or opts.useid): parser.error(u'using output template conflicts with using title, video ID or auto number') if opts.usetitle and opts.useid: @@ -620,6 +640,7 @@ def _real_main(argv=None): u' template'.format(outtmpl)) any_printing = opts.geturl or opts.gettitle or opts.getid or opts.getthumbnail or opts.getdescription or opts.getfilename or opts.getformat or opts.getduration or opts.dumpjson + download_archive_fn = os.path.expanduser(opts.download_archive) if opts.download_archive is not None else opts.download_archive ydl_opts = { 'usenetrc': opts.usenetrc, @@ -687,12 +708,14 @@ def _real_main(argv=None): 'cachedir': opts.cachedir, 'youtube_print_sig_code': opts.youtube_print_sig_code, 'age_limit': opts.age_limit, - 'download_archive': opts.download_archive, + 'download_archive': download_archive_fn, 'cookiefile': opts.cookiefile, 'nocheckcertificate': opts.no_check_certificate, 'proxy': opts.proxy, 'socket_timeout': opts.socket_timeout, 'bidi_workaround': opts.bidi_workaround, + 'debug_printtraffic': opts.debug_printtraffic, + 'prefer_ffmpeg': opts.prefer_ffmpeg, } with YoutubeDL(ydl_opts) as ydl: @@ -709,6 +732,8 @@ def _real_main(argv=None): ydl.add_post_processor(FFmpegVideoConvertor(preferedformat=opts.recodevideo)) if opts.embedsubtitles: ydl.add_post_processor(FFmpegEmbedSubtitlePP(subtitlesformat=opts.subtitlesformat)) + if opts.xattrs: + ydl.add_post_processor(XAttrMetadataPP()) # Update version if opts.update_self: diff --git a/youtube_dl/downloader/__init__.py b/youtube_dl/downloader/__init__.py new file mode 100644 index 0000000..f19b490 --- /dev/null +++ b/youtube_dl/downloader/__init__.py @@ -0,0 +1,23 @@ +from .common import FileDownloader +from .hls import HlsFD +from .http import HttpFD +from .mplayer import MplayerFD +from .rtmp import RtmpFD + +from ..utils import ( + determine_ext, +) + +def get_suitable_downloader(info_dict): + """Get the downloader class that can handle the info dict.""" + url = info_dict['url'] + + if url.startswith('rtmp'): + return RtmpFD + if determine_ext(url) == u'm3u8': + return HlsFD + if url.startswith('mms') or url.startswith('rtsp'): + return MplayerFD + else: + return HttpFD + diff --git a/youtube_dl/downloader/common.py b/youtube_dl/downloader/common.py new file mode 100644 index 0000000..10143d5 --- /dev/null +++ b/youtube_dl/downloader/common.py @@ -0,0 +1,317 @@ +import os +import re +import sys +import time + +from ..utils import ( + encodeFilename, + timeconvert, + format_bytes, +) + + +class FileDownloader(object): + """File Downloader class. + + File downloader objects are the ones responsible of downloading the + actual video file and writing it to disk. + + File downloaders accept a lot of parameters. In order not to saturate + the object constructor with arguments, it receives a dictionary of + options instead. + + Available options: + + verbose: Print additional info to stdout. + quiet: Do not print messages to stdout. + ratelimit: Download speed limit, in bytes/sec. + retries: Number of times to retry for HTTP error 5xx + buffersize: Size of download buffer in bytes. + noresizebuffer: Do not automatically resize the download buffer. + continuedl: Try to continue downloads if possible. + noprogress: Do not print the progress bar. + logtostderr: Log messages to stderr instead of stdout. + consoletitle: Display progress in console window's titlebar. + nopart: Do not use temporary .part files. + updatetime: Use the Last-modified header to set output file timestamps. + test: Download only first bytes to test the downloader. + min_filesize: Skip files smaller than this size + max_filesize: Skip files larger than this size + + Subclasses of this one must re-define the real_download method. + """ + + params = None + + def __init__(self, ydl, params): + """Create a FileDownloader object with the given options.""" + self.ydl = ydl + self._progress_hooks = [] + self.params = params + + @staticmethod + def format_seconds(seconds): + (mins, secs) = divmod(seconds, 60) + (hours, mins) = divmod(mins, 60) + if hours > 99: + return '--:--:--' + if hours == 0: + return '%02d:%02d' % (mins, secs) + else: + return '%02d:%02d:%02d' % (hours, mins, secs) + + @staticmethod + def calc_percent(byte_counter, data_len): + if data_len is None: + return None + return float(byte_counter) / float(data_len) * 100.0 + + @staticmethod + def format_percent(percent): + if percent is None: + return '---.-%' + return '%6s' % ('%3.1f%%' % percent) + + @staticmethod + def calc_eta(start, now, total, current): + if total is None: + return None + dif = now - start + if current == 0 or dif < 0.001: # One millisecond + return None + rate = float(current) / dif + return int((float(total) - float(current)) / rate) + + @staticmethod + def format_eta(eta): + if eta is None: + return '--:--' + return FileDownloader.format_seconds(eta) + + @staticmethod + def calc_speed(start, now, bytes): + dif = now - start + if bytes == 0 or dif < 0.001: # One millisecond + return None + return float(bytes) / dif + + @staticmethod + def format_speed(speed): + if speed is None: + return '%10s' % '---b/s' + return '%10s' % ('%s/s' % format_bytes(speed)) + + @staticmethod + def best_block_size(elapsed_time, bytes): + new_min = max(bytes / 2.0, 1.0) + new_max = min(max(bytes * 2.0, 1.0), 4194304) # Do not surpass 4 MB + if elapsed_time < 0.001: + return int(new_max) + rate = bytes / elapsed_time + if rate > new_max: + return int(new_max) + if rate < new_min: + return int(new_min) + return int(rate) + + @staticmethod + def parse_bytes(bytestr): + """Parse a string indicating a byte quantity into an integer.""" + matchobj = re.match(r'(?i)^(\d+(?:\.\d+)?)([kMGTPEZY]?)$', bytestr) + if matchobj is None: + return None + number = float(matchobj.group(1)) + multiplier = 1024.0 ** 'bkmgtpezy'.index(matchobj.group(2).lower()) + return int(round(number * multiplier)) + + def to_screen(self, *args, **kargs): + self.ydl.to_screen(*args, **kargs) + + def to_stderr(self, message): + self.ydl.to_screen(message) + + def to_console_title(self, message): + self.ydl.to_console_title(message) + + def trouble(self, *args, **kargs): + self.ydl.trouble(*args, **kargs) + + def report_warning(self, *args, **kargs): + self.ydl.report_warning(*args, **kargs) + + def report_error(self, *args, **kargs): + self.ydl.report_error(*args, **kargs) + + 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 temp_name(self, filename): + """Returns a temporary filename for the given filename.""" + if self.params.get('nopart', False) or filename == u'-' or \ + (os.path.exists(encodeFilename(filename)) and not os.path.isfile(encodeFilename(filename))): + return filename + return filename + u'.part' + + def undo_temp_name(self, filename): + if filename.endswith(u'.part'): + return filename[:-len(u'.part')] + return filename + + def try_rename(self, old_filename, new_filename): + try: + if old_filename == new_filename: + return + os.rename(encodeFilename(old_filename), encodeFilename(new_filename)) + except (IOError, OSError) as err: + self.report_error(u'unable to rename file: %s' % str(err)) + + def try_utime(self, filename, last_modified_hdr): + """Try to set the last-modified time of the given file.""" + if last_modified_hdr is None: + return + if not os.path.isfile(encodeFilename(filename)): + return + timestr = last_modified_hdr + if timestr is None: + return + filetime = timeconvert(timestr) + if filetime is None: + return filetime + # Ignore obviously invalid dates + if filetime == 0: + return + try: + os.utime(filename, (time.time(), filetime)) + except: + pass + return filetime + + def report_destination(self, filename): + """Report destination filename.""" + self.to_screen(u'[download] Destination: ' + filename) + + def _report_progress_status(self, msg, is_last_line=False): + fullmsg = u'[download] ' + msg + if self.params.get('progress_with_newline', False): + self.to_screen(fullmsg) + else: + if os.name == 'nt': + prev_len = getattr(self, '_report_progress_prev_line_length', + 0) + if prev_len > len(fullmsg): + fullmsg += u' ' * (prev_len - len(fullmsg)) + self._report_progress_prev_line_length = len(fullmsg) + clear_line = u'\r' + else: + clear_line = (u'\r\x1b[K' if sys.stderr.isatty() else u'\r') + self.to_screen(clear_line + fullmsg, skip_eol=not is_last_line) + self.to_console_title(u'youtube-dl ' + msg) + + def report_progress(self, percent, data_len_str, speed, eta): + """Report download progress.""" + if self.params.get('noprogress', False): + return + 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) + + msg = (u'%s of %s at %s ETA %s' % + (percent_str, data_len_str, speed_str, eta_str)) + self._report_progress_status(msg) + + def report_progress_live_stream(self, downloaded_data_len, speed, elapsed): + if self.params.get('noprogress', False): + return + downloaded_str = format_bytes(downloaded_data_len) + speed_str = self.format_speed(speed) + elapsed_str = FileDownloader.format_seconds(elapsed) + msg = u'%s at %s (%s)' % (downloaded_str, speed_str, elapsed_str) + self._report_progress_status(msg) + + def report_finish(self, data_len_str, tot_time): + """Report download finished.""" + if self.params.get('noprogress', False): + self.to_screen(u'[download] Download completed') + else: + self._report_progress_status( + (u'100%% of %s in %s' % + (data_len_str, self.format_seconds(tot_time))), + is_last_line=True) + + def report_resuming_byte(self, resume_len): + """Report attempt to resume at given byte.""" + self.to_screen(u'[download] Resuming download at byte %s' % resume_len) + + def report_retry(self, count, retries): + """Report retry in case of HTTP error 5xx""" + self.to_screen(u'[download] Got server HTTP error. Retrying (attempt %d of %d)...' % (count, retries)) + + def report_file_already_downloaded(self, file_name): + """Report file has already been fully downloaded.""" + try: + self.to_screen(u'[download] %s has already been downloaded' % file_name) + except UnicodeEncodeError: + self.to_screen(u'[download] The file has already been downloaded') + + def report_unable_to_resume(self): + """Report it was impossible to resume download.""" + self.to_screen(u'[download] Unable to resume') + + def download(self, filename, info_dict): + """Download to a filename using the info from info_dict + Return True on success and False otherwise + """ + # Check file already present + if self.params.get('continuedl', False) and os.path.isfile(encodeFilename(filename)) and not self.params.get('nopart', False): + self.report_file_already_downloaded(filename) + self._hook_progress({ + 'filename': filename, + 'status': 'finished', + 'total_bytes': os.path.getsize(encodeFilename(filename)), + }) + return True + + return self.real_download(filename, info_dict) + + def real_download(self, filename, info_dict): + """Real download process. Redefine in subclasses.""" + raise NotImplementedError(u'This method must be implemented by sublcasses') + + def _hook_progress(self, status): + for ph in self._progress_hooks: + ph(status) + + def add_progress_hook(self, ph): + """ ph gets called on download progress, with a dictionary with the entries + * filename: The final filename + * status: One of "downloading" and "finished" + + It can also have some of the following entries: + + * downloaded_bytes: Bytes on disks + * total_bytes: Total bytes, None if unknown + * tmpfilename: The filename we're currently writing to + * eta: The estimated time in seconds, None if unknown + * speed: The download speed in bytes/second, None if unknown + + Hooks are guaranteed to be called at least once (with status "finished") + if the download is successful. + """ + self._progress_hooks.append(ph) + diff --git a/youtube_dl/downloader/hls.py b/youtube_dl/downloader/hls.py new file mode 100644 index 0000000..fa98346 --- /dev/null +++ b/youtube_dl/downloader/hls.py @@ -0,0 +1,44 @@ +import os +import subprocess + +from .common import FileDownloader +from ..utils import ( + encodeFilename, +) + + +class HlsFD(FileDownloader): + def real_download(self, filename, info_dict): + url = info_dict['url'] + self.report_destination(filename) + tmpfilename = self.temp_name(filename) + + args = ['-y', '-i', url, '-f', 'mp4', '-c', 'copy', + '-bsf:a', 'aac_adtstoasc', tmpfilename] + + 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' % (cmd[0], fsize)) + self.try_rename(tmpfilename, filename) + self._hook_progress({ + 'downloaded_bytes': fsize, + 'total_bytes': fsize, + 'filename': filename, + 'status': 'finished', + }) + return True + else: + self.to_stderr(u"\n") + self.report_error(u'ffmpeg exited with code %d' % retval) + return False diff --git a/youtube_dl/downloader/http.py b/youtube_dl/downloader/http.py new file mode 100644 index 0000000..8407727 --- /dev/null +++ b/youtube_dl/downloader/http.py @@ -0,0 +1,186 @@ +import os +import time + +from .common import FileDownloader +from ..utils import ( + compat_urllib_request, + compat_urllib_error, + ContentTooShortError, + + encodeFilename, + sanitize_open, + format_bytes, +) + + +class HttpFD(FileDownloader): + def real_download(self, filename, info_dict): + url = info_dict['url'] + tmpfilename = self.temp_name(filename) + stream = None + + # Do not include the Accept-Encoding header + headers = {'Youtubedl-no-compression': 'True'} + if 'user_agent' in info_dict: + headers['Youtubedl-user-agent'] = info_dict['user_agent'] + basic_request = compat_urllib_request.Request(url, None, headers) + request = compat_urllib_request.Request(url, None, headers) + + if self.params.get('test', False): + request.add_header('Range','bytes=0-10240') + + # Establish possible resume length + if os.path.isfile(encodeFilename(tmpfilename)): + resume_len = os.path.getsize(encodeFilename(tmpfilename)) + else: + resume_len = 0 + + open_mode = 'wb' + if resume_len != 0: + if self.params.get('continuedl', False): + self.report_resuming_byte(resume_len) + request.add_header('Range','bytes=%d-' % resume_len) + open_mode = 'ab' + else: + resume_len = 0 + + count = 0 + retries = self.params.get('retries', 0) + while count <= retries: + # Establish connection + try: + data = compat_urllib_request.urlopen(request) + break + except (compat_urllib_error.HTTPError, ) as err: + if (err.code < 500 or err.code >= 600) and err.code != 416: + # Unexpected HTTP error + raise + elif err.code == 416: + # Unable to resume (requested range not satisfiable) + try: + # Open the connection again without the range header + data = compat_urllib_request.urlopen(basic_request) + content_length = data.info()['Content-Length'] + except (compat_urllib_error.HTTPError, ) as err: + if err.code < 500 or err.code >= 600: + raise + else: + # Examine the reported length + if (content_length is not None and + (resume_len - 100 < int(content_length) < resume_len + 100)): + # The file had already been fully downloaded. + # Explanation to the above condition: in issue #175 it was revealed that + # YouTube sometimes adds or removes a few bytes from the end of the file, + # changing the file size slightly and causing problems for some users. So + # I decided to implement a suggested change and consider the file + # completely downloaded if the file size differs less than 100 bytes from + # the one in the hard drive. + self.report_file_already_downloaded(filename) + self.try_rename(tmpfilename, filename) + self._hook_progress({ + 'filename': filename, + 'status': 'finished', + }) + return True + else: + # The length does not match, we start the download over + self.report_unable_to_resume() + open_mode = 'wb' + break + # Retry + count += 1 + if count <= retries: + self.report_retry(count, retries) + + if count > retries: + self.report_error(u'giving up after %s retries' % retries) + return False + + data_len = data.info().get('Content-length', None) + if data_len is not None: + data_len = int(data_len) + resume_len + min_data_len = self.params.get("min_filesize", None) + max_data_len = self.params.get("max_filesize", None) + if min_data_len is not None and data_len < min_data_len: + self.to_screen(u'\r[download] File is smaller than min-filesize (%s bytes < %s bytes). Aborting.' % (data_len, min_data_len)) + return False + if max_data_len is not None and data_len > max_data_len: + self.to_screen(u'\r[download] File is larger than max-filesize (%s bytes > %s bytes). Aborting.' % (data_len, max_data_len)) + return False + + data_len_str = format_bytes(data_len) + byte_counter = 0 + resume_len + block_size = self.params.get('buffersize', 1024) + start = time.time() + while True: + # Download and write + before = time.time() + data_block = data.read(block_size) + after = time.time() + if len(data_block) == 0: + break + byte_counter += len(data_block) + + # Open file just in time + if stream is None: + try: + (stream, tmpfilename) = sanitize_open(tmpfilename, open_mode) + assert stream is not None + filename = self.undo_temp_name(tmpfilename) + self.report_destination(filename) + except (OSError, IOError) as err: + self.report_error(u'unable to open for writing: %s' % str(err)) + return False + try: + stream.write(data_block) + except (IOError, OSError) as err: + self.to_stderr(u"\n") + self.report_error(u'unable to write data: %s' % str(err)) + return False + if not self.params.get('noresizebuffer', False): + block_size = self.best_block_size(after - before, len(data_block)) + + # Progress message + speed = self.calc_speed(start, time.time(), byte_counter - resume_len) + if data_len is 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._hook_progress({ + 'downloaded_bytes': byte_counter, + 'total_bytes': data_len, + 'tmpfilename': tmpfilename, + 'filename': filename, + 'status': 'downloading', + 'eta': eta, + 'speed': speed, + }) + + # Apply rate limit + self.slow_down(start, byte_counter - resume_len) + + if stream is None: + self.to_stderr(u"\n") + self.report_error(u'Did not get any data blocks') + return False + stream.close() + self.report_finish(data_len_str, (time.time() - start)) + if data_len is not None and byte_counter != data_len: + raise ContentTooShortError(byte_counter, int(data_len)) + self.try_rename(tmpfilename, filename) + + # Update file modification time + if self.params.get('updatetime', True): + info_dict['filetime'] = self.try_utime(filename, data.info().get('last-modified', None)) + + self._hook_progress({ + 'downloaded_bytes': byte_counter, + 'total_bytes': byte_counter, + 'filename': filename, + 'status': 'finished', + }) + + return True diff --git a/youtube_dl/downloader/mplayer.py b/youtube_dl/downloader/mplayer.py new file mode 100644 index 0000000..67e0e41 --- /dev/null +++ b/youtube_dl/downloader/mplayer.py @@ -0,0 +1,40 @@ +import os +import subprocess + +from .common import FileDownloader +from ..utils import ( + encodeFilename, +) + + +class MplayerFD(FileDownloader): + def real_download(self, filename, info_dict): + url = info_dict['url'] + self.report_destination(filename) + tmpfilename = self.temp_name(filename) + + args = ['mplayer', '-really-quiet', '-vo', 'null', '-vc', 'dummy', '-dumpstream', '-dumpfile', tmpfilename, url] + # Check for mplayer first + try: + subprocess.call(['mplayer', '-h'], stdout=(open(os.path.devnull, 'w')), stderr=subprocess.STDOUT) + except (OSError, IOError): + self.report_error(u'MMS or RTSP download detected but "%s" could not be run' % args[0] ) + return False + + # Download using mplayer. + retval = subprocess.call(args) + if retval == 0: + fsize = os.path.getsize(encodeFilename(tmpfilename)) + self.to_screen(u'\r[%s] %s bytes' % (args[0], fsize)) + self.try_rename(tmpfilename, filename) + self._hook_progress({ + 'downloaded_bytes': fsize, + 'total_bytes': fsize, + 'filename': filename, + 'status': 'finished', + }) + return True + else: + self.to_stderr(u"\n") + self.report_error(u'mplayer exited with code %d' % retval) + return False diff --git a/youtube_dl/downloader/rtmp.py b/youtube_dl/downloader/rtmp.py new file mode 100644 index 0000000..b165e39 --- /dev/null +++ b/youtube_dl/downloader/rtmp.py @@ -0,0 +1,178 @@ +import os +import re +import subprocess +import sys +import time + +from .common import FileDownloader +from ..utils import ( + encodeFilename, + format_bytes, +) + + +class RtmpFD(FileDownloader): + def real_download(self, filename, info_dict): + def run_rtmpdump(args): + start = time.time() + resume_percent = None + resume_downloaded_data_len = None + proc = subprocess.Popen(args, stderr=subprocess.PIPE) + cursor_in_new_line = True + proc_stderr_closed = False + while not proc_stderr_closed: + # read line from stderr + line = u'' + while True: + char = proc.stderr.read(1) + if not char: + proc_stderr_closed = True + break + if char in [b'\r', b'\n']: + break + line += char.decode('ascii', 'replace') + if not line: + # proc_stderr_closed is True + continue + mobj = re.search(r'([0-9]+\.[0-9]{3}) kB / [0-9]+\.[0-9]{2} sec \(([0-9]{1,2}\.[0-9])%\)', line) + if mobj: + downloaded_data_len = int(float(mobj.group(1))*1024) + percent = float(mobj.group(2)) + if not resume_percent: + resume_percent = percent + resume_downloaded_data_len = downloaded_data_len + eta = self.calc_eta(start, time.time(), 100-resume_percent, percent-resume_percent) + speed = self.calc_speed(start, time.time(), downloaded_data_len-resume_downloaded_data_len) + data_len = None + if percent > 0: + data_len = int(downloaded_data_len * 100 / percent) + data_len_str = u'~' + format_bytes(data_len) + self.report_progress(percent, data_len_str, speed, eta) + cursor_in_new_line = False + self._hook_progress({ + 'downloaded_bytes': downloaded_data_len, + 'total_bytes': data_len, + 'tmpfilename': tmpfilename, + 'filename': filename, + 'status': 'downloading', + 'eta': eta, + 'speed': speed, + }) + else: + # no percent for live streams + mobj = re.search(r'([0-9]+\.[0-9]{3}) kB / [0-9]+\.[0-9]{2} sec', line) + if mobj: + downloaded_data_len = int(float(mobj.group(1))*1024) + time_now = time.time() + speed = self.calc_speed(start, time_now, downloaded_data_len) + self.report_progress_live_stream(downloaded_data_len, speed, time_now - start) + cursor_in_new_line = False + self._hook_progress({ + 'downloaded_bytes': downloaded_data_len, + 'tmpfilename': tmpfilename, + 'filename': filename, + 'status': 'downloading', + 'speed': speed, + }) + elif self.params.get('verbose', False): + if not cursor_in_new_line: + self.to_screen(u'') + cursor_in_new_line = True + self.to_screen(u'[rtmpdump] '+line) + proc.wait() + if not cursor_in_new_line: + self.to_screen(u'') + return proc.returncode + + url = info_dict['url'] + player_url = info_dict.get('player_url', None) + page_url = info_dict.get('page_url', None) + play_path = info_dict.get('play_path', None) + tc_url = info_dict.get('tc_url', None) + live = info_dict.get('rtmp_live', False) + conn = info_dict.get('rtmp_conn', None) + + self.report_destination(filename) + tmpfilename = self.temp_name(filename) + test = self.params.get('test', False) + + # Check for rtmpdump first + try: + subprocess.call(['rtmpdump', '-h'], stdout=(open(os.path.devnull, 'w')), stderr=subprocess.STDOUT) + except (OSError, IOError): + self.report_error(u'RTMP download detected but "rtmpdump" could not be run') + return False + + # Download using rtmpdump. rtmpdump returns exit code 2 when + # the connection was interrumpted and resuming appears to be + # possible. This is part of rtmpdump's normal usage, AFAIK. + basic_args = ['rtmpdump', '--verbose', '-r', url, '-o', tmpfilename] + if player_url is not None: + basic_args += ['--swfVfy', player_url] + if page_url is not None: + basic_args += ['--pageUrl', page_url] + if play_path is not None: + basic_args += ['--playpath', play_path] + if tc_url is not None: + basic_args += ['--tcUrl', url] + if test: + basic_args += ['--stop', '1'] + if live: + basic_args += ['--live'] + if conn: + basic_args += ['--conn', conn] + args = basic_args + [[], ['--resume', '--skip', '1']][self.params.get('continuedl', False)] + + if sys.platform == 'win32' and sys.version_info < (3, 0): + # Windows subprocess module does not actually support Unicode + # on Python 2.x + # See http://stackoverflow.com/a/9951851/35070 + subprocess_encoding = sys.getfilesystemencoding() + args = [a.encode(subprocess_encoding, 'ignore') for a in args] + else: + subprocess_encoding = None + + if self.params.get('verbose', False): + if subprocess_encoding: + str_args = [ + a.decode(subprocess_encoding) if isinstance(a, bytes) else a + for a in args] + else: + str_args = args + try: + import pipes + shell_quote = lambda args: ' '.join(map(pipes.quote, str_args)) + except ImportError: + shell_quote = repr + self.to_screen(u'[debug] rtmpdump command line: ' + shell_quote(str_args)) + + retval = run_rtmpdump(args) + + while (retval == 2 or retval == 1) and not test: + prevsize = os.path.getsize(encodeFilename(tmpfilename)) + self.to_screen(u'[rtmpdump] %s bytes' % prevsize) + time.sleep(5.0) # This seems to be needed + retval = run_rtmpdump(basic_args + ['-e'] + [[], ['-k', '1']][retval == 1]) + cursize = os.path.getsize(encodeFilename(tmpfilename)) + if prevsize == cursize and retval == 1: + break + # Some rtmp streams seem abort after ~ 99.8%. Don't complain for those + if prevsize == cursize and retval == 2 and cursize > 1024: + self.to_screen(u'[rtmpdump] Could not download the whole video. This can happen for some advertisements.') + retval = 0 + break + if retval == 0 or (test and retval == 2): + fsize = os.path.getsize(encodeFilename(tmpfilename)) + self.to_screen(u'[rtmpdump] %s bytes' % fsize) + self.try_rename(tmpfilename, filename) + self._hook_progress({ + 'downloaded_bytes': fsize, + 'total_bytes': fsize, + 'filename': filename, + 'status': 'finished', + }) + return True + else: + self.to_stderr(u"\n") + self.report_error(u'rtmpdump exited with code %d' % retval) + return False diff --git a/youtube_dl/extractor/__init__.py b/youtube_dl/extractor/__init__.py index a39a1e2..d66f7b0 100644 --- a/youtube_dl/extractor/__init__.py +++ b/youtube_dl/extractor/__init__.py @@ -28,6 +28,7 @@ from .channel9 import Channel9IE from .cinemassacre import CinemassacreIE from .clipfish import ClipfishIE from .clipsyndicate import ClipsyndicateIE +from .cmt import CMTIE from .cnn import CNNIE from .collegehumor import CollegeHumorIE from .comedycentral import ComedyCentralIE, ComedyCentralShowsIE @@ -51,6 +52,7 @@ from .ehow import EHowIE from .eighttracks import EightTracksIE from .eitb import EitbIE from .escapist import EscapistIE +from .everyonesmixtape import EveryonesMixtapeIE from .exfm import ExfmIE from .extremetube import ExtremeTubeIE from .facebook import FacebookIE @@ -60,11 +62,13 @@ from .fktv import ( FKTVPosteckeIE, ) from .flickr import FlickrIE +from .franceinter import FranceInterIE from .francetv import ( PluzzIE, FranceTvInfoIE, FranceTVIE, - GenerationQuoiIE + GenerationQuoiIE, + CultureboxIE, ) from .freesound import FreesoundIE from .funnyordie import FunnyOrDieIE @@ -79,7 +83,10 @@ from .hotnewhiphop import HotNewHipHopIE from .howcast import HowcastIE from .hypem import HypemIE from .ign import IGNIE, OneUPIE -from .imdb import ImdbIE +from .imdb import ( + ImdbIE, + ImdbListIE +) from .ina import InaIE from .infoq import InfoQIE from .instagram import InstagramIE @@ -91,17 +98,25 @@ from .ivi import ( from .jeuxvideo import JeuxVideoIE from .jukebox import JukeboxIE from .justintv import JustinTVIE +from .jpopsukitv import JpopsukiIE from .kankan import KankanIE from .keezmovies import KeezMoviesIE +from .khanacademy import KhanAcademyIE from .kickstarter import KickStarterIE from .keek import KeekIE from .liveleak import LiveLeakIE from .livestream import LivestreamIE, LivestreamOriginalIE +from .lynda import ( + LyndaIE, + LyndaCourseIE +) +from .macgamestore import MacGameStoreIE from .mdr import MDRIE from .metacafe import MetacafeIE from .metacritic import MetacriticIE from .mit import TechTVMITIE, MITIE from .mixcloud import MixcloudIE +from .mpora import MporaIE from .mofosex import MofosexIE from .mtv import MTVIE from .muzu import MuzuTVIE @@ -116,6 +131,7 @@ from .newgrounds import NewgroundsIE from .nhl import NHLIE, NHLVideocenterIE from .niconico import NiconicoIE from .ninegag import NineGagIE +from .novamov import NovamovIE from .nowvideo import NowVideoIE from .ooyala import OoyalaIE from .orf import ORFIE @@ -189,6 +205,7 @@ from .vimeo import ( VimeoUserIE, VimeoAlbumIE, VimeoGroupsIE, + VimeoReviewIE, ) from .vine import VineIE from .viki import VikiIE diff --git a/youtube_dl/extractor/academicearth.py b/youtube_dl/extractor/academicearth.py index ac05f82..72f81d0 100644 --- a/youtube_dl/extractor/academicearth.py +++ b/youtube_dl/extractor/academicearth.py @@ -1,3 +1,4 @@ +from __future__ import unicode_literals import re from .common import InfoExtractor @@ -5,7 +6,7 @@ from .common import InfoExtractor class AcademicEarthCourseIE(InfoExtractor): _VALID_URL = r'^https?://(?:www\.)?academicearth\.org/(?:courses|playlists)/(?P[^?#/]+)' - IE_NAME = u'AcademicEarth:Course' + IE_NAME = 'AcademicEarth:Course' def _real_extract(self, url): m = re.match(self._VALID_URL, url) diff --git a/youtube_dl/extractor/appletrailers.py b/youtube_dl/extractor/appletrailers.py index ef5644a..922cede 100644 --- a/youtube_dl/extractor/appletrailers.py +++ b/youtube_dl/extractor/appletrailers.py @@ -1,3 +1,5 @@ +from __future__ import unicode_literals + import re import json @@ -11,46 +13,46 @@ from ..utils import ( class AppleTrailersIE(InfoExtractor): _VALID_URL = r'https?://(?:www\.)?trailers\.apple\.com/trailers/(?P[^/]+)/(?P[^/]+)' _TEST = { - u"url": u"http://trailers.apple.com/trailers/wb/manofsteel/", - u"playlist": [ + "url": "http://trailers.apple.com/trailers/wb/manofsteel/", + "playlist": [ { - u"file": u"manofsteel-trailer4.mov", - u"md5": u"d97a8e575432dbcb81b7c3acb741f8a8", - u"info_dict": { - u"duration": 111, - u"title": u"Trailer 4", - u"upload_date": u"20130523", - u"uploader_id": u"wb", + "file": "manofsteel-trailer4.mov", + "md5": "d97a8e575432dbcb81b7c3acb741f8a8", + "info_dict": { + "duration": 111, + "title": "Trailer 4", + "upload_date": "20130523", + "uploader_id": "wb", }, }, { - u"file": u"manofsteel-trailer3.mov", - u"md5": u"b8017b7131b721fb4e8d6f49e1df908c", - u"info_dict": { - u"duration": 182, - u"title": u"Trailer 3", - u"upload_date": u"20130417", - u"uploader_id": u"wb", + "file": "manofsteel-trailer3.mov", + "md5": "b8017b7131b721fb4e8d6f49e1df908c", + "info_dict": { + "duration": 182, + "title": "Trailer 3", + "upload_date": "20130417", + "uploader_id": "wb", }, }, { - u"file": u"manofsteel-trailer.mov", - u"md5": u"d0f1e1150989b9924679b441f3404d48", - u"info_dict": { - u"duration": 148, - u"title": u"Trailer", - u"upload_date": u"20121212", - u"uploader_id": u"wb", + "file": "manofsteel-trailer.mov", + "md5": "d0f1e1150989b9924679b441f3404d48", + "info_dict": { + "duration": 148, + "title": "Trailer", + "upload_date": "20121212", + "uploader_id": "wb", }, }, { - u"file": u"manofsteel-teaser.mov", - u"md5": u"5fe08795b943eb2e757fa95cb6def1cb", - u"info_dict": { - u"duration": 93, - u"title": u"Teaser", - u"upload_date": u"20120721", - u"uploader_id": u"wb", + "file": "manofsteel-teaser.mov", + "md5": "5fe08795b943eb2e757fa95cb6def1cb", + "info_dict": { + "duration": 93, + "title": "Teaser", + "upload_date": "20120721", + "uploader_id": "wb", }, } ] @@ -110,7 +112,8 @@ class AppleTrailersIE(InfoExtractor): 'width': format['width'], 'height': int(format['height']), }) - formats = sorted(formats, key=lambda f: (f['height'], f['width'])) + + self._sort_formats(formats) playlist.append({ '_type': 'video', diff --git a/youtube_dl/extractor/archiveorg.py b/youtube_dl/extractor/archiveorg.py index 8bb5464..34ce842 100644 --- a/youtube_dl/extractor/archiveorg.py +++ b/youtube_dl/extractor/archiveorg.py @@ -1,9 +1,10 @@ +from __future__ import unicode_literals + import json import re from .common import InfoExtractor from ..utils import ( - determine_ext, unified_strdate, ) @@ -13,23 +14,22 @@ class ArchiveOrgIE(InfoExtractor): IE_DESC = 'archive.org videos' _VALID_URL = r'(?:https?://)?(?:www\.)?archive\.org/details/(?P[^?/]+)(?:[?].*)?$' _TEST = { - u"url": u"http://archive.org/details/XD300-23_68HighlightsAResearchCntAugHumanIntellect", - u'file': u'XD300-23_68HighlightsAResearchCntAugHumanIntellect.ogv', - u'md5': u'8af1d4cf447933ed3c7f4871162602db', - u'info_dict': { - u"title": u"1968 Demo - FJCC Conference Presentation Reel #1", - u"description": u"Reel 1 of 3: Also known as the \"Mother of All Demos\", Doug Engelbart's presentation at the Fall Joint Computer Conference in San Francisco, December 9, 1968 titled \"A Research Center for Augmenting Human Intellect.\" For this presentation, Doug and his team astonished the audience by not only relating their research, but demonstrating it live. This was the debut of the mouse, interactive computing, hypermedia, computer supported software engineering, video teleconferencing, etc. See also Doug's 1968 Demo page for more background, highlights, links, and the detailed paper published in this conference proceedings. Filmed on 3 reels: Reel 1 | Reel 2 | Reel 3", - u"upload_date": u"19681210", - u"uploader": u"SRI International" + "url": "http://archive.org/details/XD300-23_68HighlightsAResearchCntAugHumanIntellect", + 'file': 'XD300-23_68HighlightsAResearchCntAugHumanIntellect.ogv', + 'md5': '8af1d4cf447933ed3c7f4871162602db', + 'info_dict': { + "title": "1968 Demo - FJCC Conference Presentation Reel #1", + "description": "Reel 1 of 3: Also known as the \"Mother of All Demos\", Doug Engelbart's presentation at the Fall Joint Computer Conference in San Francisco, December 9, 1968 titled \"A Research Center for Augmenting Human Intellect.\" For this presentation, Doug and his team astonished the audience by not only relating their research, but demonstrating it live. This was the debut of the mouse, interactive computing, hypermedia, computer supported software engineering, video teleconferencing, etc. See also Doug's 1968 Demo page for more background, highlights, links, and the detailed paper published in this conference proceedings. Filmed on 3 reels: Reel 1 | Reel 2 | Reel 3", + "upload_date": "19681210", + "uploader": "SRI International" } } - def _real_extract(self, url): mobj = re.match(self._VALID_URL, url) video_id = mobj.group('id') - json_url = url + (u'?' if u'?' in url else '&') + u'output=json' + json_url = url + ('?' if '?' in url else '&') + 'output=json' json_data = self._download_webpage(json_url, video_id) data = json.loads(json_data) @@ -38,16 +38,16 @@ class ArchiveOrgIE(InfoExtractor): uploader = data['metadata']['creator'][0] upload_date = unified_strdate(data['metadata']['date'][0]) - formats = [{ + formats = [ + { 'format': fdata['format'], 'url': 'http://' + data['server'] + data['dir'] + fn, 'file_size': int(fdata['size']), } - for fn,fdata in data['files'].items() + for fn, fdata in data['files'].items() if 'Video' in fdata['format']] - formats.sort(key=lambda fdata: fdata['file_size']) - for f in formats: - f['ext'] = determine_ext(f['url']) + + self._sort_formats(formats) return { '_type': 'video', diff --git a/youtube_dl/extractor/arte.py b/youtube_dl/extractor/arte.py index 9254fbf..7cf3785 100644 --- a/youtube_dl/extractor/arte.py +++ b/youtube_dl/extractor/arte.py @@ -1,4 +1,6 @@ # encoding: utf-8 +from __future__ import unicode_literals + import re import json @@ -22,7 +24,7 @@ class ArteTvIE(InfoExtractor): _LIVEWEB_URL = r'(?:http://)?liveweb\.arte\.tv/(?Pfr|de)/(?P.+?)/(?P.+)' _LIVE_URL = r'index-[0-9]+\.html$' - IE_NAME = u'arte.tv' + IE_NAME = 'arte.tv' @classmethod def suitable(cls, url): @@ -37,7 +39,7 @@ class ArteTvIE(InfoExtractor): # r'src="(.*?/videothek_js.*?\.js)', # 0, # [ - # (1, 'url', u'Invalid URL: %s' % url) + # (1, 'url', 'Invalid URL: %s' % url) # ] # ) # http_host = url.split('/')[2] @@ -49,12 +51,12 @@ class ArteTvIE(InfoExtractor): # '(rtmp://.*?)\'', # re.DOTALL, # [ - # (1, 'path', u'could not extract video path: %s' % url), - # (2, 'player', u'could not extract video player: %s' % url), - # (3, 'url', u'could not extract video url: %s' % url) + # (1, 'path', 'could not extract video path: %s' % url), + # (2, 'player', 'could not extract video player: %s' % url), + # (3, 'url', 'could not extract video url: %s' % url) # ] # ) - # video_url = u'%s/%s' % (info.get('url'), info.get('path')) + # video_url = '%s/%s' % (info.get('url'), info.get('path')) def _real_extract(self, url): mobj = re.match(self._VIDEOS_URL, url) @@ -107,9 +109,9 @@ class ArteTvIE(InfoExtractor): def _extract_liveweb(self, url, name, lang): """Extract form http://liveweb.arte.tv/""" webpage = self._download_webpage(url, name) - video_id = self._search_regex(r'eventId=(\d+?)("|&)', webpage, u'event id') + video_id = self._search_regex(r'eventId=(\d+?)("|&)', webpage, 'event id') config_doc = self._download_xml('http://download.liveweb.arte.tv/o21/liveweb/events/event-%s.xml' % video_id, - video_id, u'Downloading information') + video_id, 'Downloading information') event_doc = config_doc.find('event') url_node = event_doc.find('video').find('urlHd') if url_node is None: @@ -124,7 +126,7 @@ class ArteTvIE(InfoExtractor): class ArteTVPlus7IE(InfoExtractor): - IE_NAME = u'arte.tv:+7' + IE_NAME = 'arte.tv:+7' _VALID_URL = r'https?://www\.arte.tv/guide/(?Pfr|de)/(?:(?:sendungen|emissions)/)?(?P.*?)/(?P.*?)(\?.*)?' @classmethod @@ -207,7 +209,7 @@ class ArteTVPlus7IE(InfoExtractor): 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']) + format_id = '%s-%s' % (quality, format_info['versionCode']) else: format_id = quality info = { @@ -216,7 +218,7 @@ class ArteTVPlus7IE(InfoExtractor): 'width': format_info.get('width'), 'height': height, } - if format_info['mediaType'] == u'rtmp': + if format_info['mediaType'] == 'rtmp': info['url'] = format_info['streamer'] info['play_path'] = 'mp4:' + format_info['url'] info['ext'] = 'flv' @@ -231,27 +233,27 @@ class ArteTVPlus7IE(InfoExtractor): # It also uses the arte_vp_url url from the webpage to extract the information class ArteTVCreativeIE(ArteTVPlus7IE): - IE_NAME = u'arte.tv:creative' + IE_NAME = 'arte.tv:creative' _VALID_URL = r'https?://creative\.arte\.tv/(?Pfr|de)/magazine?/(?P.+)' _TEST = { - 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 / Agence Amateur #2 : Corporate Design', + 'url': 'http://creative.arte.tv/de/magazin/agentur-amateur-corporate-design', + 'file': '050489-002.mp4', + 'info_dict': { + 'title': 'Agentur Amateur / Agence Amateur #2 : Corporate Design', }, } class ArteTVFutureIE(ArteTVPlus7IE): - IE_NAME = u'arte.tv:future' + IE_NAME = 'arte.tv:future' _VALID_URL = r'https?://future\.arte\.tv/(?Pfr|de)/(thema|sujet)/.*?#article-anchor-(?P\d+)' _TEST = { - u'url': u'http://future.arte.tv/fr/sujet/info-sciences#article-anchor-7081', - u'file': u'050940-003.mp4', - u'info_dict': { - u'title': u'Les champignons au secours de la planète', + 'url': 'http://future.arte.tv/fr/sujet/info-sciences#article-anchor-7081', + 'file': '050940-003.mp4', + 'info_dict': { + 'title': 'Les champignons au secours de la planète', }, } @@ -263,7 +265,7 @@ class ArteTVFutureIE(ArteTVPlus7IE): class ArteTVDDCIE(ArteTVPlus7IE): - IE_NAME = u'arte.tv:ddc' + IE_NAME = 'arte.tv:ddc' _VALID_URL = r'http?://ddc\.arte\.tv/(?Pemission|folge)/(?P.+)' def _real_extract(self, url): diff --git a/youtube_dl/extractor/auengine.py b/youtube_dl/extractor/auengine.py index bcccc0b..c6f30e6 100644 --- a/youtube_dl/extractor/auengine.py +++ b/youtube_dl/extractor/auengine.py @@ -1,3 +1,5 @@ +from __future__ import unicode_literals + import re from .common import InfoExtractor @@ -7,13 +9,14 @@ from ..utils import ( ExtractorError, ) + class AUEngineIE(InfoExtractor): _TEST = { - u'url': u'http://auengine.com/embed.php?file=lfvlytY6&w=650&h=370', - u'file': u'lfvlytY6.mp4', - u'md5': u'48972bdbcf1a3a2f5533e62425b41d4f', - u'info_dict': { - u"title": u"[Commie]The Legend of the Legendary Heroes - 03 - Replication Eye (Alpha Stigma)[F9410F5A]" + 'url': 'http://auengine.com/embed.php?file=lfvlytY6&w=650&h=370', + 'file': 'lfvlytY6.mp4', + 'md5': '48972bdbcf1a3a2f5533e62425b41d4f', + 'info_dict': { + 'title': '[Commie]The Legend of the Legendary Heroes - 03 - Replication Eye (Alpha Stigma)[F9410F5A]' } } _VALID_URL = r'(?:http://)?(?:www\.)?auengine\.com/embed\.php\?.*?file=([^&]+).*?' @@ -23,7 +26,7 @@ class AUEngineIE(InfoExtractor): video_id = mobj.group(1) webpage = self._download_webpage(url, video_id) title = self._html_search_regex(r'(?P<title>.+?)', - webpage, u'title') + webpage, 'title') title = title.strip() links = re.findall(r'\s(?:file|url):\s*["\']([^\'"]+)["\']', webpage) links = map(compat_urllib_parse.unquote, links) @@ -37,7 +40,7 @@ class AUEngineIE(InfoExtractor): video_url = link if not video_url: raise ExtractorError(u'Could not find video URL') - ext = u'.' + determine_ext(video_url) + ext = '.' + determine_ext(video_url) if ext == title[-len(ext):]: title = title[:-len(ext)] diff --git a/youtube_dl/extractor/bambuser.py b/youtube_dl/extractor/bambuser.py index d48c0c3..ccd31c4 100644 --- a/youtube_dl/extractor/bambuser.py +++ b/youtube_dl/extractor/bambuser.py @@ -1,3 +1,5 @@ +from __future__ import unicode_literals + import re import json import itertools @@ -9,26 +11,26 @@ from ..utils import ( class BambuserIE(InfoExtractor): - IE_NAME = u'bambuser' + IE_NAME = 'bambuser' _VALID_URL = r'https?://bambuser\.com/v/(?P\d+)' _API_KEY = '005f64509e19a868399060af746a00aa' _TEST = { - u'url': u'http://bambuser.com/v/4050584', + 'url': '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', + #u'md5': 'fba8f7693e48fd4e8641b3fd5539a641', + 'info_dict': { + 'id': '4050584', + 'ext': 'flv', + 'title': 'Education engineering days - lightning talks', + 'duration': 3741, + 'uploader': 'pixelversity', + 'uploader_id': '344706', }, - u'params': { + 'params': { # It doesn't respect the 'Range' header, it would download the whole video # caused the travis builds to fail: https://travis-ci.org/rg3/youtube-dl/jobs/14493845#L59 - u'skip_download': True, + 'skip_download': True, }, } @@ -53,7 +55,7 @@ class BambuserIE(InfoExtractor): class BambuserChannelIE(InfoExtractor): - IE_NAME = u'bambuser:channel' + IE_NAME = 'bambuser:channel' _VALID_URL = r'https?://bambuser\.com/channel/(?P.*?)(?:/|#|\?|$)' # The maximum number we can get with each request _STEP = 50 @@ -72,7 +74,7 @@ class BambuserChannelIE(InfoExtractor): # 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) + 'Downloading page %d' % i) results = json.loads(info_json)['result'] if len(results) == 0: break diff --git a/youtube_dl/extractor/bandcamp.py b/youtube_dl/extractor/bandcamp.py index 3a32c14..886b0df 100644 --- a/youtube_dl/extractor/bandcamp.py +++ b/youtube_dl/extractor/bandcamp.py @@ -1,3 +1,5 @@ +from __future__ import unicode_literals + import json import re @@ -10,16 +12,16 @@ from ..utils import ( class BandcampIE(InfoExtractor): - IE_NAME = u'Bandcamp' _VALID_URL = r'http://.*?\.bandcamp\.com/track/(?P.*)' _TESTS = [{ - u'url': u'http://youtube-dl.bandcamp.com/track/youtube-dl-test-song', - u'file': u'1812978515.mp3', - u'md5': u'cdeb30cdae1921719a3cbcab696ef53c', - u'info_dict': { - u"title": u"youtube-dl test song \"'/\\\u00e4\u21ad" + 'url': 'http://youtube-dl.bandcamp.com/track/youtube-dl-test-song', + 'file': '1812978515.mp3', + 'md5': 'c557841d5e50261777a6585648adf439', + 'info_dict': { + "title": "youtube-dl \"'/\\\u00e4\u21ad - youtube-dl test song \"'/\\\u00e4\u21ad", + "duration": 10, }, - u'skip': u'There is a limit of 200 free downloads / month for the test song' + '_skip': 'There is a limit of 200 free downloads / month for the test song' }] def _real_extract(self, url): @@ -30,85 +32,98 @@ class BandcampIE(InfoExtractor): m_download = re.search(r'freeDownloadPage: "(.*?)"', webpage) if m_download is None: m_trackinfo = re.search(r'trackinfo: (.+),\s*?\n', webpage) - if m_trackinfo: - json_code = m_trackinfo.group(1) - data = json.loads(json_code) - - for d in data: - formats = [{ - 'format_id': 'format_id', - 'url': format_url, - 'ext': format_id.partition('-')[0] - } for format_id, format_url in sorted(d['file'].items())] + if m_trackinfo: + json_code = m_trackinfo.group(1) + data = json.loads(json_code) + d = data[0] + + duration = int(round(d['duration'])) + formats = [] + for format_id, format_url in d['file'].items(): + ext, _, abr_str = format_id.partition('-') + + formats.append({ + 'format_id': format_id, + 'url': format_url, + 'ext': format_id.partition('-')[0], + 'vcodec': 'none', + 'acodec': format_id.partition('-')[0], + 'abr': int(format_id.partition('-')[2]), + }) + + self._sort_formats(formats) + return { 'id': compat_str(d['id']), 'title': d['title'], 'formats': formats, + 'duration': duration, } - else: - raise ExtractorError(u'No free songs found') + else: + raise ExtractorError('No free songs found') download_link = m_download.group(1) - id = re.search(r'var TralbumData = {(.*?)id: (?P<id>\d*?)$', - webpage, re.MULTILINE|re.DOTALL).group('id') + video_id = re.search( + r'var TralbumData = {(.*?)id: (?P<id>\d*?)$', + webpage, re.MULTILINE | re.DOTALL).group('id') - download_webpage = self._download_webpage(download_link, id, + download_webpage = self._download_webpage(download_link, video_id, 'Downloading free downloads page') # We get the dictionary of the track from some javascrip code info = re.search(r'items: (.*?),$', download_webpage, re.MULTILINE).group(1) info = json.loads(info)[0] # We pick mp3-320 for now, until format selection can be easily implemented. - mp3_info = info[u'downloads'][u'mp3-320'] + mp3_info = info['downloads']['mp3-320'] # If we try to use this url it says the link has expired - initial_url = mp3_info[u'url'] + initial_url = mp3_info['url'] re_url = r'(?P<server>http://(.*?)\.bandcamp\.com)/download/track\?enc=mp3-320&fsig=(?P<fsig>.*?)&id=(?P<id>.*?)&ts=(?P<ts>.*)$' m_url = re.match(re_url, initial_url) #We build the url we will use to get the final track url # This url is build in Bandcamp in the script download_bunde_*.js - request_url = '%s/statdownload/track?enc=mp3-320&fsig=%s&id=%s&ts=%s&.rand=665028774616&.vrs=1' % (m_url.group('server'), m_url.group('fsig'), id, m_url.group('ts')) - final_url_webpage = self._download_webpage(request_url, id, 'Requesting download url') + request_url = '%s/statdownload/track?enc=mp3-320&fsig=%s&id=%s&ts=%s&.rand=665028774616&.vrs=1' % (m_url.group('server'), m_url.group('fsig'), video_id, m_url.group('ts')) + final_url_webpage = self._download_webpage(request_url, video_id, 'Requesting download url') # If we could correctly generate the .rand field the url would be #in the "download_url" key final_url = re.search(r'"retry_url":"(.*?)"', final_url_webpage).group(1) - track_info = {'id':id, - 'title' : info[u'title'], - 'ext' : 'mp3', - 'url' : final_url, - 'thumbnail' : info[u'thumb_url'], - 'uploader' : info[u'artist'] - } - - return [track_info] + return { + 'id': video_id, + 'title': info['title'], + 'ext': 'mp3', + 'vcodec': 'none', + 'url': final_url, + 'thumbnail': info.get('thumb_url'), + 'uploader': info.get('artist'), + } class BandcampAlbumIE(InfoExtractor): - IE_NAME = u'Bandcamp:album' + IE_NAME = 'Bandcamp:album' _VALID_URL = r'http://.*?\.bandcamp\.com/album/(?P<title>.*)' _TEST = { - u'url': u'http://blazo.bandcamp.com/album/jazz-format-mixtape-vol-1', - u'playlist': [ + 'url': 'http://blazo.bandcamp.com/album/jazz-format-mixtape-vol-1', + 'playlist': [ { - u'file': u'1353101989.mp3', - u'md5': u'39bc1eded3476e927c724321ddf116cf', - u'info_dict': { - u'title': u'Intro', + 'file': '1353101989.mp3', + 'md5': '39bc1eded3476e927c724321ddf116cf', + 'info_dict': { + 'title': 'Intro', } }, { - u'file': u'38097443.mp3', - u'md5': u'1a2c32e2691474643e912cc6cd4bffaa', - u'info_dict': { - u'title': u'Kero One - Keep It Alive (Blazo remix)', + 'file': '38097443.mp3', + 'md5': '1a2c32e2691474643e912cc6cd4bffaa', + 'info_dict': { + 'title': 'Kero One - Keep It Alive (Blazo remix)', } }, ], - u'params': { - u'playlistend': 2 + 'params': { + 'playlistend': 2 }, - u'skip': u'Bancamp imposes download limits. See test_playlists:test_bandcamp_album for the playlist test' + 'skip': 'Bancamp imposes download limits. See test_playlists:test_bandcamp_album for the playlist test' } def _real_extract(self, url): @@ -117,11 +132,11 @@ class BandcampAlbumIE(InfoExtractor): webpage = self._download_webpage(url, title) tracks_paths = re.findall(r'<a href="(.*?)" itemprop="url">', webpage) if not tracks_paths: - raise ExtractorError(u'The page doesn\'t contain any track') + raise ExtractorError('The page doesn\'t contain any tracks') entries = [ self.url_result(compat_urlparse.urljoin(url, t_path), ie=BandcampIE.ie_key()) for t_path in tracks_paths] - title = self._search_regex(r'album_title : "(.*?)"', webpage, u'title') + title = self._search_regex(r'album_title : "(.*?)"', webpage, 'title') return { '_type': 'playlist', 'title': title, diff --git a/youtube_dl/extractor/blinkx.py b/youtube_dl/extractor/blinkx.py index 144ce64..96408e4 100644 --- a/youtube_dl/extractor/blinkx.py +++ b/youtube_dl/extractor/blinkx.py @@ -1,3 +1,5 @@ +from __future__ import unicode_literals + import datetime import json import re @@ -10,19 +12,19 @@ from ..utils import ( class BlinkxIE(InfoExtractor): _VALID_URL = r'^(?:https?://(?:www\.)blinkx\.com/#?ce/|blinkx:)(?P<id>[^?]+)' - _IE_NAME = u'blinkx' + IE_NAME = 'blinkx' _TEST = { - u'url': u'http://www.blinkx.com/ce/8aQUy7GVFYgFzpKhT0oqsilwOGFRVXk3R1ZGWWdGenBLaFQwb3FzaWx3OGFRVXk3R1ZGWWdGenB', - u'file': u'8aQUy7GV.mp4', - u'md5': u'2e9a07364af40163a908edbf10bb2492', - u'info_dict': { - u"title": u"Police Car Rolls Away", - u"uploader": u"stupidvideos.com", - u"upload_date": u"20131215", - u"description": u"A police car gently rolls away from a fight. Maybe it felt weird being around a confrontation and just had to get out of there!", - u"duration": 14.886, - u"thumbnails": [{ + 'url': 'http://www.blinkx.com/ce/8aQUy7GVFYgFzpKhT0oqsilwOGFRVXk3R1ZGWWdGenBLaFQwb3FzaWx3OGFRVXk3R1ZGWWdGenB', + 'file': '8aQUy7GV.mp4', + 'md5': '2e9a07364af40163a908edbf10bb2492', + 'info_dict': { + "title": "Police Car Rolls Away", + "uploader": "stupidvideos.com", + "upload_date": "20131215", + "description": "A police car gently rolls away from a fight. Maybe it felt weird being around a confrontation and just had to get out of there!", + "duration": 14.886, + "thumbnails": [{ "width": 100, "height": 76, "url": "http://cdn.blinkx.com/stream/b/41/StupidVideos/20131215/1873969261/1873969261_tn_0.jpg", @@ -30,17 +32,17 @@ class BlinkxIE(InfoExtractor): }, } - def _real_extract(self, url): - m = re.match(self._VALID_URL, url) + def _real_extract(self, rl): + m = re.match(self._VALID_URL, rl) video_id = m.group('id') display_id = video_id[:8] api_url = (u'https://apib4.blinkx.com/api.php?action=play_video&' + - u'video=%s' % video_id) + 'video=%s' % video_id) data_json = self._download_webpage(api_url, display_id) data = json.loads(data_json)['api']['results'][0] dt = datetime.datetime.fromtimestamp(data['pubdate_epoch']) - upload_date = dt.strftime('%Y%m%d') + pload_date = dt.strftime('%Y%m%d') duration = None thumbnails = [] @@ -61,9 +63,10 @@ class BlinkxIE(InfoExtractor): elif m['type'] in ('flv', 'mp4'): vcodec = remove_start(m['vcodec'], 'ff') acodec = remove_start(m['acodec'], 'ff') + tbr = (int(m['vbr']) + int(m['abr'])) // 1000 format_id = (u'%s-%sk-%s' % (vcodec, - (int(m['vbr']) + int(m['abr'])) // 1000, + tbr, m['w'])) formats.append({ 'format_id': format_id, @@ -72,10 +75,12 @@ class BlinkxIE(InfoExtractor): 'acodec': acodec, 'abr': int(m['abr']) // 1000, 'vbr': int(m['vbr']) // 1000, + 'tbr': tbr, 'width': int(m['w']), 'height': int(m['h']), }) - formats.sort(key=lambda f: (f['width'], f['vbr'], f['abr'])) + + self._sort_formats(formats) return { 'id': display_id, @@ -83,7 +88,7 @@ class BlinkxIE(InfoExtractor): 'title': data['title'], 'formats': formats, 'uploader': data['channel_name'], - 'upload_date': upload_date, + 'upload_date': pload_date, 'description': data.get('description'), 'thumbnails': thumbnails, 'duration': duration, diff --git a/youtube_dl/extractor/bliptv.py b/youtube_dl/extractor/bliptv.py index 5e33a69..3ce9b53 100644 --- a/youtube_dl/extractor/bliptv.py +++ b/youtube_dl/extractor/bliptv.py @@ -1,16 +1,15 @@ +from __future__ import unicode_literals + import datetime import json -import os import re import socket from .common import InfoExtractor from ..utils import ( compat_http_client, - compat_parse_qs, compat_str, compat_urllib_error, - compat_urllib_parse_urlparse, compat_urllib_request, ExtractorError, @@ -22,42 +21,35 @@ class BlipTVIE(InfoExtractor): """Information extractor for blip.tv""" _VALID_URL = r'^(?:https?://)?(?:\w+\.)?blip\.tv/((.+/)|(play/)|(api\.swf#))(.+)$' - _URL_EXT = r'^.*\.([a-z0-9]+)$' - IE_NAME = u'blip.tv' + _TEST = { - u'url': u'http://blip.tv/cbr/cbr-exclusive-gotham-city-imposters-bats-vs-jokerz-short-3-5796352', - u'file': u'5779306.m4v', - u'md5': u'80baf1ec5c3d2019037c1c707d676b9f', - u'info_dict': { - u"upload_date": u"20111205", - u"description": u"md5:9bc31f227219cde65e47eeec8d2dc596", - u"uploader": u"Comic Book Resources - CBR TV", - u"title": u"CBR EXCLUSIVE: \"Gotham City Imposters\" Bats VS Jokerz Short 3" + 'url': 'http://blip.tv/cbr/cbr-exclusive-gotham-city-imposters-bats-vs-jokerz-short-3-5796352', + 'file': '5779306.mov', + 'md5': 'c6934ad0b6acf2bd920720ec888eb812', + 'info_dict': { + 'upload_date': '20111205', + 'description': 'md5:9bc31f227219cde65e47eeec8d2dc596', + 'uploader': 'Comic Book Resources - CBR TV', + 'title': 'CBR EXCLUSIVE: "Gotham City Imposters" Bats VS Jokerz Short 3', } } def report_direct_download(self, title): """Report information extraction.""" - self.to_screen(u'%s: Direct download detected' % title) + self.to_screen('%s: Direct download detected' % title) def _real_extract(self, url): mobj = re.match(self._VALID_URL, url) if mobj is None: - raise ExtractorError(u'Invalid URL: %s' % url) + raise ExtractorError('Invalid URL: %s' % url) # See https://github.com/rg3/youtube-dl/issues/857 - api_mobj = re.match(r'http://a\.blip\.tv/api\.swf#(?P<video_id>[\d\w]+)', url) - if api_mobj is not None: - url = 'http://blip.tv/play/g_%s' % api_mobj.group('video_id') - urlp = compat_urllib_parse_urlparse(url) - if urlp.path.startswith('/play/'): - response = self._request_webpage(url, None, False) - redirecturl = response.geturl() - rurlp = compat_urllib_parse_urlparse(redirecturl) - file_id = compat_parse_qs(rurlp.fragment)['file'][0].rpartition('/')[2] - url = 'http://blip.tv/a/a-' + file_id - return self._real_extract(url) - + embed_mobj = re.search(r'^(?:https?://)?(?:\w+\.)?blip\.tv/(?:play/|api\.swf#)([a-zA-Z0-9]+)', url) + if embed_mobj: + info_url = 'http://blip.tv/play/%s.x?p=1' % embed_mobj.group(1) + info_page = self._download_webpage(info_url, embed_mobj.group(1)) + video_id = self._search_regex(r'data-episode-id="(\d+)', info_page, 'video_id') + return self.url_result('http://blip.tv/a/a-' + video_id, 'BlipTV') if '?' in url: cchar = '&' @@ -67,67 +59,55 @@ class BlipTVIE(InfoExtractor): request = compat_urllib_request.Request(json_url) request.add_header('User-Agent', 'iTunes/10.6.1') self.report_extraction(mobj.group(1)) - info = None urlh = self._request_webpage(request, None, False, - u'unable to download video info webpage') - if urlh.headers.get('Content-Type', '').startswith('video/'): # Direct download - basename = url.split('/')[-1] - title,ext = os.path.splitext(basename) - title = title.decode('UTF-8') - ext = ext.replace('.', '') - self.report_direct_download(title) - info = { - 'id': title, - 'url': url, - 'uploader': None, - 'upload_date': None, - 'title': title, - 'ext': ext, - 'urlhandle': urlh + 'unable to download video info webpage') + + try: + json_code_bytes = urlh.read() + json_code = json_code_bytes.decode('utf-8') + except (compat_urllib_error.URLError, compat_http_client.HTTPException, socket.error) as err: + raise ExtractorError('Unable to read video info webpage: %s' % compat_str(err)) + + try: + json_data = json.loads(json_code) + if 'Post' in json_data: + data = json_data['Post'] + else: + data = json_data + + upload_date = datetime.datetime.strptime(data['datestamp'], '%m-%d-%y %H:%M%p').strftime('%Y%m%d') + formats = [] + if 'additionalMedia' in data: + for f in sorted(data['additionalMedia'], key=lambda f: int(f['media_height'])): + if not int(f['media_width']): # filter m3u8 + continue + formats.append({ + 'url': f['url'], + 'format_id': f['role'], + 'width': int(f['media_width']), + 'height': int(f['media_height']), + }) + else: + formats.append({ + 'url': data['media']['url'], + 'width': int(data['media']['width']), + 'height': int(data['media']['height']), + }) + + self._sort_formats(formats) + + return { + 'id': compat_str(data['item_id']), + 'uploader': data['display_name'], + 'upload_date': upload_date, + 'title': data['title'], + 'thumbnail': data['thumbnailUrl'], + 'description': data['description'], + 'user_agent': 'iTunes/10.6.1', + 'formats': formats, } - if info is None: # Regular URL - try: - json_code_bytes = urlh.read() - json_code = json_code_bytes.decode('utf-8') - except (compat_urllib_error.URLError, compat_http_client.HTTPException, socket.error) as err: - raise ExtractorError(u'Unable to read video info webpage: %s' % compat_str(err)) - - try: - json_data = json.loads(json_code) - if 'Post' in json_data: - data = json_data['Post'] - else: - data = json_data - - upload_date = datetime.datetime.strptime(data['datestamp'], '%m-%d-%y %H:%M%p').strftime('%Y%m%d') - if 'additionalMedia' in data: - formats = sorted(data['additionalMedia'], key=lambda f: int(f['media_height'])) - best_format = formats[-1] - video_url = best_format['url'] - else: - video_url = data['media']['url'] - umobj = re.match(self._URL_EXT, video_url) - if umobj is None: - raise ValueError('Can not determine filename extension') - ext = umobj.group(1) - - info = { - 'id': compat_str(data['item_id']), - 'url': video_url, - 'uploader': data['display_name'], - 'upload_date': upload_date, - 'title': data['title'], - 'ext': ext, - 'format': data['media']['mimeType'], - 'thumbnail': data['thumbnailUrl'], - 'description': data['description'], - 'player_url': data['embedUrl'], - 'user_agent': 'iTunes/10.6.1', - } - except (ValueError,KeyError) as err: - raise ExtractorError(u'Unable to parse video information: %s' % repr(err)) - - return [info] + except (ValueError, KeyError) as err: + raise ExtractorError('Unable to parse video information: %s' % repr(err)) class BlipTVUserIE(InfoExtractor): @@ -135,19 +115,19 @@ class BlipTVUserIE(InfoExtractor): _VALID_URL = r'(?:(?:(?:https?://)?(?:\w+\.)?blip\.tv/)|bliptvuser:)([^/]+)/*$' _PAGE_SIZE = 12 - IE_NAME = u'blip.tv:user' + IE_NAME = 'blip.tv:user' def _real_extract(self, url): # Extract username mobj = re.match(self._VALID_URL, url) if mobj is None: - raise ExtractorError(u'Invalid URL: %s' % url) + raise ExtractorError('Invalid URL: %s' % url) username = mobj.group(1) page_base = 'http://m.blip.tv/pr/show_get_full_episode_list?users_id=%s&lite=0&esi=1' - page = self._download_webpage(url, username, u'Downloading user page') + page = self._download_webpage(url, username, 'Downloading user page') mobj = re.search(r'data-users-id="([^"]+)"', page) page_base = page_base % mobj.group(1) @@ -163,7 +143,7 @@ class BlipTVUserIE(InfoExtractor): while True: url = page_base + "&page=" + str(pagenum) page = self._download_webpage(url, username, - u'Downloading video ids from page %d' % pagenum) + 'Downloading video ids from page %d' % pagenum) # Extract video identifiers ids_in_page = [] @@ -185,6 +165,6 @@ class BlipTVUserIE(InfoExtractor): pagenum += 1 - urls = [u'http://blip.tv/%s' % video_id for video_id in video_ids] + urls = ['http://blip.tv/%s' % video_id for video_id in video_ids] url_entries = [self.url_result(vurl, 'BlipTV') for vurl in urls] return [self.playlist_result(url_entries, playlist_title = username)] diff --git a/youtube_dl/extractor/bloomberg.py b/youtube_dl/extractor/bloomberg.py index 755d9c9..d18bc7e 100644 --- a/youtube_dl/extractor/bloomberg.py +++ b/youtube_dl/extractor/bloomberg.py @@ -1,6 +1,7 @@ import re from .common import InfoExtractor +from .ooyala import OoyalaIE class BloombergIE(InfoExtractor): @@ -23,5 +24,5 @@ class BloombergIE(InfoExtractor): mobj = re.match(self._VALID_URL, url) name = mobj.group('name') webpage = self._download_webpage(url, name) - ooyala_url = self._og_search_video_url(webpage) - return self.url_result(ooyala_url, ie='Ooyala') + ooyala_code = self._search_regex(r'<source src="http://player.ooyala.com/player/[^/]+/([^".]+)', webpage, u'ooyala url') + return OoyalaIE._build_url_result(ooyala_code) diff --git a/youtube_dl/extractor/brightcove.py b/youtube_dl/extractor/brightcove.py index f7f0041..8ac38f4 100644 --- a/youtube_dl/extractor/brightcove.py +++ b/youtube_dl/extractor/brightcove.py @@ -1,4 +1,5 @@ # encoding: utf-8 +from __future__ import unicode_literals import re import json @@ -13,6 +14,7 @@ from ..utils import ( compat_urllib_request, ExtractorError, + unsmuggle_url, ) @@ -24,47 +26,47 @@ class BrightcoveIE(InfoExtractor): _TESTS = [ { # 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'5423e113865d26e40624dce2e4b45d95', - 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”', - u'uploader': u'8TV', - u'description': u'md5:a950cc4285c43e44d763d036710cd9cd', + 'url': 'http://c.brightcove.com/services/viewer/htmlFederated?playerID=1654948606001&flashID=myExperience&%40videoPlayer=2371591881001', + 'file': '2371591881001.mp4', + 'md5': '5423e113865d26e40624dce2e4b45d95', + 'note': 'Test Brightcove downloads and detection in GenericIE', + 'info_dict': { + 'title': 'Xavier Sala i Martín: “Un banc que no presta és un banc zombi que no serveix per a res”', + 'uploader': '8TV', + 'description': 'md5:a950cc4285c43e44d763d036710cd9cd', } }, { # From http://medianetwork.oracle.com/video/player/1785452137001 - u'url': u'http://c.brightcove.com/services/viewer/htmlFederated?playerID=1217746023001&flashID=myPlayer&%40videoPlayer=1785452137001', - u'file': u'1785452137001.flv', - u'info_dict': { - u'title': u'JVMLS 2012: Arrays 2.0 - Opportunities and Challenges', - u'description': u'John Rose speaks at the JVM Language Summit, August 1, 2012.', - u'uploader': u'Oracle', + 'url': 'http://c.brightcove.com/services/viewer/htmlFederated?playerID=1217746023001&flashID=myPlayer&%40videoPlayer=1785452137001', + 'file': '1785452137001.flv', + 'info_dict': { + 'title': 'JVMLS 2012: Arrays 2.0 - Opportunities and Challenges', + 'description': 'John Rose speaks at the JVM Language Summit, August 1, 2012.', + 'uploader': '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', + 'url': 'http://c.brightcove.com/services/viewer/federated_f9?&playerID=1265504713001&publisherID=AQ%7E%7E%2CAAABBzUwv1E%7E%2CxP-xFHVUstiMFlNYfvF4G9yFnNaqCw_9&videoID=2750934548001', + 'info_dict': { + 'id': '2750934548001', + 'ext': 'mp4', + 'title': 'This Bracelet Acts as a Personal Thermostat', + 'description': 'md5:547b78c64f4112766ccf4e151c20b6a0', + 'uploader': 'Mashable', }, }, { # test that the default referer works # from http://national.ballet.ca/interact/video/Lost_in_Motion_II/ - u'url': u'http://link.brightcove.com/services/player/bcpid756015033001?bckey=AQ~~,AAAApYJi_Ck~,GxhXCegT1Dp39ilhXuxMJxasUhVNZiil&bctid=2878862109001', - u'info_dict': { - u'id': u'2878862109001', - u'ext': u'mp4', - u'title': u'Lost in Motion II', - u'description': u'md5:363109c02998fee92ec02211bd8000df', - u'uploader': u'National Ballet of Canada', + 'url': 'http://link.brightcove.com/services/player/bcpid756015033001?bckey=AQ~~,AAAApYJi_Ck~,GxhXCegT1Dp39ilhXuxMJxasUhVNZiil&bctid=2878862109001', + 'info_dict': { + 'id': '2878862109001', + 'ext': 'mp4', + 'title': 'Lost in Motion II', + 'description': 'md5:363109c02998fee92ec02211bd8000df', + 'uploader': 'National Ballet of Canada', }, }, ] @@ -80,13 +82,13 @@ class BrightcoveIE(InfoExtractor): object_str = re.sub(r'(<param name="[^"]+" value="[^"]+")>', lambda m: m.group(1) + '/>', object_str) # Fix up some stupid XML, see https://github.com/rg3/youtube-dl/issues/1608 - object_str = object_str.replace(u'<--', u'<!--') + object_str = object_str.replace('<--', '<!--') object_doc = xml.etree.ElementTree.fromstring(object_str) - assert u'BrightcoveExperience' in object_doc.attrib['class'] - params = {'flashID': object_doc.attrib['id'], - 'playerID': find_xpath_attr(object_doc, './param', 'name', 'playerID').attrib['value'], - } + assert 'BrightcoveExperience' in object_doc.attrib['class'] + params = { + 'playerID': find_xpath_attr(object_doc, './param', 'name', 'playerID').attrib['value'], + } def find_param(name): node = find_xpath_attr(object_doc, './param', 'name', name) if node is not None: @@ -120,6 +122,8 @@ class BrightcoveIE(InfoExtractor): return None def _real_extract(self, url): + url, smuggled_data = unsmuggle_url(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 @@ -130,9 +134,10 @@ class BrightcoveIE(InfoExtractor): videoPlayer = query.get('@videoPlayer') if videoPlayer: - return self._get_video_info(videoPlayer[0], query_str, query, - # We set the original url as the default 'Referer' header - referer=url) + # We set the original url as the default 'Referer' header + referer = smuggled_data.get('Referer', url) + return self._get_video_info( + videoPlayer[0], query_str, query, referer=referer) else: player_key = query['playerKey'] return self._get_playlist_info(player_key[0]) @@ -156,11 +161,11 @@ class BrightcoveIE(InfoExtractor): def _get_playlist_info(self, player_key): playlist_info = self._download_webpage(self._PLAYLIST_URL_TEMPLATE % player_key, - player_key, u'Downloading playlist information') + player_key, 'Downloading playlist information') json_data = json.loads(playlist_info) if 'videoList' not in json_data: - raise ExtractorError(u'Empty playlist') + raise ExtractorError('Empty playlist') playlist_info = json_data['videoList'] videos = [self._extract_video_info(video_info) for video_info in playlist_info['mediaCollectionDTO']['videoDTOs']] @@ -189,5 +194,5 @@ class BrightcoveIE(InfoExtractor): 'url': video_info['FLVFullLengthURL'], }) else: - raise ExtractorError(u'Unable to extract video url for %s' % info['id']) + raise ExtractorError('Unable to extract video url for %s' % info['id']) return info diff --git a/youtube_dl/extractor/c56.py b/youtube_dl/extractor/c56.py index dc3a8d4..690bc7c 100644 --- a/youtube_dl/extractor/c56.py +++ b/youtube_dl/extractor/c56.py @@ -1,21 +1,21 @@ # coding: utf-8 +from __future__ import unicode_literals import re import json from .common import InfoExtractor -from ..utils import determine_ext + class C56IE(InfoExtractor): _VALID_URL = r'https?://((www|player)\.)?56\.com/(.+?/)?(v_|(play_album.+-))(?P<textid>.+?)\.(html|swf)' - IE_NAME = u'56.com' - - _TEST ={ - u'url': u'http://www.56.com/u39/v_OTM0NDA3MTY.html', - u'file': u'93440716.flv', - u'md5': u'e59995ac63d0457783ea05f93f12a866', - u'info_dict': { - u'title': u'网事知多少 第32期:车怒', + IE_NAME = '56.com' + _TEST = { + 'url': 'http://www.56.com/u39/v_OTM0NDA3MTY.html', + 'file': '93440716.flv', + 'md5': 'e59995ac63d0457783ea05f93f12a866', + 'info_dict': { + 'title': '网事知多少 第32期:车怒', }, } @@ -23,14 +23,18 @@ class C56IE(InfoExtractor): mobj = re.match(self._VALID_URL, url, flags=re.VERBOSE) text_id = mobj.group('textid') info_page = self._download_webpage('http://vxml.56.com/json/%s/' % text_id, - text_id, u'Downloading video info') + text_id, 'Downloading video info') info = json.loads(info_page)['info'] - best_format = sorted(info['rfiles'], key=lambda f: int(f['filesize']))[-1] - video_url = best_format['url'] + formats = [{ + 'format_id': f['type'], + 'filesize': int(f['filesize']), + 'url': f['url'] + } for f in info['rfiles']] + self._sort_formats(formats) - return {'id': info['vid'], - 'title': info['Subject'], - 'url': video_url, - 'ext': determine_ext(video_url), - 'thumbnail': info.get('bimg') or info.get('img'), - } + return { + 'id': info['vid'], + 'title': info['Subject'], + 'formats': formats, + 'thumbnail': info.get('bimg') or info.get('img'), + } diff --git a/youtube_dl/extractor/channel9.py b/youtube_dl/extractor/channel9.py index ae70ea2..574881b 100644 --- a/youtube_dl/extractor/channel9.py +++ b/youtube_dl/extractor/channel9.py @@ -76,14 +76,18 @@ class Channel9IE(InfoExtractor): </div>)? # File size part may be missing ''' # Extract known formats - formats = [{'url': x.group('url'), - 'format_id': x.group('quality'), - 'format_note': x.group('note'), - 'format': '%s (%s)' % (x.group('quality'), x.group('note')), - 'filesize': self._restore_bytes(x.group('filesize')), # File size is approximate - } for x in list(re.finditer(FORMAT_REGEX, html)) if x.group('quality') in self._known_formats] - # Sort according to known formats list - formats.sort(key=lambda fmt: self._known_formats.index(fmt['format_id'])) + formats = [{ + 'url': x.group('url'), + 'format_id': x.group('quality'), + 'format_note': x.group('note'), + 'format': u'%s (%s)' % (x.group('quality'), x.group('note')), + 'filesize': self._restore_bytes(x.group('filesize')), # File size is approximate + 'preference': self._known_formats.index(x.group('quality')), + 'vcodec': 'none' if x.group('note') == 'Audio only' else None, + } for x in list(re.finditer(FORMAT_REGEX, html)) if x.group('quality') in self._known_formats] + + self._sort_formats(formats) + return formats def _extract_title(self, html): diff --git a/youtube_dl/extractor/cmt.py b/youtube_dl/extractor/cmt.py new file mode 100644 index 0000000..88e0e9a --- /dev/null +++ b/youtube_dl/extractor/cmt.py @@ -0,0 +1,19 @@ +from .mtv import MTVIE + +class CMTIE(MTVIE): + IE_NAME = u'cmt.com' + _VALID_URL = r'https?://www\.cmt\.com/videos/.+?/(?P<videoid>[^/]+)\.jhtml' + _FEED_URL = 'http://www.cmt.com/sitewide/apps/player/embed/rss/' + + _TESTS = [ + { + u'url': u'http://www.cmt.com/videos/garth-brooks/989124/the-call-featuring-trisha-yearwood.jhtml#artist=30061', + u'md5': u'e6b7ef3c4c45bbfae88061799bbba6c2', + u'info_dict': { + u'id': u'989124', + u'ext': u'mp4', + u'title': u'Garth Brooks - "The Call (featuring Trisha Yearwood)"', + u'description': u'Blame It All On My Roots', + }, + }, + ] diff --git a/youtube_dl/extractor/cnn.py b/youtube_dl/extractor/cnn.py index a034bb2..c9e7cc5 100644 --- a/youtube_dl/extractor/cnn.py +++ b/youtube_dl/extractor/cnn.py @@ -1,7 +1,12 @@ +from __future__ import unicode_literals + import re from .common import InfoExtractor -from ..utils import determine_ext +from ..utils import ( + int_or_none, + parse_duration, +) class CNNIE(InfoExtractor): @@ -9,12 +14,14 @@ class CNNIE(InfoExtractor): (?P<path>.+?/(?P<title>[^/]+?)(?:\.cnn|(?=&)))''' _TESTS = [{ - u'url': u'http://edition.cnn.com/video/?/video/sports/2013/06/09/nadal-1-on-1.cnn', - u'file': u'sports_2013_06_09_nadal-1-on-1.cnn.mp4', - u'md5': u'3e6121ea48df7e2259fe73a0628605c4', - u'info_dict': { - u'title': u'Nadal wins 8th French Open title', - u'description': u'World Sport\'s Amanda Davies chats with 2013 French Open champion Rafael Nadal.', + 'url': 'http://edition.cnn.com/video/?/video/sports/2013/06/09/nadal-1-on-1.cnn', + 'file': 'sports_2013_06_09_nadal-1-on-1.cnn.mp4', + 'md5': '3e6121ea48df7e2259fe73a0628605c4', + 'info_dict': { + 'title': 'Nadal wins 8th French Open title', + 'description': 'World Sport\'s Amanda Davies chats with 2013 French Open champion Rafael Nadal.', + 'duration': 135, + 'upload_date': '20130609', }, }, { @@ -31,26 +38,62 @@ class CNNIE(InfoExtractor): mobj = re.match(self._VALID_URL, url) path = mobj.group('path') page_title = mobj.group('title') - info_url = u'http://cnn.com/video/data/3.0/%s/index.xml' % path + info_url = 'http://cnn.com/video/data/3.0/%s/index.xml' % path info = self._download_xml(info_url, page_title) formats = [] + rex = re.compile(r'''(?x) + (?P<width>[0-9]+)x(?P<height>[0-9]+) + (?:_(?P<bitrate>[0-9]+)k)? + ''') for f in info.findall('files/file'): - mf = re.match(r'(\d+)x(\d+)(?:_(.*)k)?',f.attrib['bitrate']) - if mf is not None: - formats.append((int(mf.group(1)), int(mf.group(2)), int(mf.group(3) or 0), f.text)) - formats = sorted(formats) - (_,_,_, video_path) = formats[-1] - video_url = 'http://ht.cdn.turner.com/cnn/big%s' % video_path + video_url = 'http://ht.cdn.turner.com/cnn/big%s' % (f.text.strip()) + fdct = { + 'format_id': f.attrib['bitrate'], + 'url': video_url, + } + + mf = rex.match(f.attrib['bitrate']) + if mf: + fdct['width'] = int(mf.group('width')) + fdct['height'] = int(mf.group('height')) + fdct['tbr'] = int_or_none(mf.group('bitrate')) + else: + mf = rex.search(f.text) + if mf: + fdct['width'] = int(mf.group('width')) + fdct['height'] = int(mf.group('height')) + fdct['tbr'] = int_or_none(mf.group('bitrate')) + else: + mi = re.match(r'ios_(audio|[0-9]+)$', f.attrib['bitrate']) + if mi: + if mi.group(1) == 'audio': + fdct['vcodec'] = 'none' + fdct['ext'] = 'm4a' + else: + fdct['tbr'] = int(mi.group(1)) + + formats.append(fdct) + + self._sort_formats(formats) thumbnails = sorted([((int(t.attrib['height']),int(t.attrib['width'])), t.text) for t in info.findall('images/image')]) thumbs_dict = [{'resolution': res, 'url': t_url} for (res, t_url) in thumbnails] - return {'id': info.attrib['id'], - 'title': info.find('headline').text, - 'url': video_url, - 'ext': determine_ext(video_url), - 'thumbnail': thumbnails[-1][1], - 'thumbnails': thumbs_dict, - 'description': info.find('description').text, - } + metas_el = info.find('metas') + upload_date = ( + metas_el.attrib.get('version') if metas_el is not None else None) + + duration_el = info.find('length') + duration = parse_duration(duration_el.text) + + return { + 'id': info.attrib['id'], + 'title': info.find('headline').text, + 'formats': formats, + 'thumbnail': thumbnails[-1][1], + 'thumbnails': thumbs_dict, + 'description': info.find('description').text, + 'duration': duration, + 'upload_date': upload_date, + } diff --git a/youtube_dl/extractor/collegehumor.py b/youtube_dl/extractor/collegehumor.py index b27c1df..d10b7bd 100644 --- a/youtube_dl/extractor/collegehumor.py +++ b/youtube_dl/extractor/collegehumor.py @@ -1,82 +1,68 @@ +from __future__ import unicode_literals + +import json import re from .common import InfoExtractor -from ..utils import ( - compat_urllib_parse_urlparse, - determine_ext, - - ExtractorError, -) class CollegeHumorIE(InfoExtractor): _VALID_URL = r'^(?:https?://)?(?:www\.)?collegehumor\.com/(video|embed|e)/(?P<videoid>[0-9]+)/?(?P<shorttitle>.*)$' _TESTS = [{ - u'url': u'http://www.collegehumor.com/video/6902724/comic-con-cosplay-catastrophe', - u'file': u'6902724.mp4', - u'md5': u'1264c12ad95dca142a9f0bf7968105a0', - u'info_dict': { - u'title': u'Comic-Con Cosplay Catastrophe', - u'description': u'Fans get creative this year at San Diego. Too creative. And yes, that\'s really Joss Whedon.', + 'url': 'http://www.collegehumor.com/video/6902724/comic-con-cosplay-catastrophe', + 'file': '6902724.mp4', + 'md5': 'dcc0f5c1c8be98dc33889a191f4c26bd', + 'info_dict': { + 'title': 'Comic-Con Cosplay Catastrophe', + 'description': 'Fans get creative this year at San Diego. Too', + 'age_limit': 13, }, }, { - u'url': u'http://www.collegehumor.com/video/3505939/font-conference', - u'file': u'3505939.mp4', - u'md5': u'c51ca16b82bb456a4397987791a835f5', - u'info_dict': { - u'title': u'Font Conference', - u'description': u'This video wasn\'t long enough, so we made it double-spaced.', + 'url': 'http://www.collegehumor.com/video/3505939/font-conference', + 'file': '3505939.mp4', + 'md5': '72fa701d8ef38664a4dbb9e2ab721816', + 'info_dict': { + 'title': 'Font Conference', + 'description': 'This video wasn\'t long enough, so we made it double-spaced.', + 'age_limit': 10, }, }] def _real_extract(self, url): mobj = re.match(self._VALID_URL, url) - if mobj is None: - raise ExtractorError(u'Invalid URL: %s' % url) video_id = mobj.group('videoid') - info = { - 'id': video_id, - 'uploader': None, - 'upload_date': None, - } - - self.report_extraction(video_id) - xmlUrl = 'http://www.collegehumor.com/moogaloop/video/' + video_id - mdoc = self._download_xml(xmlUrl, video_id, - u'Downloading info XML', - u'Unable to download video info XML') + jsonUrl = 'http://www.collegehumor.com/moogaloop/video/' + video_id + '.json' + data = json.loads(self._download_webpage( + jsonUrl, video_id, 'Downloading info JSON')) + vdata = data['video'] - try: - videoNode = mdoc.findall('./video')[0] - youtubeIdNode = videoNode.find('./youtubeID') - if youtubeIdNode is not None: - return self.url_result(youtubeIdNode.text, 'Youtube') - info['description'] = videoNode.findall('./description')[0].text - info['title'] = videoNode.findall('./caption')[0].text - info['thumbnail'] = videoNode.findall('./thumbnail')[0].text - next_url = videoNode.findall('./file')[0].text - except IndexError: - raise ExtractorError(u'Invalid metadata XML file') - - if next_url.endswith(u'manifest.f4m'): - manifest_url = next_url + '?hdcore=2.10.3' - adoc = self._download_xml(manifest_url, video_id, - u'Downloading XML manifest', - u'Unable to download video info XML') - - try: - video_id = adoc.findall('./{http://ns.adobe.com/f4m/1.0}id')[0].text - except IndexError: - raise ExtractorError(u'Invalid manifest file') - url_pr = compat_urllib_parse_urlparse(info['thumbnail']) - info['url'] = url_pr.scheme + '://' + url_pr.netloc + video_id[:-2].replace('.csmil','').replace(',','') - info['ext'] = 'mp4' + AGE_LIMITS = {'nc17': 18, 'r': 18, 'pg13': 13, 'pg': 10, 'g': 0} + rating = vdata.get('rating') + if rating: + age_limit = AGE_LIMITS.get(rating.lower()) else: - # Old-style direct links - info['url'] = next_url - info['ext'] = determine_ext(info['url']) + age_limit = None # None = No idea + + PREFS = {'high_quality': 2, 'low_quality': 0} + formats = [] + for format_key in ('mp4', 'webm'): + for qname, qurl in vdata[format_key].items(): + formats.append({ + 'format_id': format_key + '_' + qname, + 'url': qurl, + 'format': format_key, + 'preference': PREFS.get(qname), + }) + self._sort_formats(formats) - return info + return { + 'id': video_id, + 'title': vdata['title'], + 'description': vdata.get('description'), + 'thumbnail': vdata.get('thumbnail'), + 'formats': formats, + 'age_limit': age_limit, + } diff --git a/youtube_dl/extractor/comedycentral.py b/youtube_dl/extractor/comedycentral.py index a54ce3e..27bd825 100644 --- a/youtube_dl/extractor/comedycentral.py +++ b/youtube_dl/extractor/comedycentral.py @@ -12,7 +12,9 @@ from ..utils import ( class ComedyCentralIE(MTVServicesInfoExtractor): - _VALID_URL = r'https?://(?:www.)?comedycentral.com/(video-clips|episodes|cc-studios)/(?P<title>.*)' + _VALID_URL = r'''(?x)https?://(?:www.)?comedycentral.com/ + (video-clips|episodes|cc-studios|video-collections) + /(?P<title>.*)''' _FEED_URL = u'http://comedycentral.com/feeds/mrss/' _TEST = { diff --git a/youtube_dl/extractor/common.py b/youtube_dl/extractor/common.py index ba46a7b..ce3d169 100644 --- a/youtube_dl/extractor/common.py +++ b/youtube_dl/extractor/common.py @@ -1,4 +1,5 @@ import base64 +import json import os import re import socket @@ -9,6 +10,7 @@ import xml.etree.ElementTree from ..utils import ( compat_http_client, compat_urllib_error, + compat_urllib_parse_urlparse, compat_str, clean_html, @@ -37,10 +39,12 @@ class InfoExtractor(object): id: Video identifier. title: Video title, unescaped. - Additionally, it must contain either a formats entry or url and ext: + Additionally, it must contain either a formats entry or a url one: - formats: A list of dictionaries for each format available, it must - be ordered from worst to best quality. Potential fields: + formats: A list of dictionaries for each format available, ordered + from worst to best quality. + + Potential fields: * url Mandatory. The URL of the video file * ext Will be calculated from url if missing * format A human-readable description of the format @@ -48,23 +52,36 @@ class InfoExtractor(object): 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") + ("mp4_h264_opus" or "19"). + Technically optional, but strongly recommended. * 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 + * resolution Textual description of width and height + * tbr Average bitrate of audio and video in KBit/s * 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 * filesize The number of bytes, if known in advance * player_url SWF Player URL (used for rtmpdump). + * protocol The protocol that will be used for the actual + download, lower-case. + "http", "https", "rtsp", "rtmp" or so. + * preference Order number of this format. If this field is + present and not None, the formats get sorted + by this field. + -1 for default (order by other properties), + -2 or smaller for less than default. + * quality Order number of the video quality of this + format, irrespective of the file format. + -1 for default (order by other properties), + -2 or smaller for less than default. url: Final video URL. ext: Video filename extension. format: The video format, defaults to ext (used for --get-format) player_url: SWF Player URL (used for rtmpdump). - urlhandle: [internal] The urlHandle to be used to download the file, - like returned by urllib.request.urlopen The following fields are optional: @@ -244,6 +261,20 @@ class InfoExtractor(object): xml_string = transform_source(xml_string) return xml.etree.ElementTree.fromstring(xml_string.encode('utf-8')) + def _download_json(self, url_or_request, video_id, + note=u'Downloading JSON metadata', + errnote=u'Unable to download JSON metadata'): + json_string = self._download_webpage(url_or_request, video_id, note, errnote) + try: + return json.loads(json_string) + except ValueError as ve: + raise ExtractorError('Failed to download JSON', cause=ve) + + def report_warning(self, msg, video_id=None): + idstr = u'' if video_id is None else u'%s: ' % video_id + self._downloader.report_warning( + u'[%s] %s%s' % (self.IE_NAME, idstr, msg)) + def to_screen(self, msg): """Print msg to screen, prefixing it with '[ie_name]'""" self._downloader.to_screen(u'[%s] %s' % (self.IE_NAME, msg)) @@ -361,7 +392,7 @@ class InfoExtractor(object): @staticmethod def _og_regexes(prop): content_re = r'content=(?:"([^>]+?)"|\'(.+?)\')' - property_re = r'property=[\'"]og:%s[\'"]' % re.escape(prop) + property_re = r'(?:name|property)=[\'"]og:%s[\'"]' % re.escape(prop) template = r'<meta[^>]+?%s[^>]+?%s' return [ template % (property_re, content_re), @@ -426,6 +457,58 @@ class InfoExtractor(object): } return RATING_TABLE.get(rating.lower(), None) + def _sort_formats(self, formats): + def _formats_key(f): + # TODO remove the following workaround + from ..utils import determine_ext + if not f.get('ext') and 'url' in f: + f['ext'] = determine_ext(f['url']) + + preference = f.get('preference') + if preference is None: + proto = f.get('protocol') + if proto is None: + proto = compat_urllib_parse_urlparse(f.get('url', '')).scheme + + preference = 0 if proto in ['http', 'https'] else -0.1 + if f.get('ext') in ['f4f', 'f4m']: # Not yet supported + preference -= 0.5 + + if f.get('vcodec') == 'none': # audio only + if self._downloader.params.get('prefer_free_formats'): + ORDER = [u'aac', u'mp3', u'm4a', u'webm', u'ogg', u'opus'] + else: + ORDER = [u'webm', u'opus', u'ogg', u'mp3', u'aac', u'm4a'] + ext_preference = 0 + try: + audio_ext_preference = ORDER.index(f['ext']) + except ValueError: + audio_ext_preference = -1 + else: + if self._downloader.params.get('prefer_free_formats'): + ORDER = [u'flv', u'mp4', u'webm'] + else: + ORDER = [u'webm', u'flv', u'mp4'] + try: + ext_preference = ORDER.index(f['ext']) + except ValueError: + ext_preference = -1 + audio_ext_preference = 0 + + return ( + preference, + f.get('quality') if f.get('quality') is not None else -1, + f.get('height') if f.get('height') is not None else -1, + f.get('width') if f.get('width') is not None else -1, + ext_preference, + f.get('tbr') if f.get('tbr') is not None else -1, + f.get('vbr') if f.get('vbr') is not None else -1, + f.get('abr') if f.get('abr') is not None else -1, + audio_ext_preference, + f.get('filesize') if f.get('filesize') is not None else -1, + f.get('format_id'), + ) + formats.sort(key=_formats_key) class SearchInfoExtractor(InfoExtractor): diff --git a/youtube_dl/extractor/condenast.py b/youtube_dl/extractor/condenast.py index f336a3c..03b75b8 100644 --- a/youtube_dl/extractor/condenast.py +++ b/youtube_dl/extractor/condenast.py @@ -1,4 +1,5 @@ # coding: utf-8 +from __future__ import unicode_literals import re import json @@ -20,30 +21,31 @@ class CondeNastIE(InfoExtractor): # The keys are the supported sites and the values are the name to be shown # to the user and in the extractor description. - _SITES = {'wired': u'WIRED', - 'gq': u'GQ', - 'vogue': u'Vogue', - 'glamour': u'Glamour', - 'wmagazine': u'W Magazine', - 'vanityfair': u'Vanity Fair', - } + _SITES = { + 'wired': 'WIRED', + 'gq': 'GQ', + 'vogue': 'Vogue', + 'glamour': 'Glamour', + 'wmagazine': 'W Magazine', + 'vanityfair': 'Vanity Fair', + } _VALID_URL = r'http://(video|www).(?P<site>%s).com/(?P<type>watch|series|video)/(?P<id>.+)' % '|'.join(_SITES.keys()) - IE_DESC = u'Condé Nast media group: %s' % ', '.join(sorted(_SITES.values())) + IE_DESC = 'Condé Nast media group: %s' % ', '.join(sorted(_SITES.values())) _TEST = { - u'url': u'http://video.wired.com/watch/3d-printed-speakers-lit-with-led', - u'file': u'5171b343c2b4c00dd0c1ccb3.mp4', - u'md5': u'1921f713ed48aabd715691f774c451f7', - u'info_dict': { - u'title': u'3D Printed Speakers Lit With LED', - u'description': u'Check out these beautiful 3D printed LED speakers. You can\'t actually buy them, but LumiGeek is working on a board that will let you make you\'re own.', + 'url': 'http://video.wired.com/watch/3d-printed-speakers-lit-with-led', + 'file': '5171b343c2b4c00dd0c1ccb3.mp4', + 'md5': '1921f713ed48aabd715691f774c451f7', + 'info_dict': { + 'title': '3D Printed Speakers Lit With LED', + 'description': 'Check out these beautiful 3D printed LED speakers. You can\'t actually buy them, but LumiGeek is working on a board that will let you make you\'re own.', } } def _extract_series(self, url, webpage): title = self._html_search_regex(r'<div class="cne-series-info">.*?<h1>(.+?)</h1>', - webpage, u'series title', flags=re.DOTALL) + webpage, 'series title', flags=re.DOTALL) url_object = compat_urllib_parse_urlparse(url) base_url = '%s://%s' % (url_object.scheme, url_object.netloc) m_paths = re.finditer(r'<p class="cne-thumb-title">.*?<a href="(/watch/.+?)["\?]', @@ -57,39 +59,41 @@ class CondeNastIE(InfoExtractor): description = self._html_search_regex([r'<div class="cne-video-description">(.+?)</div>', r'<div class="video-post-content">(.+?)</div>', ], - webpage, u'description', + webpage, 'description', fatal=False, flags=re.DOTALL) params = self._search_regex(r'var params = {(.+?)}[;,]', webpage, - u'player params', flags=re.DOTALL) - video_id = self._search_regex(r'videoId: [\'"](.+?)[\'"]', params, u'video id') - player_id = self._search_regex(r'playerId: [\'"](.+?)[\'"]', params, u'player id') - target = self._search_regex(r'target: [\'"](.+?)[\'"]', params, u'target') + 'player params', flags=re.DOTALL) + video_id = self._search_regex(r'videoId: [\'"](.+?)[\'"]', params, 'video id') + player_id = self._search_regex(r'playerId: [\'"](.+?)[\'"]', params, 'player id') + target = self._search_regex(r'target: [\'"](.+?)[\'"]', params, 'target') data = compat_urllib_parse.urlencode({'videoId': video_id, 'playerId': player_id, 'target': target, }) base_info_url = self._search_regex(r'url = [\'"](.+?)[\'"][,;]', - webpage, u'base info url', + webpage, 'base info url', default='http://player.cnevids.com/player/loader.js?') info_url = base_info_url + data info_page = self._download_webpage(info_url, video_id, - u'Downloading video info') - video_info = self._search_regex(r'var video = ({.+?});', info_page, u'video info') + 'Downloading video info') + video_info = self._search_regex(r'var video = ({.+?});', info_page, 'video info') video_info = json.loads(video_info) - def _formats_sort_key(f): - type_ord = 1 if f['type'] == 'video/mp4' else 0 - quality_ord = 1 if f['quality'] == 'high' else 0 - return (quality_ord, type_ord) - best_format = sorted(video_info['sources'][0], key=_formats_sort_key)[-1] + formats = [{ + 'format_id': '%s-%s' % (fdata['type'].split('/')[-1], fdata['quality']), + 'url': fdata['src'], + 'ext': fdata['type'].split('/')[-1], + 'quality': 1 if fdata['quality'] == 'high' else 0, + } for fdata in video_info['sources'][0]] + self._sort_formats(formats) - return {'id': video_id, - 'url': best_format['src'], - 'ext': best_format['type'].split('/')[-1], - 'title': video_info['title'], - 'thumbnail': video_info['poster_frame'], - 'description': description, - } + return { + 'id': video_id, + 'formats': formats, + 'title': video_info['title'], + 'thumbnail': video_info['poster_frame'], + 'description': description, + } def _real_extract(self, url): mobj = re.match(self._VALID_URL, url) diff --git a/youtube_dl/extractor/cspan.py b/youtube_dl/extractor/cspan.py index d573068..a2cbd4d 100644 --- a/youtube_dl/extractor/cspan.py +++ b/youtube_dl/extractor/cspan.py @@ -1,20 +1,25 @@ +from __future__ import unicode_literals + +import json import re from .common import InfoExtractor from ..utils import ( - compat_urllib_parse, + unescapeHTML, ) + class CSpanIE(InfoExtractor): _VALID_URL = r'http://www\.c-spanvideo\.org/program/(.*)' + IE_DESC = 'C-SPAN' _TEST = { - u'url': u'http://www.c-spanvideo.org/program/HolderonV', - u'file': u'315139.flv', - u'md5': u'74a623266956f69e4df0068ab6c80fe4', - u'info_dict': { - u"title": u"Attorney General Eric Holder on Voting Rights Act Decision" + 'url': 'http://www.c-spanvideo.org/program/HolderonV', + 'file': '315139.mp4', + 'md5': '8e44ce11f0f725527daccc453f553eb0', + 'info_dict': { + 'title': 'Attorney General Eric Holder on Voting Rights Act Decision', + 'description': 'Attorney General Eric Holder spoke to reporters following the Supreme Court decision in [Shelby County v. Holder] in which the court ruled that the preclearance provisions of the Voting Rights Act could not be enforced until Congress established new guidelines for review.', }, - u'skip': u'Requires rtmpdump' } def _real_extract(self, url): @@ -22,30 +27,22 @@ class CSpanIE(InfoExtractor): prog_name = mobj.group(1) webpage = self._download_webpage(url, prog_name) video_id = self._search_regex(r'programid=(.*?)&', webpage, 'video id') - data = compat_urllib_parse.urlencode({'programid': video_id, - 'dynamic':'1'}) - info_url = 'http://www.c-spanvideo.org/common/services/flashXml.php?' + data - video_info = self._download_webpage(info_url, video_id, u'Downloading video info') - - self.report_extraction(video_id) - - title = self._html_search_regex(r'<string name="title">(.*?)</string>', - video_info, 'title') - description = self._html_search_regex(r'<meta (?:property="og:|name=")description" content="(.*?)"', - webpage, 'description', - flags=re.MULTILINE|re.DOTALL) - - url = self._search_regex(r'<string name="URL">(.*?)</string>', - video_info, 'video url') - url = url.replace('$(protocol)', 'rtmp').replace('$(port)', '443') - path = self._search_regex(r'<string name="path">(.*?)</string>', - video_info, 'rtmp play path') - - return {'id': video_id, - 'title': title, - 'ext': 'flv', - 'url': url, - 'play_path': path, - 'description': description, - 'thumbnail': self._og_search_thumbnail(webpage), - } + + title = self._html_search_regex( + r'<!-- title -->\n\s*<h1[^>]*>(.*?)</h1>', webpage, 'title') + description = self._og_search_description(webpage) + + info_url = 'http://c-spanvideo.org/videoLibrary/assets/player/ajax-player.php?os=android&html5=program&id=' + video_id + data_json = self._download_webpage( + info_url, video_id, 'Downloading video info') + data = json.loads(data_json) + + url = unescapeHTML(data['video']['files'][0]['path']['#text']) + + return { + 'id': video_id, + 'title': title, + 'url': url, + 'description': description, + 'thumbnail': self._og_search_thumbnail(webpage), + } diff --git a/youtube_dl/extractor/defense.py b/youtube_dl/extractor/defense.py index 424d960..c5529f8 100644 --- a/youtube_dl/extractor/defense.py +++ b/youtube_dl/extractor/defense.py @@ -1,3 +1,5 @@ +from __future__ import unicode_literals + import re import json @@ -5,15 +7,14 @@ from .common import InfoExtractor class DefenseGouvFrIE(InfoExtractor): - _IE_NAME = 'defense.gouv.fr' + IE_NAME = 'defense.gouv.fr' _VALID_URL = (r'http://.*?\.defense\.gouv\.fr/layout/set/' r'ligthboxvideo/base-de-medias/webtv/(.*)') _TEST = { - u'url': (u'http://www.defense.gouv.fr/layout/set/ligthboxvideo/' - u'base-de-medias/webtv/attaque-chimique-syrienne-du-21-aout-2013-1'), - u'file': u'11213.mp4', - u'md5': u'75bba6124da7e63d2d60b5244ec9430c', + 'url': 'http://www.defense.gouv.fr/layout/set/ligthboxvideo/base-de-medias/webtv/attaque-chimique-syrienne-du-21-aout-2013-1', + 'file': '11213.mp4', + 'md5': '75bba6124da7e63d2d60b5244ec9430c', "info_dict": { "title": "attaque-chimique-syrienne-du-21-aout-2013-1" } diff --git a/youtube_dl/extractor/dreisat.py b/youtube_dl/extractor/dreisat.py index cb7226f..0b11d1f 100644 --- a/youtube_dl/extractor/dreisat.py +++ b/youtube_dl/extractor/dreisat.py @@ -4,18 +4,17 @@ import re from .common import InfoExtractor from ..utils import ( - determine_ext, unified_strdate, ) class DreiSatIE(InfoExtractor): IE_NAME = '3sat' - _VALID_URL = r'(?:http://)?(?:www\.)?3sat\.de/mediathek/index\.php\?(?:(?:mode|display)=[^&]+&)*obj=(?P<id>[0-9]+)$' + _VALID_URL = r'(?:http://)?(?:www\.)?3sat\.de/mediathek/(?:index\.php)?\?(?:(?:mode|display)=[^&]+&)*obj=(?P<id>[0-9]+)$' _TEST = { u"url": u"http://www.3sat.de/mediathek/index.php?obj=36983", - u'file': u'36983.webm', - u'md5': u'57c97d0469d71cf874f6815aa2b7c944', + u'file': u'36983.mp4', + u'md5': u'9dcfe344732808dbfcc901537973c922', u'info_dict': { u"title": u"Kaffeeland Schweiz", u"description": u"Über 80 Kaffeeröstereien liefern in der Schweiz das Getränk, in das das Land so vernarrt ist: Mehr als 1000 Tassen trinkt ein Schweizer pro Jahr. SCHWEIZWEIT nimmt die Kaffeekultur unter die...", @@ -52,18 +51,12 @@ class DreiSatIE(InfoExtractor): 'width': int(fe.find('./width').text), 'height': int(fe.find('./height').text), 'url': fe.find('./url').text, - 'ext': determine_ext(fe.find('./url').text), 'filesize': int(fe.find('./filesize').text), 'video_bitrate': int(fe.find('./videoBitrate').text), - '3sat_qualityname': fe.find('./quality').text, } for fe in format_els if not fe.find('./url').text.startswith('http://www.metafilegenerator.de/')] - def _sortkey(format): - qidx = ['low', 'med', 'high', 'veryhigh'].index(format['3sat_qualityname']) - prefer_http = 1 if 'rtmp' in format['url'] else 0 - return (qidx, prefer_http, format['video_bitrate']) - formats.sort(key=_sortkey) + self._sort_formats(formats) return { '_type': 'video', diff --git a/youtube_dl/extractor/everyonesmixtape.py b/youtube_dl/extractor/everyonesmixtape.py new file mode 100644 index 0000000..12829cb --- /dev/null +++ b/youtube_dl/extractor/everyonesmixtape.py @@ -0,0 +1,69 @@ +from __future__ import unicode_literals + +import re + +from .common import InfoExtractor +from ..utils import ( + compat_urllib_request, + ExtractorError, +) + + +class EveryonesMixtapeIE(InfoExtractor): + _VALID_URL = r'https?://(?:www\.)?everyonesmixtape\.com/#/mix/(?P<id>[0-9a-zA-Z]+)(?:/(?P<songnr>[0-9]))?$' + + _TEST = { + 'url': 'http://everyonesmixtape.com/#/mix/m7m0jJAbMQi/5', + 'file': '5bfseWNmlds.mp4', + "info_dict": { + "title": "Passion Pit - \"Sleepyhead\" (Official Music Video)", + "uploader": "FKR.TV", + "uploader_id": "frenchkissrecords", + "description": "Music video for \"Sleepyhead\" from Passion Pit's debut EP Chunk Of Change.\nBuy on iTunes: https://itunes.apple.com/us/album/chunk-of-change-ep/id300087641\n\nDirected by The Wilderness.\n\nhttp://www.passionpitmusic.com\nhttp://www.frenchkissrecords.com", + "upload_date": "20081015" + }, + 'params': { + 'skip_download': True, # This is simply YouTube + } + } + + def _real_extract(self, url): + mobj = re.match(self._VALID_URL, url) + playlist_id = mobj.group('id') + + pllist_url = 'http://everyonesmixtape.com/mixtape.php?a=getMixes&u=-1&linked=%s&explore=' % playlist_id + pllist_req = compat_urllib_request.Request(pllist_url) + pllist_req.add_header('X-Requested-With', 'XMLHttpRequest') + + playlist_list = self._download_json( + pllist_req, playlist_id, note='Downloading playlist metadata') + try: + playlist_no = next(playlist['id'] + for playlist in playlist_list + if playlist['code'] == playlist_id) + except StopIteration: + raise ExtractorError('Playlist id not found') + + pl_url = 'http://everyonesmixtape.com/mixtape.php?a=getMix&id=%s&userId=null&code=' % playlist_no + pl_req = compat_urllib_request.Request(pl_url) + pl_req.add_header('X-Requested-With', 'XMLHttpRequest') + playlist = self._download_json( + pl_req, playlist_id, note='Downloading playlist info') + + entries = [{ + '_type': 'url', + 'url': t['url'], + 'title': t['title'], + } for t in playlist['tracks']] + + if mobj.group('songnr'): + songnr = int(mobj.group('songnr')) - 1 + return entries[songnr] + + playlist_title = playlist['mixData']['name'] + return { + '_type': 'playlist', + 'id': playlist_id, + 'title': playlist_title, + 'entries': entries, + } diff --git a/youtube_dl/extractor/flickr.py b/youtube_dl/extractor/flickr.py index e1d2f05..21ea5ec 100644 --- a/youtube_dl/extractor/flickr.py +++ b/youtube_dl/extractor/flickr.py @@ -1,3 +1,5 @@ +from __future__ import unicode_literals + import re from .common import InfoExtractor @@ -11,13 +13,13 @@ class FlickrIE(InfoExtractor): """Information Extractor for Flickr videos""" _VALID_URL = r'(?:https?://)?(?:www\.|secure\.)?flickr\.com/photos/(?P<uploader_id>[\w\-_@]+)/(?P<id>\d+).*' _TEST = { - u'url': u'http://www.flickr.com/photos/forestwander-nature-pictures/5645318632/in/photostream/', - u'file': u'5645318632.mp4', - u'md5': u'6fdc01adbc89d72fc9c4f15b4a4ba87b', - u'info_dict': { - u"description": u"Waterfalls in the Springtime at Dark Hollow Waterfalls. These are located just off of Skyline Drive in Virginia. They are only about 6/10 of a mile hike but it is a pretty steep hill and a good climb back up.", - u"uploader_id": u"forestwander-nature-pictures", - u"title": u"Dark Hollow Waterfalls" + 'url': 'http://www.flickr.com/photos/forestwander-nature-pictures/5645318632/in/photostream/', + 'file': '5645318632.mp4', + 'md5': '6fdc01adbc89d72fc9c4f15b4a4ba87b', + 'info_dict': { + "description": "Waterfalls in the Springtime at Dark Hollow Waterfalls. These are located just off of Skyline Drive in Virginia. They are only about 6/10 of a mile hike but it is a pretty steep hill and a good climb back up.", + "uploader_id": "forestwander-nature-pictures", + "title": "Dark Hollow Waterfalls" } } @@ -29,13 +31,13 @@ class FlickrIE(InfoExtractor): webpage_url = 'http://www.flickr.com/photos/' + video_uploader_id + '/' + video_id webpage = self._download_webpage(webpage_url, video_id) - secret = self._search_regex(r"photo_secret: '(\w+)'", webpage, u'secret') + secret = self._search_regex(r"photo_secret: '(\w+)'", webpage, 'secret') first_url = 'https://secure.flickr.com/apps/video/video_mtl_xml.gne?v=x&photo_id=' + video_id + '&secret=' + secret + '&bitrate=700&target=_self' first_xml = self._download_webpage(first_url, video_id, 'Downloading first data webpage') node_id = self._html_search_regex(r'<Item id="id">(\d+-\d+)</Item>', - first_xml, u'node_id') + first_xml, 'node_id') second_url = 'https://secure.flickr.com/video_playlist.gne?node_id=' + node_id + '&tech=flash&mode=playlist&bitrate=700&secret=' + secret + '&rd=video.yahoo.com&noad=1' second_xml = self._download_webpage(second_url, video_id, 'Downloading second data webpage') @@ -44,7 +46,7 @@ class FlickrIE(InfoExtractor): mobj = re.search(r'<STREAM APP="(.+?)" FULLPATH="(.+?)"', second_xml) if mobj is None: - raise ExtractorError(u'Unable to extract video url') + raise ExtractorError('Unable to extract video url') video_url = mobj.group(1) + unescapeHTML(mobj.group(2)) return [{ diff --git a/youtube_dl/extractor/franceinter.py b/youtube_dl/extractor/franceinter.py new file mode 100644 index 0000000..deb1b0b --- /dev/null +++ b/youtube_dl/extractor/franceinter.py @@ -0,0 +1,38 @@ +# coding: utf-8 +from __future__ import unicode_literals + +import re + +from .common import InfoExtractor + + +class FranceInterIE(InfoExtractor): + _VALID_URL = r'http://(?:www\.)?franceinter\.fr/player/reecouter\?play=(?P<id>[0-9]{6})' + _TEST = { + 'url': 'http://www.franceinter.fr/player/reecouter?play=793962', + 'file': '793962.mp3', + 'md5': '4764932e466e6f6c79c317d2e74f6884', + "info_dict": { + "title": "L’Histoire dans les jeux vidéo", + }, + } + + def _real_extract(self, url): + mobj = re.match(self._VALID_URL, url) + video_id = mobj.group('id') + + webpage = self._download_webpage(url, video_id) + title = self._html_search_regex( + r'<span class="roll_overflow">(.*?)</span></h1>', webpage, 'title') + path = self._search_regex( + r'&urlAOD=(.*?)&startTime', webpage, 'video url') + video_url = 'http://www.franceinter.fr/' + path + + return { + 'id': video_id, + 'formats': [{ + 'url': video_url, + 'vcodec': 'none', + }], + 'title': title, + } diff --git a/youtube_dl/extractor/francetv.py b/youtube_dl/extractor/francetv.py index ad85bc1..b32ff9f 100644 --- a/youtube_dl/extractor/francetv.py +++ b/youtube_dl/extractor/francetv.py @@ -191,3 +191,29 @@ class GenerationQuoiIE(InfoExtractor): info = json.loads(info_json) return self.url_result('http://www.dailymotion.com/video/%s' % info['id'], ie='Dailymotion') + + +class CultureboxIE(FranceTVBaseInfoExtractor): + IE_NAME = u'culturebox.francetvinfo.fr' + _VALID_URL = r'https?://culturebox\.francetvinfo\.fr/(?P<name>.*?)(\?|$)' + + _TEST = { + u'url': u'http://culturebox.francetvinfo.fr/einstein-on-the-beach-au-theatre-du-chatelet-146813', + u'info_dict': { + u'id': u'EV_6785', + u'ext': u'mp4', + u'title': u'Einstein on the beach au Théâtre du Châtelet', + u'description': u'md5:9ce2888b1efefc617b5e58b3f6200eeb', + }, + u'params': { + # m3u8 download + u'skip_download': True, + }, + } + + def _real_extract(self, url): + mobj = re.match(self._VALID_URL, url) + name = mobj.group('name') + webpage = self._download_webpage(url, name) + video_id = self._search_regex(r'"http://videos\.francetv\.fr/video/(.*?)"', webpage, u'video id') + return self._extract_video(video_id) diff --git a/youtube_dl/extractor/gamespot.py b/youtube_dl/extractor/gamespot.py index 26b7d2a..380ebbe 100644 --- a/youtube_dl/extractor/gamespot.py +++ b/youtube_dl/extractor/gamespot.py @@ -1,3 +1,5 @@ +from __future__ import unicode_literals + import re import json @@ -13,12 +15,12 @@ from ..utils import ( class GameSpotIE(InfoExtractor): _VALID_URL = r'(?:http://)?(?:www\.)?gamespot\.com/.*-(?P<page_id>\d+)/?' _TEST = { - u"url": u"http://www.gamespot.com/arma-iii/videos/arma-iii-community-guide-sitrep-i-6410818/", - u"file": u"gs-2300-6410818.mp4", - u"md5": u"b2a30deaa8654fcccd43713a6b6a4825", - u"info_dict": { - u"title": u"Arma 3 - Community Guide: SITREP I", - u'description': u'Check out this video where some of the basics of Arma 3 is explained.', + "url": "http://www.gamespot.com/arma-iii/videos/arma-iii-community-guide-sitrep-i-6410818/", + "file": "gs-2300-6410818.mp4", + "md5": "b2a30deaa8654fcccd43713a6b6a4825", + "info_dict": { + "title": "Arma 3 - Community Guide: SITREP I", + 'description': 'Check out this video where some of the basics of Arma 3 is explained.', } } diff --git a/youtube_dl/extractor/generic.py b/youtube_dl/extractor/generic.py index 7a14c98..8395309 100644 --- a/youtube_dl/extractor/generic.py +++ b/youtube_dl/extractor/generic.py @@ -1,9 +1,12 @@ # encoding: utf-8 +from __future__ import unicode_literals + import os import re from .common import InfoExtractor +from .youtube import YoutubeIE from ..utils import ( compat_urllib_error, compat_urllib_parse, @@ -22,78 +25,78 @@ from .ooyala import OoyalaIE class GenericIE(InfoExtractor): - IE_DESC = u'Generic downloader that works on some sites' + IE_DESC = 'Generic downloader that works on some sites' _VALID_URL = r'.*' - IE_NAME = u'generic' + IE_NAME = 'generic' _TESTS = [ { - u'url': u'http://www.hodiho.fr/2013/02/regis-plante-sa-jeep.html', - u'file': u'13601338388002.mp4', - u'md5': u'6e15c93721d7ec9e9ca3fdbf07982cfd', - u'info_dict': { - u"uploader": u"www.hodiho.fr", - u"title": u"R\u00e9gis plante sa Jeep" + 'url': 'http://www.hodiho.fr/2013/02/regis-plante-sa-jeep.html', + 'file': '13601338388002.mp4', + 'md5': '6e15c93721d7ec9e9ca3fdbf07982cfd', + 'info_dict': { + 'uploader': 'www.hodiho.fr', + 'title': 'R\u00e9gis plante sa Jeep', } }, # 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', - u'info_dict': { - u'title': u'ACCU 2011: Move Semantics,Perfect Forwarding, and Rvalue references- Scott Meyers- 13/04/2011', - u"uploader_id": u"skillsmatter", - u"uploader": u"Skills Matter", + 'add_ie': ['Vimeo'], + 'url': 'http://skillsmatter.com/podcast/home/move-semanticsperfect-forwarding-and-rvalue-references', + 'file': '22444065.mp4', + 'md5': '2903896e23df39722c33f015af0666e2', + 'info_dict': { + 'title': 'ACCU 2011: Move Semantics,Perfect Forwarding, and Rvalue references- Scott Meyers- 13/04/2011', + 'uploader_id': 'skillsmatter', + 'uploader': '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', + 'add_ie': ['Bandcamp'], + 'url': 'http://bronyrock.com/track/the-pony-mash', + 'file': '3235767654.mp3', + 'info_dict': { + 'title': 'The Pony Mash', + 'uploader': 'M_Pallante', }, - u'skip': u'There is a limit of 200 free downloads / month for the test song', + 'skip': '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', + 'add_ie': ['Brightcove'], + 'url': 'http://www.bfmtv.com/video/bfmbusiness/cours-bourse/cours-bourse-l-analyse-technique-154522/', + 'info_dict': { + 'id': '2765128793001', + 'ext': 'mp4', + 'title': 'Le cours de bourse : l’analyse technique', + 'description': 'md5:7e9ad046e968cb2d1114004aba466fd9', + 'uploader': 'BFM BUSINESS', }, - u'params': { - u'skip_download': True, + 'params': { + 'skip_download': True, }, }, # Direct link to a video { - u'url': u'http://media.w3.org/2010/05/sintel/trailer.mp4', - u'file': u'trailer.mp4', - u'md5': u'67d406c2bcb6af27fa886f31aa934bbe', - u'info_dict': { - u'id': u'trailer', - u'title': u'trailer', - u'upload_date': u'20100513', + 'url': 'http://media.w3.org/2010/05/sintel/trailer.mp4', + 'file': 'trailer.mp4', + 'md5': '67d406c2bcb6af27fa886f31aa934bbe', + 'info_dict': { + 'id': 'trailer', + 'title': 'trailer', + 'upload_date': '20100513', } }, # ooyala video { - u'url': u'http://www.rollingstone.com/music/videos/norwegian-dj-cashmere-cat-goes-spartan-on-with-me-premiere-20131219', - u'md5': u'5644c6ca5d5782c1d0d350dad9bd840c', - u'info_dict': { - u'id': u'BwY2RxaTrTkslxOfcan0UCf0YqyvWysJ', - u'ext': u'mp4', - u'title': u'2cc213299525360.mov', #that's what we get + 'url': 'http://www.rollingstone.com/music/videos/norwegian-dj-cashmere-cat-goes-spartan-on-with-me-premiere-20131219', + 'md5': '5644c6ca5d5782c1d0d350dad9bd840c', + 'info_dict': { + 'id': 'BwY2RxaTrTkslxOfcan0UCf0YqyvWysJ', + 'ext': 'mp4', + 'title': '2cc213299525360.mov', #that's what we get }, }, ] @@ -101,12 +104,12 @@ class GenericIE(InfoExtractor): def report_download_webpage(self, video_id): """Report webpage download.""" if not self._downloader.params.get('test', False): - self._downloader.report_warning(u'Falling back on generic information extractor.') + self._downloader.report_warning('Falling back on generic information extractor.') super(GenericIE, self).report_download_webpage(video_id) def report_following_redirect(self, new_url): """Report information extraction.""" - self._downloader.to_screen(u'[redirect] Following redirect to %s' % new_url) + self._downloader.to_screen('[redirect] Following redirect to %s' % new_url) def _send_head(self, url): """Check if it is a redirect, like url shorteners, in case return the new url.""" @@ -152,7 +155,7 @@ class GenericIE(InfoExtractor): response = opener.open(HEADRequest(url)) if response is None: - raise ExtractorError(u'Invalid URL protocol') + raise ExtractorError('Invalid URL protocol') return response def _real_extract(self, url): @@ -162,6 +165,8 @@ class GenericIE(InfoExtractor): return self.url_result('http://' + url) video_id = os.path.splitext(url.split('/')[-1])[0] + self.to_screen('%s: Requesting header' % video_id) + try: response = self._send_head(url) @@ -184,7 +189,7 @@ class GenericIE(InfoExtractor): 'formats': [{ 'format_id': m.group('format_id'), 'url': url, - 'vcodec': u'none' if m.group('type') == 'audio' else None + 'vcodec': 'none' if m.group('type') == 'audio' else None }], 'upload_date': upload_date, } @@ -198,7 +203,7 @@ class GenericIE(InfoExtractor): except ValueError: # since this is the last-resort InfoExtractor, if # this error is thrown, it'll be thrown here - raise ExtractorError(u'Failed to download URL: %s' % url) + raise ExtractorError('Failed to download URL: %s' % url) self.report_extraction(video_id) @@ -209,22 +214,23 @@ class GenericIE(InfoExtractor): # Video Title - Tagline | Site Name # and so on and so forth; it's just not practical video_title = self._html_search_regex( - r'(?s)<title>(.*?)', webpage, u'video title', - default=u'video') + r'(?s)(.*?)', webpage, 'video title', + default='video') # video uploader is domain name video_uploader = self._search_regex( - r'^(?:https?://)?([^/]*)/.*', url, u'video uploader') + r'^(?:https?://)?([^/]*)/.*', url, 'video uploader') # Look for BrightCove: bc_url = BrightcoveIE._extract_brightcove_url(webpage) if bc_url is not None: - self.to_screen(u'Brightcove video detected.') - return self.url_result(bc_url, 'Brightcove') + self.to_screen('Brightcove video detected.') + surl = smuggle_url(bc_url, {'Referer': url}) + return self.url_result(surl, 'Brightcove') # Look for embedded (iframe) Vimeo player mobj = re.search( - r']+?src="(https?://player.vimeo.com/video/.+?)"', webpage) + r']+?src="((?:https?:)?//player.vimeo.com/video/.+?)"', webpage) if mobj: player_url = unescapeHTML(mobj.group(1)) surl = smuggle_url(player_url, {'Referer': url}) @@ -271,16 +277,12 @@ class GenericIE(InfoExtractor): } # Look for embedded blip.tv player - mobj = re.search(r']*https?://api.blip.tv/\w+/redirect/\w+/(\d+)', webpage) + mobj = re.search(r']*https?://api\.blip\.tv/\w+/redirect/\w+/(\d+)', webpage) if mobj: - return self.url_result('http://blip.tv/seo/-'+mobj.group(1), 'BlipTV') - mobj = re.search(r'<(?:iframe|embed|object)\s[^>]*https?://(?:\w+\.)?blip.tv/(?:play/|api\.swf#)([a-zA-Z0-9]+)', webpage) + return self.url_result('http://blip.tv/a/a-'+mobj.group(1), 'BlipTV') + mobj = re.search(r'<(?:iframe|embed|object)\s[^>]*(https?://(?:\w+\.)?blip\.tv/(?:play/|api\.swf#)[a-zA-Z0-9]+)', webpage) if mobj: - player_url = 'http://blip.tv/play/%s.x?p=1' % mobj.group(1) - player_page = self._download_webpage(player_url, mobj.group(1)) - blip_video_id = self._search_regex(r'data-episode-id="(\d+)', player_page, u'blip_video_id', fatal=False) - if blip_video_id: - return self.url_result('http://blip.tv/seo/-'+blip_video_id, 'BlipTV') + return self.url_result(mobj.group(1), 'BlipTV') # Look for Bandcamp pages with custom domain mobj = re.search(r']*?content="(.*?bandcamp\.com.*?)"', webpage) @@ -301,18 +303,32 @@ class GenericIE(InfoExtractor): return OoyalaIE._build_url_result(mobj.group(1)) # Look for Aparat videos - mobj = re.search(r'