diff --git a/onionshare/helpers.py b/onionshare/helpers.py index 1a48147f..c2c4793f 100644 --- a/onionshare/helpers.py +++ b/onionshare/helpers.py @@ -19,6 +19,8 @@ along with this program. If not, see . """ import os, inspect, hashlib, base64, hmac, platform, zipfile, tempfile from itertools import izip +import math +import time # hack to make unicode filenames work (#141) import sys @@ -112,6 +114,44 @@ def human_readable_filesize(b): return '{0:.1f} {1:s}'.format(round(b, 1), units[u]) +def format_seconds(seconds): + """Return a human-readable string of the format 1d2h3m4s""" + seconds_in_a_minute = 60 + seconds_in_an_hour = seconds_in_a_minute * 60 + seconds_in_a_day = seconds_in_an_hour * 24 + + days = math.floor(seconds / seconds_in_a_day) + + hour_seconds = seconds % seconds_in_a_day + hours = math.floor(hour_seconds / seconds_in_an_hour) + + minute_seconds = hour_seconds % seconds_in_an_hour + minutes = math.floor(minute_seconds / seconds_in_a_minute) + + remaining_seconds = minute_seconds % seconds_in_a_minute + seconds = math.ceil(remaining_seconds) + + human_readable = [] + if days > 0: + human_readable.append("{}d".format(int(days))) + if hours > 0: + human_readable.append("{}h".format(int(hours))) + if minutes > 0: + human_readable.append("{}m".format(int(minutes))) + if seconds > 0: + human_readable.append("{}s".format(int(seconds))) + return ''.join(human_readable) + + +def estimated_time_remaining(bytes_downloaded, total_bytes, started): + now = time.time() + time_elapsed = now - started # in seconds + download_rate = bytes_downloaded / time_elapsed + remaining_bytes = total_bytes - bytes_downloaded + eta = remaining_bytes / download_rate + return format_seconds(eta) + + def is_root(): """ Returns if user is root. diff --git a/onionshare_gui/downloads.py b/onionshare_gui/downloads.py index 0ecb49f0..8b563d1c 100644 --- a/onionshare_gui/downloads.py +++ b/onionshare_gui/downloads.py @@ -17,12 +17,68 @@ GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . """ +import time + from PyQt4 import QtCore, QtGui import common from onionshare import strings, helpers +class Download(object): + + def __init__(self, download_id, total_bytes): + self.download_id = download_id + self.started = time.time() + self.total_bytes = total_bytes + self.downloaded_bytes = 0 + + # make a new progress bar + self.progress_bar = QtGui.QProgressBar() + self.progress_bar.setTextVisible(True) + self.progress_bar.setAlignment(QtCore.Qt.AlignHCenter) + self.progress_bar.setMinimum(0) + self.progress_bar.setMaximum(total_bytes) + self.progress_bar.setValue(0) + self.progress_bar.setStyleSheet( + "QProgressBar::chunk { background-color: #05B8CC; }") + self.progress_bar.total_bytes = total_bytes + + # start at 0 + self.update(0) + + def update(self, downloaded_bytes): + self.downloaded_bytes = downloaded_bytes + + self.progress_bar.setValue(downloaded_bytes) + if downloaded_bytes == self.progress_bar.total_bytes: + pb_fmt = "%p%, Time Elapsed: {0:s}".format( + helpers.format_seconds(time.time() - self.started)) + else: + elapsed = time.time() - self.started + if elapsed < 10: + # Wait a couple of seconds for the download rate to stabilize. + # This prevents an "Windows copy dialog"-esque experience at + # the beginning of the download. + pb_fmt = "{0:s}, %p% (Computing ETA)".format( + helpers.human_readable_filesize(downloaded_bytes)) + else: + pb_fmt = "{0:s}, ETA: {1:s}, %p%".format( + helpers.human_readable_filesize(downloaded_bytes), + self.estimated_time_remaining) + + self.progress_bar.setFormat(pb_fmt) + + def cancel(self): + self.progress_bar.setFormat(strings._('gui_canceled')) + + @property + def estimated_time_remaining(self): + return helpers.estimated_time_remaining(self.downloaded_bytes, + self.total_bytes, + self.started) + + class Downloads(QtGui.QVBoxLayout): """ The downloads chunk of the GUI. This lists all of the active download @@ -31,7 +87,7 @@ class Downloads(QtGui.QVBoxLayout): def __init__(self): super(Downloads, self).__init__() - self.progress_bars = {} + self.downloads = {} # downloads label self.downloads_label = QtGui.QLabel(strings._('gui_downloads', True)) @@ -46,40 +102,19 @@ class Downloads(QtGui.QVBoxLayout): """ self.downloads_label.show() - # make a new progress bar - pb = QtGui.QProgressBar() - pb.setTextVisible(True) - pb.setAlignment(QtCore.Qt.AlignHCenter) - pb.setMinimum(0) - pb.setMaximum(total_bytes) - pb.setValue(0) - pb.setStyleSheet("QProgressBar::chunk { background-color: #05B8CC; }") - pb.total_bytes = total_bytes - # add it to the list - self.progress_bars[download_id] = pb - self.addWidget(pb) + download = Download(download_id, total_bytes) + self.downloads[download_id] = download + self.addWidget(download.progress_bar) - # start at 0 - self.update_download(download_id, total_bytes, 0) - - def update_download(self, download_id, total_bytes, downloaded_bytes): + def update_download(self, download_id, downloaded_bytes): """ Update the progress of a download progress bar. """ - if download_id not in self.progress_bars: - self.add_download(download_id, total_bytes) - - pb = self.progress_bars[download_id] - pb.setValue(downloaded_bytes) - if downloaded_bytes == pb.total_bytes: - pb.setFormat("%p%") - else: - pb.setFormat("{0:s}, %p%".format(helpers.human_readable_filesize(downloaded_bytes))) + self.downloads[download_id].update(downloaded_bytes) def cancel_download(self, download_id): """ Update a download progress bar to show that it has been canceled. """ - pb = self.progress_bars[download_id] - pb.setFormat(strings._('gui_canceled')) + self.downloads[download_id].cancel() diff --git a/onionshare_gui/onionshare_gui.py b/onionshare_gui/onionshare_gui.py index df517997..9f2eea70 100644 --- a/onionshare_gui/onionshare_gui.py +++ b/onionshare_gui/onionshare_gui.py @@ -216,7 +216,7 @@ class OnionShareGui(QtGui.QWidget): self.downloads.add_download(event["data"]["id"], web.zip_filesize) elif event["type"] == web.REQUEST_PROGRESS: - self.downloads.update_download(event["data"]["id"], web.zip_filesize, event["data"]["bytes"]) + self.downloads.update_download(event["data"]["id"], event["data"]["bytes"]) # is the download complete? if event["data"]["bytes"] == web.zip_filesize: