瀏覽代碼

backups: use backup_cancel() to interrupt backup/restore operation (#793)

Marek Marczykowski-Górecki 10 年之前
父節點
當前提交
3016c06d91
共有 2 個文件被更改,包括 62 次插入29 次删除
  1. 25 21
      qubesmanager/backup.py
  2. 37 8
      qubesmanager/restore.py

+ 25 - 21
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

+ 37 - 8
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)"),
+                      '<b><font color="red">{0}</font></b>'.format("Restore aborted!"))
+        elif len(err_msg) > 0:
             thread_monitor.set_error_msg('\n'.join(err_msg))
             self.emit(SIGNAL("restore_progress(QString)"),'<b><font color="red">{0}</font></b>'.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)"),'<font '
+                                                              'color="red">{'
+                                                              '0}</font>'
+                          .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())