From 3016c06d916513c9d2506e7798d5a66b9cf836ee Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marek=20Marczykowski-G=C3=B3recki?= Date: Sat, 8 Mar 2014 03:58:59 +0100 Subject: [PATCH] backups: use backup_cancel() to interrupt backup/restore operation (#793) --- qubesmanager/backup.py | 46 ++++++++++++++++++++++------------------- qubesmanager/restore.py | 45 +++++++++++++++++++++++++++++++++------- 2 files changed, 62 insertions(+), 29 deletions(-) diff --git a/qubesmanager/backup.py b/qubesmanager/backup.py index 640096f..1172b7e 100644 --- a/qubesmanager/backup.py +++ b/qubesmanager/backup.py @@ -24,6 +24,7 @@ import sys import os import signal +import shutil from PyQt4.QtCore import * from PyQt4.QtGui import * @@ -68,6 +69,8 @@ class BackupVMsWindow(Ui_Backup, QWizard): self.dev_mount_path = None self.func_output = [] self.selected_vms = [] + self.tmpdir_to_remove = None + self.canceled = False self.vm = self.qvm_collection[0] self.files_to_backup = None @@ -293,6 +296,11 @@ class BackupVMsWindow(Ui_Backup, QWizard): encrypted=self.encryption_checkbox.isChecked(), appvm=self.target_appvm) #simulate_long_lasting_proces(10, self.update_progress_bar) + except backup.BackupCanceledError as ex: + msg.append(str(ex)) + self.canceled = True + if ex.tmpdir: + self.tmpdir_to_remove = ex.tmpdir except Exception as ex: print "Exception:",ex msg.append(str(ex)) @@ -328,12 +336,10 @@ class BackupVMsWindow(Ui_Backup, QWizard): elif self.currentPage() is self.commit_page: self.button(self.FinishButton).setDisabled(True) - self.button(self.CancelButton).setDisabled(True) self.thread_monitor = ThreadMonitor() thread = threading.Thread (target= self.__do_backup__ , args=(self.thread_monitor,)) thread.daemon = True thread.start() - self.button(self.CancelButton).setDisabled(False) counter = 0 while not self.thread_monitor.is_finished(): @@ -341,9 +347,18 @@ class BackupVMsWindow(Ui_Backup, QWizard): time.sleep (0.1) if not self.thread_monitor.success: - self.progress_status.setText("Backup error.") - QMessageBox.warning (self, "Backup error!", "ERROR: {}".format( - self.thread_monitor.error_msg)) + if self.canceled: + self.progress_status.setText("Backup aborted.") + if self.tmpdir_to_remove: + if QMessageBox.warning(None, "Backup aborted", + "Do you want to remove temporary files from " + "%s?" % self.tmpdir_to_remove, + QMessageBox.Yes, QMessageBox.No) == QMessageBox.Yes: + shutil.rmtree(self.tmpdir_to_remove) + else: + self.progress_status.setText("Backup error.") + QMessageBox.warning (self, "Backup error!", "ERROR: {}".format( + self.thread_monitor.error_msg)) else: self.progress_bar.setValue(100) self.progress_status.setText("Backup finished.") @@ -358,6 +373,7 @@ class BackupVMsWindow(Ui_Backup, QWizard): detach_device(self, str(self.dev_combobox.itemData( self.dev_combobox.currentIndex()).toString())) self.dev_mount_path = None + self.button(self.CancelButton).setEnabled(False) self.button(self.FinishButton).setEnabled(True) signal.signal(signal.SIGCHLD, old_sigchld_handler) @@ -365,22 +381,10 @@ class BackupVMsWindow(Ui_Backup, QWizard): #cancell clicked while the backup is in progress. #calling kill on tar. if self.currentPage() is self.commit_page: - manager_pid = os.getpid() - archive_pid_cmd = ["ps" ,"--ppid", str(manager_pid)] - - while not self.thread_monitor.is_finished(): - archive_pid = subprocess.Popen(archive_pid_cmd, stdout = subprocess.PIPE) - output = archive_pid.stdout.readlines() - - for l in output: - if l.strip().endswith("tar"): - os.kill(int(l.split(" ")[0]), signal.SIGTERM) - time.sleep(0.1) - - if self.dev_mount_path != None: - umount_device(self.dev_mount_path) - self.done(0) - + if backup.backup_cancel(): + self.button(self.CancelButton).setDisabled(True) + else: + self.done(0) def has_selected_vms(self): return self.select_vms_widget.selected_list.count() > 0 diff --git a/qubesmanager/restore.py b/qubesmanager/restore.py index 01197bf..3729a03 100644 --- a/qubesmanager/restore.py +++ b/qubesmanager/restore.py @@ -23,6 +23,7 @@ import sys import os +import shutil from PyQt4.QtCore import * from PyQt4.QtGui import * @@ -67,6 +68,8 @@ class RestoreVMsWindow(Ui_Restore, QWizard): self.vms_to_restore = None self.func_output = [] self.feedback_queue = Queue() + self.canceled = False + self.tmpdir_to_remove = None self.excluded = {} @@ -189,12 +192,19 @@ class RestoreVMsWindow(Ui_Restore, QWizard): print_callback=self.restore_output, error_callback=self.restore_error_output, progress_callback=self.update_progress_bar) + except backup.BackupCanceledError as ex: + self.canceled = True + self.tmpdir_to_remove = ex.tmpdir + err_msg.append(str(ex)) except Exception as ex: print "Exception:", ex err_msg.append(str(ex)) self.qvm_collection.unlock_db() - if len(err_msg) > 0 : + if self.canceled: + self.emit(SIGNAL("restore_progress(QString)"), + '{0}'.format("Restore aborted!")) + elif len(err_msg) > 0: thread_monitor.set_error_msg('\n'.join(err_msg)) self.emit(SIGNAL("restore_progress(QString)"),'{0}'.format("Finished with errors!")) else: @@ -229,7 +239,6 @@ class RestoreVMsWindow(Ui_Restore, QWizard): self.confirm_page.emit(SIGNAL("completeChanged()")) elif self.currentPage() is self.commit_page: - self.button(self.CancelButton).setDisabled(True) self.button(self.FinishButton).setDisabled(True) self.thread_monitor = ThreadMonitor() @@ -247,7 +256,18 @@ class RestoreVMsWindow(Ui_Restore, QWizard): pass if not self.thread_monitor.success: - QMessageBox.warning (None, "Backup error!", "ERROR: {1}".format(self.vm.name, self.thread_monitor.error_msg)) + if self.canceled: + if self.tmpdir_to_remove and \ + QMessageBox.warning(None, "Restore aborted", + "Do you want to remove temporary " + "files from %s?" % self + .tmpdir_to_remove, + QMessageBox.Yes, QMessageBox.No) == \ + QMessageBox.Yes: + shutil.rmtree(self.tmpdir_to_remove) + else: + QMessageBox.warning (None, "Backup error!", "ERROR: {1}" + .format(self.vm.name, self.thread_monitor.error_msg)) if self.dev_mount_path != None: umount_device(self.dev_mount_path) @@ -257,6 +277,7 @@ class RestoreVMsWindow(Ui_Restore, QWizard): self.progress_bar.setValue(100) self.button(self.FinishButton).setEnabled(True) + self.button(self.CancelButton).setEnabled(False) signal.signal(signal.SIGCHLD, old_sigchld_handler) @@ -269,11 +290,19 @@ class RestoreVMsWindow(Ui_Restore, QWizard): return True def reject(self): - if self.dev_mount_path != None: - umount_device(self.dev_mount_path) - detach_device(self, str(self.dev_combobox.itemData( - self.dev_combobox.currentIndex()).toString())) - self.done(0) + if self.currentPage() is self.commit_page: + if backup.backup_cancel(): + self.emit(SIGNAL("restore_progress(QString)"),'{' + '0}' + .format("Aborting the operation...")) + self.button(self.CancelButton).setDisabled(True) + else: + if self.dev_mount_path != None: + umount_device(self.dev_mount_path) + detach_device(self, str(self.dev_combobox.itemData( + self.dev_combobox.currentIndex()).toString())) + self.done(0) def has_selected_dir(self): backup_location = unicode(self.dir_line_edit.text())