|
|
@ -77,26 +77,43 @@ class FileDownloader(object):
|
|
|
|
@staticmethod
|
|
|
|
@staticmethod
|
|
|
|
def calc_percent(byte_counter, data_len):
|
|
|
|
def calc_percent(byte_counter, data_len):
|
|
|
|
if data_len is None:
|
|
|
|
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 '---.-%'
|
|
|
|
return '%6s' % ('%3.1f%%' % (float(byte_counter) / float(data_len) * 100.0))
|
|
|
|
return '%6s' % ('%3.1f%%' % percent)
|
|
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
@staticmethod
|
|
|
|
def calc_eta(start, now, total, current):
|
|
|
|
def calc_eta(start, now, total, current):
|
|
|
|
if total is None:
|
|
|
|
if total is None:
|
|
|
|
return '--:--'
|
|
|
|
return None
|
|
|
|
dif = now - start
|
|
|
|
dif = now - start
|
|
|
|
if current == 0 or dif < 0.001: # One millisecond
|
|
|
|
if current == 0 or dif < 0.001: # One millisecond
|
|
|
|
return '--:--'
|
|
|
|
return None
|
|
|
|
rate = float(current) / dif
|
|
|
|
rate = float(current) / dif
|
|
|
|
eta = int((float(total) - float(current)) / rate)
|
|
|
|
return int((float(total) - float(current)) / rate)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
|
|
|
def format_eta(eta):
|
|
|
|
|
|
|
|
if eta is None:
|
|
|
|
|
|
|
|
return '--:--'
|
|
|
|
return FileDownloader.format_seconds(eta)
|
|
|
|
return FileDownloader.format_seconds(eta)
|
|
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
@staticmethod
|
|
|
|
def calc_speed(start, now, bytes):
|
|
|
|
def calc_speed(start, now, bytes):
|
|
|
|
dif = now - start
|
|
|
|
dif = now - start
|
|
|
|
if bytes == 0 or dif < 0.001: # One millisecond
|
|
|
|
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' % '---b/s'
|
|
|
|
return '%10s' % ('%s/s' % FileDownloader.format_bytes(float(bytes) / dif))
|
|
|
|
return '%10s' % ('%s/s' % FileDownloader.format_bytes(speed))
|
|
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
@staticmethod
|
|
|
|
def best_block_size(elapsed_time, bytes):
|
|
|
|
def best_block_size(elapsed_time, bytes):
|
|
|
@ -205,11 +222,14 @@ class FileDownloader(object):
|
|
|
|
"""Report destination filename."""
|
|
|
|
"""Report destination filename."""
|
|
|
|
self.to_screen(u'[download] Destination: ' + filename)
|
|
|
|
self.to_screen(u'[download] Destination: ' + filename)
|
|
|
|
|
|
|
|
|
|
|
|
def report_progress(self, percent_str, data_len_str, speed_str, eta_str):
|
|
|
|
def report_progress(self, percent, data_len_str, speed, eta):
|
|
|
|
"""Report download progress."""
|
|
|
|
"""Report download progress."""
|
|
|
|
if self.params.get('noprogress', False):
|
|
|
|
if self.params.get('noprogress', False):
|
|
|
|
return
|
|
|
|
return
|
|
|
|
clear_line = (u'\x1b[K' if sys.stderr.isatty() and os.name != 'nt' else u'')
|
|
|
|
clear_line = (u'\x1b[K' if sys.stderr.isatty() and os.name != 'nt' else u'')
|
|
|
|
|
|
|
|
eta_str = self.format_eta(eta)
|
|
|
|
|
|
|
|
percent_str = self.format_percent(percent)
|
|
|
|
|
|
|
|
speed_str = self.format_speed(speed)
|
|
|
|
if self.params.get('progress_with_newline', False):
|
|
|
|
if self.params.get('progress_with_newline', False):
|
|
|
|
self.to_screen(u'[download] %s of %s at %s ETA %s' %
|
|
|
|
self.to_screen(u'[download] %s of %s at %s ETA %s' %
|
|
|
|
(percent_str, data_len_str, speed_str, eta_str))
|
|
|
|
(percent_str, data_len_str, speed_str, eta_str))
|
|
|
@ -378,6 +398,7 @@ class FileDownloader(object):
|
|
|
|
self._hook_progress({
|
|
|
|
self._hook_progress({
|
|
|
|
'filename': filename,
|
|
|
|
'filename': filename,
|
|
|
|
'status': 'finished',
|
|
|
|
'status': 'finished',
|
|
|
|
|
|
|
|
'total_bytes': os.path.getsize(encodeFilename(filename)),
|
|
|
|
})
|
|
|
|
})
|
|
|
|
return True
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
@ -524,13 +545,14 @@ class FileDownloader(object):
|
|
|
|
block_size = self.best_block_size(after - before, len(data_block))
|
|
|
|
block_size = self.best_block_size(after - before, len(data_block))
|
|
|
|
|
|
|
|
|
|
|
|
# Progress message
|
|
|
|
# Progress message
|
|
|
|
speed_str = self.calc_speed(start, time.time(), byte_counter - resume_len)
|
|
|
|
speed = self.calc_speed(start, time.time(), byte_counter - resume_len)
|
|
|
|
if data_len is None:
|
|
|
|
if data_len is None:
|
|
|
|
self.report_progress('Unknown %', data_len_str, speed_str, 'Unknown ETA')
|
|
|
|
self.report_progress('Unknown %', data_len_str, speed_str, 'Unknown ETA')
|
|
|
|
|
|
|
|
eta = None
|
|
|
|
else:
|
|
|
|
else:
|
|
|
|
percent_str = self.calc_percent(byte_counter, data_len)
|
|
|
|
percent = self.calc_percent(byte_counter, data_len)
|
|
|
|
eta_str = self.calc_eta(start, time.time(), data_len - resume_len, byte_counter - resume_len)
|
|
|
|
eta = self.calc_eta(start, time.time(), data_len - resume_len, byte_counter - resume_len)
|
|
|
|
self.report_progress(percent_str, data_len_str, speed_str, eta_str)
|
|
|
|
self.report_progress(percent, data_len_str, speed, eta)
|
|
|
|
|
|
|
|
|
|
|
|
self._hook_progress({
|
|
|
|
self._hook_progress({
|
|
|
|
'downloaded_bytes': byte_counter,
|
|
|
|
'downloaded_bytes': byte_counter,
|
|
|
@ -538,6 +560,8 @@ class FileDownloader(object):
|
|
|
|
'tmpfilename': tmpfilename,
|
|
|
|
'tmpfilename': tmpfilename,
|
|
|
|
'filename': filename,
|
|
|
|
'filename': filename,
|
|
|
|
'status': 'downloading',
|
|
|
|
'status': 'downloading',
|
|
|
|
|
|
|
|
'eta': eta,
|
|
|
|
|
|
|
|
'speed': speed,
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
# Apply rate limit
|
|
|
|
# Apply rate limit
|
|
|
@ -580,6 +604,8 @@ class FileDownloader(object):
|
|
|
|
* downloaded_bytes: Bytes on disks
|
|
|
|
* downloaded_bytes: Bytes on disks
|
|
|
|
* total_bytes: Total bytes, None if unknown
|
|
|
|
* total_bytes: Total bytes, None if unknown
|
|
|
|
* tmpfilename: The filename we're currently writing to
|
|
|
|
* 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")
|
|
|
|
Hooks are guaranteed to be called at least once (with status "finished")
|
|
|
|
if the download is successful.
|
|
|
|
if the download is successful.
|
|
|
|