| 
									
										
										
										
											2015-02-17 21:37:48 +01:00
										 |  |  | from __future__ import division, unicode_literals | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | import os | 
					
						
							|  |  |  | import re | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | import time | 
					
						
							| 
									
										
										
										
											2016-08-04 15:47:22 +05:30
										 |  |  | import random | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-03 19:24:24 +08:00
										 |  |  | from ..compat import compat_os_name | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | from ..utils import ( | 
					
						
							| 
									
										
										
										
											2017-06-17 23:50:21 +07:00
										 |  |  |     decodeArgument, | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |     encodeFilename, | 
					
						
							| 
									
										
										
										
											2015-12-20 07:00:39 +06:00
										 |  |  |     error_to_compat_str, | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |     format_bytes, | 
					
						
							| 
									
										
										
										
											2017-06-17 23:50:21 +07:00
										 |  |  |     shell_quote, | 
					
						
							| 
									
										
										
										
											2014-04-04 14:59:11 +02:00
										 |  |  |     timeconvert, | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-25 04:49:44 +01:00
										 |  |  |     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 | 
					
						
							|  |  |  |     xattr_set_filesize: Set ytdl.filesize user xattribute with expected size. | 
					
						
							| 
									
										
										
										
											2015-03-02 15:06:09 +01:00
										 |  |  |     external_downloader_args:  A list of additional command-line arguments for the | 
					
						
							|  |  |  |                         external downloader. | 
					
						
							| 
									
										
										
										
											2016-01-30 12:26:40 +01:00
										 |  |  |     hls_use_mpegts:     Use the mpegts container for HLS videos. | 
					
						
							| 
									
										
										
										
											2018-02-17 19:10:12 +07:00
										 |  |  |     http_chunk_size:    Size of a chunk for chunk-based HTTP downloading. May be | 
					
						
							| 
									
										
										
										
											2018-02-04 02:53:50 +07:00
										 |  |  |                         useful for bypassing bandwidth throttling imposed by | 
					
						
							|  |  |  |                         a webserver (experimental) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Subclasses of this one must re-define the real_download method. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-24 14:38:40 +02:00
										 |  |  |     _TEST_FILE_SIZE = 10241 | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |     params = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, ydl, params): | 
					
						
							|  |  |  |         """Create a FileDownloader object with the given options.""" | 
					
						
							|  |  |  |         self.ydl = ydl | 
					
						
							|  |  |  |         self._progress_hooks = [] | 
					
						
							|  |  |  |         self.params = params | 
					
						
							| 
									
										
										
										
											2015-02-17 21:37:48 +01:00
										 |  |  |         self.add_progress_hook(self.report_progress) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @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 | 
					
						
							| 
									
										
										
										
											2014-07-31 03:08:24 +02:00
										 |  |  |         if now is None: | 
					
						
							|  |  |  |             now = time.time() | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         dif = now - start | 
					
						
							| 
									
										
										
										
											2014-11-23 20:41:03 +01:00
										 |  |  |         if current == 0 or dif < 0.001:  # One millisecond | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |             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 | 
					
						
							| 
									
										
										
										
											2014-11-23 20:41:03 +01:00
										 |  |  |         if bytes == 0 or dif < 0.001:  # One millisecond | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |             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)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-19 20:51:30 +06:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def format_retries(retries): | 
					
						
							|  |  |  |         return 'inf' if retries == float('inf') else '%.0f' % retries | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def best_block_size(elapsed_time, bytes): | 
					
						
							|  |  |  |         new_min = max(bytes / 2.0, 1.0) | 
					
						
							| 
									
										
										
										
											2014-11-23 20:41:03 +01:00
										 |  |  |         new_max = min(max(bytes * 2.0, 1.0), 4194304)  # Do not surpass 4 MB | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         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) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-31 03:08:24 +02:00
										 |  |  |     def slow_down(self, start_time, now, byte_counter): | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         """Sleep if the download speed is over the rate limit.""" | 
					
						
							| 
									
										
										
										
											2016-02-14 14:25:04 +06:00
										 |  |  |         rate_limit = self.params.get('ratelimit') | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         if rate_limit is None or byte_counter == 0: | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2014-07-31 03:08:24 +02:00
										 |  |  |         if now is None: | 
					
						
							|  |  |  |             now = time.time() | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         elapsed = now - start_time | 
					
						
							|  |  |  |         if elapsed <= 0.0: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         speed = float(byte_counter) / elapsed | 
					
						
							|  |  |  |         if speed > rate_limit: | 
					
						
							| 
									
										
										
										
											2019-06-05 03:06:35 +07:00
										 |  |  |             sleep_time = float(byte_counter) / rate_limit - elapsed | 
					
						
							|  |  |  |             if sleep_time > 0: | 
					
						
							|  |  |  |                 time.sleep(sleep_time) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def temp_name(self, filename): | 
					
						
							|  |  |  |         """Returns a temporary filename for the given filename.""" | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |         if self.params.get('nopart', False) or filename == '-' or \ | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |                 (os.path.exists(encodeFilename(filename)) and not os.path.isfile(encodeFilename(filename))): | 
					
						
							|  |  |  |             return filename | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |         return filename + '.part' | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def undo_temp_name(self, filename): | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |         if filename.endswith('.part'): | 
					
						
							|  |  |  |             return filename[:-len('.part')] | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         return filename | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-19 18:34:25 +01:00
										 |  |  |     def ytdl_filename(self, filename): | 
					
						
							|  |  |  |         return filename + '.ytdl' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |     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: | 
					
						
							| 
									
										
										
										
											2015-12-20 07:00:39 +06:00
										 |  |  |             self.report_error('unable to rename file: %s' % error_to_compat_str(err)) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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)) | 
					
						
							| 
									
										
										
										
											2015-03-27 13:02:20 +01:00
										 |  |  |         except Exception: | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |             pass | 
					
						
							|  |  |  |         return filetime | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def report_destination(self, filename): | 
					
						
							|  |  |  |         """Report destination filename.""" | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |         self.to_screen('[download] Destination: ' + filename) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _report_progress_status(self, msg, is_last_line=False): | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |         fullmsg = '[download] ' + msg | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         if self.params.get('progress_with_newline', False): | 
					
						
							|  |  |  |             self.to_screen(fullmsg) | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2016-03-03 19:24:24 +08:00
										 |  |  |             if compat_os_name == 'nt': | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |                 prev_len = getattr(self, '_report_progress_prev_line_length', | 
					
						
							|  |  |  |                                    0) | 
					
						
							|  |  |  |                 if prev_len > len(fullmsg): | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |                     fullmsg += ' ' * (prev_len - len(fullmsg)) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |                 self._report_progress_prev_line_length = len(fullmsg) | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |                 clear_line = '\r' | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |                 clear_line = ('\r\x1b[K' if sys.stderr.isatty() else '\r') | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |             self.to_screen(clear_line + fullmsg, skip_eol=not is_last_line) | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |         self.to_console_title('youtube-dl ' + msg) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-17 21:37:48 +01:00
										 |  |  |     def report_progress(self, s): | 
					
						
							|  |  |  |         if s['status'] == 'finished': | 
					
						
							|  |  |  |             if self.params.get('noprogress', False): | 
					
						
							|  |  |  |                 self.to_screen('[download] Download completed') | 
					
						
							|  |  |  |             else: | 
					
						
							| 
									
										
										
										
											2018-03-24 16:27:36 +07:00
										 |  |  |                 msg_template = '100%%' | 
					
						
							| 
									
										
										
										
											2016-10-08 09:27:24 -04:00
										 |  |  |                 if s.get('total_bytes') is not None: | 
					
						
							|  |  |  |                     s['_total_bytes_str'] = format_bytes(s['total_bytes']) | 
					
						
							| 
									
										
										
										
											2018-03-24 16:27:36 +07:00
										 |  |  |                     msg_template += ' of %(_total_bytes_str)s' | 
					
						
							| 
									
										
										
										
											2015-02-17 21:37:48 +01:00
										 |  |  |                 if s.get('elapsed') is not None: | 
					
						
							|  |  |  |                     s['_elapsed_str'] = self.format_seconds(s['elapsed']) | 
					
						
							| 
									
										
										
										
											2016-10-08 09:27:24 -04:00
										 |  |  |                     msg_template += ' in %(_elapsed_str)s' | 
					
						
							| 
									
										
										
										
											2015-02-17 21:37:48 +01:00
										 |  |  |                 self._report_progress_status( | 
					
						
							|  |  |  |                     msg_template % s, is_last_line=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if self.params.get('noprogress'): | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |             return | 
					
						
							| 
									
										
										
										
											2015-02-17 21:37:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if s['status'] != 'downloading': | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if s.get('eta') is not None: | 
					
						
							|  |  |  |             s['_eta_str'] = self.format_eta(s['eta']) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2015-02-17 21:37:48 +01:00
										 |  |  |             s['_eta_str'] = 'Unknown ETA' | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-17 21:37:48 +01:00
										 |  |  |         if s.get('total_bytes') and s.get('downloaded_bytes') is not None: | 
					
						
							|  |  |  |             s['_percent_str'] = self.format_percent(100 * s['downloaded_bytes'] / s['total_bytes']) | 
					
						
							|  |  |  |         elif s.get('total_bytes_estimate') and s.get('downloaded_bytes') is not None: | 
					
						
							|  |  |  |             s['_percent_str'] = self.format_percent(100 * s['downloaded_bytes'] / s['total_bytes_estimate']) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             if s.get('downloaded_bytes') == 0: | 
					
						
							|  |  |  |                 s['_percent_str'] = self.format_percent(0) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 s['_percent_str'] = 'Unknown %' | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-17 21:37:48 +01:00
										 |  |  |         if s.get('speed') is not None: | 
					
						
							|  |  |  |             s['_speed_str'] = self.format_speed(s['speed']) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             s['_speed_str'] = 'Unknown speed' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if s.get('total_bytes') is not None: | 
					
						
							|  |  |  |             s['_total_bytes_str'] = format_bytes(s['total_bytes']) | 
					
						
							|  |  |  |             msg_template = '%(_percent_str)s of %(_total_bytes_str)s at %(_speed_str)s ETA %(_eta_str)s' | 
					
						
							|  |  |  |         elif s.get('total_bytes_estimate') is not None: | 
					
						
							|  |  |  |             s['_total_bytes_estimate_str'] = format_bytes(s['total_bytes_estimate']) | 
					
						
							|  |  |  |             msg_template = '%(_percent_str)s of ~%(_total_bytes_estimate_str)s at %(_speed_str)s ETA %(_eta_str)s' | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2015-02-17 21:37:48 +01:00
										 |  |  |             if s.get('downloaded_bytes') is not None: | 
					
						
							|  |  |  |                 s['_downloaded_bytes_str'] = format_bytes(s['downloaded_bytes']) | 
					
						
							|  |  |  |                 if s.get('elapsed'): | 
					
						
							|  |  |  |                     s['_elapsed_str'] = self.format_seconds(s['elapsed']) | 
					
						
							|  |  |  |                     msg_template = '%(_downloaded_bytes_str)s at %(_speed_str)s (%(_elapsed_str)s)' | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     msg_template = '%(_downloaded_bytes_str)s at %(_speed_str)s' | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 msg_template = '%(_percent_str)s % at %(_speed_str)s ETA %(_eta_str)s' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._report_progress_status(msg_template % s) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def report_resuming_byte(self, resume_len): | 
					
						
							|  |  |  |         """Report attempt to resume at given byte.""" | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |         self.to_screen('[download] Resuming download at byte %s' % resume_len) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-26 23:55:48 +07:00
										 |  |  |     def report_retry(self, err, count, retries): | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         """Report retry in case of HTTP error 5xx""" | 
					
						
							| 
									
										
										
										
											2016-03-19 20:51:30 +06:00
										 |  |  |         self.to_screen( | 
					
						
							| 
									
										
										
										
											2017-08-26 23:55:48 +07:00
										 |  |  |             '[download] Got server HTTP error: %s. Retrying (attempt %d of %s)...' | 
					
						
							|  |  |  |             % (error_to_compat_str(err), count, self.format_retries(retries))) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def report_file_already_downloaded(self, file_name): | 
					
						
							|  |  |  |         """Report file has already been fully downloaded.""" | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |             self.to_screen('[download] %s has already been downloaded' % file_name) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         except UnicodeEncodeError: | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |             self.to_screen('[download] The file has already been downloaded') | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def report_unable_to_resume(self): | 
					
						
							|  |  |  |         """Report it was impossible to resume download.""" | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |         self.to_screen('[download] Unable to resume') | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def download(self, filename, info_dict): | 
					
						
							|  |  |  |         """Download to a filename using the info from info_dict
 | 
					
						
							|  |  |  |         Return True on success and False otherwise | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2015-01-23 12:05:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-25 19:37:20 +03:00
										 |  |  |         nooverwrites_and_exists = ( | 
					
						
							| 
									
										
										
										
											2019-05-11 03:56:22 +07:00
										 |  |  |             self.params.get('nooverwrites', False) | 
					
						
							|  |  |  |             and os.path.exists(encodeFilename(filename)) | 
					
						
							| 
									
										
										
										
											2014-09-25 19:37:20 +03:00
										 |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-28 18:07:50 +01:00
										 |  |  |         if not hasattr(filename, 'write'): | 
					
						
							|  |  |  |             continuedl_and_exists = ( | 
					
						
							| 
									
										
										
										
											2019-05-11 03:56:22 +07:00
										 |  |  |                 self.params.get('continuedl', True) | 
					
						
							|  |  |  |                 and os.path.isfile(encodeFilename(filename)) | 
					
						
							|  |  |  |                 and not self.params.get('nopart', False) | 
					
						
							| 
									
										
										
										
											2016-06-28 18:07:50 +01:00
										 |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Check file already present | 
					
						
							|  |  |  |             if filename != '-' and (nooverwrites_and_exists or continuedl_and_exists): | 
					
						
							|  |  |  |                 self.report_file_already_downloaded(filename) | 
					
						
							|  |  |  |                 self._hook_progress({ | 
					
						
							|  |  |  |                     'filename': filename, | 
					
						
							|  |  |  |                     'status': 'finished', | 
					
						
							|  |  |  |                     'total_bytes': os.path.getsize(encodeFilename(filename)), | 
					
						
							|  |  |  |                 }) | 
					
						
							|  |  |  |                 return True | 
					
						
							| 
									
										
										
										
											2013-12-23 16:03:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-09 03:47:56 +07:00
										 |  |  |         min_sleep_interval = self.params.get('sleep_interval') | 
					
						
							|  |  |  |         if min_sleep_interval: | 
					
						
							|  |  |  |             max_sleep_interval = self.params.get('max_sleep_interval', min_sleep_interval) | 
					
						
							|  |  |  |             sleep_interval = random.uniform(min_sleep_interval, max_sleep_interval) | 
					
						
							| 
									
										
										
										
											2016-12-24 21:05:41 +05:30
										 |  |  |             self.to_screen( | 
					
						
							|  |  |  |                 '[download] Sleeping %s seconds...' % ( | 
					
						
							|  |  |  |                     int(sleep_interval) if sleep_interval.is_integer() | 
					
						
							|  |  |  |                     else '%.2f' % sleep_interval)) | 
					
						
							| 
									
										
										
										
											2015-01-23 12:05:01 +01:00
										 |  |  |             time.sleep(sleep_interval) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-23 16:03:06 +01:00
										 |  |  |         return self.real_download(filename, info_dict) | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def real_download(self, filename, info_dict): | 
					
						
							|  |  |  |         """Real download process. Redefine in subclasses.""" | 
					
						
							| 
									
										
										
										
											2014-11-16 15:06:59 +01:00
										 |  |  |         raise NotImplementedError('This method must be implemented by subclasses') | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _hook_progress(self, status): | 
					
						
							|  |  |  |         for ph in self._progress_hooks: | 
					
						
							|  |  |  |             ph(status) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_progress_hook(self, ph): | 
					
						
							| 
									
										
										
										
											2014-12-15 01:26:18 +01:00
										 |  |  |         # See YoutubeDl.py (search for progress_hooks) for a description of | 
					
						
							|  |  |  |         # this interface | 
					
						
							| 
									
										
										
										
											2013-09-23 17:59:27 +02:00
										 |  |  |         self._progress_hooks.append(ph) | 
					
						
							| 
									
										
										
										
											2015-01-24 01:38:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-26 04:30:45 +06:00
										 |  |  |     def _debug_cmd(self, args, exe=None): | 
					
						
							| 
									
										
										
										
											2015-01-24 01:38:48 +01:00
										 |  |  |         if not self.params.get('verbose', False): | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-26 04:30:45 +06:00
										 |  |  |         str_args = [decodeArgument(a) for a in args] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-24 01:38:48 +01:00
										 |  |  |         if exe is None: | 
					
						
							| 
									
										
										
										
											2015-04-26 04:30:45 +06:00
										 |  |  |             exe = os.path.basename(str_args[0]) | 
					
						
							| 
									
										
										
										
											2015-01-24 01:38:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.to_screen('[debug] %s command line: %s' % ( | 
					
						
							|  |  |  |             exe, shell_quote(str_args))) |