Merge branch 'oed-prevent_downloading_existing_file'
This commit is contained in:
commit
aef693805a
@ -133,6 +133,7 @@ Contributors, sorted by the number of commits in descending order:
|
|||||||
* Mathias Fussenegger
|
* Mathias Fussenegger
|
||||||
* Larry Hynes
|
* Larry Hynes
|
||||||
* Johannes Altmanninger
|
* Johannes Altmanninger
|
||||||
|
* Joel Torstensson
|
||||||
* Regina Hug
|
* Regina Hug
|
||||||
* Peter Vilim
|
* Peter Vilim
|
||||||
* Helen Sherwood-Taylor
|
* Helen Sherwood-Taylor
|
||||||
|
@ -171,6 +171,7 @@ class DownloadItem(QObject):
|
|||||||
_read_timer: A QTimer which reads the QNetworkReply into self._buffer
|
_read_timer: A QTimer which reads the QNetworkReply into self._buffer
|
||||||
periodically.
|
periodically.
|
||||||
_retry_info: A RetryInfo instance.
|
_retry_info: A RetryInfo instance.
|
||||||
|
_win_id: The window ID the DownloadItem runs in.
|
||||||
|
|
||||||
Signals:
|
Signals:
|
||||||
data_changed: The downloads metadata changed.
|
data_changed: The downloads metadata changed.
|
||||||
@ -193,7 +194,7 @@ class DownloadItem(QObject):
|
|||||||
redirected = pyqtSignal(QNetworkRequest, QNetworkReply)
|
redirected = pyqtSignal(QNetworkRequest, QNetworkReply)
|
||||||
do_retry = pyqtSignal('QNetworkReply')
|
do_retry = pyqtSignal('QNetworkReply')
|
||||||
|
|
||||||
def __init__(self, reply, parent=None):
|
def __init__(self, reply, win_id, parent=None):
|
||||||
"""Constructor.
|
"""Constructor.
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
@ -217,6 +218,7 @@ class DownloadItem(QObject):
|
|||||||
self.fileobj = None
|
self.fileobj = None
|
||||||
self._filename = None
|
self._filename = None
|
||||||
self.init_reply(reply)
|
self.init_reply(reply)
|
||||||
|
self._win_id = win_id
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return utils.get_repr(self, basename=self.basename)
|
return utils.get_repr(self, basename=self.basename)
|
||||||
@ -256,6 +258,27 @@ class DownloadItem(QObject):
|
|||||||
name=self.basename, speed=speed, remaining=remaining,
|
name=self.basename, speed=speed, remaining=remaining,
|
||||||
perc=perc, down=down, total=total, errmsg=errmsg))
|
perc=perc, down=down, total=total, errmsg=errmsg))
|
||||||
|
|
||||||
|
def _create_fileobj(self):
|
||||||
|
"""Creates a file object using the internal filename."""
|
||||||
|
try:
|
||||||
|
fileobj = open(self._filename, 'wb')
|
||||||
|
except OSError as e:
|
||||||
|
self._die(e.strerror)
|
||||||
|
else:
|
||||||
|
self.set_fileobj(fileobj)
|
||||||
|
|
||||||
|
def _ask_overwrite_question(self):
|
||||||
|
"""Create a Question object to be asked."""
|
||||||
|
q = usertypes.Question(self)
|
||||||
|
q.text = self._filename + " already exists. Overwrite? (y/n)"
|
||||||
|
q.mode = usertypes.PromptMode.yesno
|
||||||
|
q.answered_yes.connect(self._create_fileobj)
|
||||||
|
q.answered_no.connect(functools.partial(self.cancel, False))
|
||||||
|
q.cancelled.connect(functools.partial(self.cancel, False))
|
||||||
|
message_bridge = objreg.get('message-bridge', scope='window',
|
||||||
|
window=self._win_id)
|
||||||
|
message_bridge.ask(q, blocking=False)
|
||||||
|
|
||||||
def _die(self, msg):
|
def _die(self, msg):
|
||||||
"""Abort the download and emit an error."""
|
"""Abort the download and emit an error."""
|
||||||
assert not self.successful
|
assert not self.successful
|
||||||
@ -312,8 +335,12 @@ class DownloadItem(QObject):
|
|||||||
return utils.interpolate_color(
|
return utils.interpolate_color(
|
||||||
start, stop, self.stats.percentage(), system)
|
start, stop, self.stats.percentage(), system)
|
||||||
|
|
||||||
def cancel(self):
|
def cancel(self, remove_data=True):
|
||||||
"""Cancel the download."""
|
"""Cancel the download.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
remove_data: Whether to remove the downloaded data.
|
||||||
|
"""
|
||||||
log.downloads.debug("cancelled")
|
log.downloads.debug("cancelled")
|
||||||
self._read_timer.stop()
|
self._read_timer.stop()
|
||||||
self.cancelled.emit()
|
self.cancelled.emit()
|
||||||
@ -325,7 +352,8 @@ class DownloadItem(QObject):
|
|||||||
if self.fileobj is not None:
|
if self.fileobj is not None:
|
||||||
self.fileobj.close()
|
self.fileobj.close()
|
||||||
try:
|
try:
|
||||||
if self._filename is not None and os.path.exists(self._filename):
|
if (self._filename is not None and os.path.exists(self._filename)
|
||||||
|
and remove_data):
|
||||||
os.remove(self._filename)
|
os.remove(self._filename)
|
||||||
except OSError:
|
except OSError:
|
||||||
log.downloads.exception("Failed to remove partial file")
|
log.downloads.exception("Failed to remove partial file")
|
||||||
@ -376,12 +404,12 @@ class DownloadItem(QObject):
|
|||||||
self._filename = os.path.join(download_dir, filename)
|
self._filename = os.path.join(download_dir, filename)
|
||||||
self.basename = filename
|
self.basename = filename
|
||||||
log.downloads.debug("Setting filename to {}".format(filename))
|
log.downloads.debug("Setting filename to {}".format(filename))
|
||||||
try:
|
if os.path.isfile(self._filename):
|
||||||
fileobj = open(self._filename, 'wb')
|
# The file already exists, so ask the user if it should be
|
||||||
except OSError as e:
|
# overwritten.
|
||||||
self._die(e.strerror)
|
self._ask_overwrite_question()
|
||||||
else:
|
else:
|
||||||
self.set_fileobj(fileobj)
|
self._create_fileobj()
|
||||||
|
|
||||||
def set_fileobj(self, fileobj):
|
def set_fileobj(self, fileobj):
|
||||||
""""Set the file object to write the download to.
|
""""Set the file object to write the download to.
|
||||||
@ -666,7 +694,7 @@ class DownloadManager(QAbstractListModel):
|
|||||||
_inline, suggested_filename = http.parse_content_disposition(reply)
|
_inline, suggested_filename = http.parse_content_disposition(reply)
|
||||||
log.downloads.debug("fetch: {} -> {}".format(reply.url(),
|
log.downloads.debug("fetch: {} -> {}".format(reply.url(),
|
||||||
suggested_filename))
|
suggested_filename))
|
||||||
download = DownloadItem(reply, self)
|
download = DownloadItem(reply, self._win_id, self)
|
||||||
download.cancelled.connect(
|
download.cancelled.connect(
|
||||||
functools.partial(self.remove_item, download))
|
functools.partial(self.remove_item, download))
|
||||||
if config.get('ui', 'remove-finished-downloads') or auto_remove:
|
if config.get('ui', 'remove-finished-downloads') or auto_remove:
|
||||||
|
Loading…
Reference in New Issue
Block a user