backups: use backup_cancel() to interrupt backup/restore operation (#793)
This commit is contained in:
parent
3d13bfd8a1
commit
3016c06d91
@ -24,6 +24,7 @@
|
|||||||
import sys
|
import sys
|
||||||
import os
|
import os
|
||||||
import signal
|
import signal
|
||||||
|
import shutil
|
||||||
from PyQt4.QtCore import *
|
from PyQt4.QtCore import *
|
||||||
from PyQt4.QtGui import *
|
from PyQt4.QtGui import *
|
||||||
|
|
||||||
@ -68,6 +69,8 @@ class BackupVMsWindow(Ui_Backup, QWizard):
|
|||||||
self.dev_mount_path = None
|
self.dev_mount_path = None
|
||||||
self.func_output = []
|
self.func_output = []
|
||||||
self.selected_vms = []
|
self.selected_vms = []
|
||||||
|
self.tmpdir_to_remove = None
|
||||||
|
self.canceled = False
|
||||||
|
|
||||||
self.vm = self.qvm_collection[0]
|
self.vm = self.qvm_collection[0]
|
||||||
self.files_to_backup = None
|
self.files_to_backup = None
|
||||||
@ -293,6 +296,11 @@ class BackupVMsWindow(Ui_Backup, QWizard):
|
|||||||
encrypted=self.encryption_checkbox.isChecked(),
|
encrypted=self.encryption_checkbox.isChecked(),
|
||||||
appvm=self.target_appvm)
|
appvm=self.target_appvm)
|
||||||
#simulate_long_lasting_proces(10, self.update_progress_bar)
|
#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:
|
except Exception as ex:
|
||||||
print "Exception:",ex
|
print "Exception:",ex
|
||||||
msg.append(str(ex))
|
msg.append(str(ex))
|
||||||
@ -328,12 +336,10 @@ class BackupVMsWindow(Ui_Backup, QWizard):
|
|||||||
|
|
||||||
elif self.currentPage() is self.commit_page:
|
elif self.currentPage() is self.commit_page:
|
||||||
self.button(self.FinishButton).setDisabled(True)
|
self.button(self.FinishButton).setDisabled(True)
|
||||||
self.button(self.CancelButton).setDisabled(True)
|
|
||||||
self.thread_monitor = ThreadMonitor()
|
self.thread_monitor = ThreadMonitor()
|
||||||
thread = threading.Thread (target= self.__do_backup__ , args=(self.thread_monitor,))
|
thread = threading.Thread (target= self.__do_backup__ , args=(self.thread_monitor,))
|
||||||
thread.daemon = True
|
thread.daemon = True
|
||||||
thread.start()
|
thread.start()
|
||||||
self.button(self.CancelButton).setDisabled(False)
|
|
||||||
|
|
||||||
counter = 0
|
counter = 0
|
||||||
while not self.thread_monitor.is_finished():
|
while not self.thread_monitor.is_finished():
|
||||||
@ -341,9 +347,18 @@ class BackupVMsWindow(Ui_Backup, QWizard):
|
|||||||
time.sleep (0.1)
|
time.sleep (0.1)
|
||||||
|
|
||||||
if not self.thread_monitor.success:
|
if not self.thread_monitor.success:
|
||||||
self.progress_status.setText("Backup error.")
|
if self.canceled:
|
||||||
QMessageBox.warning (self, "Backup error!", "ERROR: {}".format(
|
self.progress_status.setText("Backup aborted.")
|
||||||
self.thread_monitor.error_msg))
|
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:
|
else:
|
||||||
self.progress_bar.setValue(100)
|
self.progress_bar.setValue(100)
|
||||||
self.progress_status.setText("Backup finished.")
|
self.progress_status.setText("Backup finished.")
|
||||||
@ -358,6 +373,7 @@ class BackupVMsWindow(Ui_Backup, QWizard):
|
|||||||
detach_device(self, str(self.dev_combobox.itemData(
|
detach_device(self, str(self.dev_combobox.itemData(
|
||||||
self.dev_combobox.currentIndex()).toString()))
|
self.dev_combobox.currentIndex()).toString()))
|
||||||
self.dev_mount_path = None
|
self.dev_mount_path = None
|
||||||
|
self.button(self.CancelButton).setEnabled(False)
|
||||||
self.button(self.FinishButton).setEnabled(True)
|
self.button(self.FinishButton).setEnabled(True)
|
||||||
signal.signal(signal.SIGCHLD, old_sigchld_handler)
|
signal.signal(signal.SIGCHLD, old_sigchld_handler)
|
||||||
|
|
||||||
@ -365,22 +381,10 @@ class BackupVMsWindow(Ui_Backup, QWizard):
|
|||||||
#cancell clicked while the backup is in progress.
|
#cancell clicked while the backup is in progress.
|
||||||
#calling kill on tar.
|
#calling kill on tar.
|
||||||
if self.currentPage() is self.commit_page:
|
if self.currentPage() is self.commit_page:
|
||||||
manager_pid = os.getpid()
|
if backup.backup_cancel():
|
||||||
archive_pid_cmd = ["ps" ,"--ppid", str(manager_pid)]
|
self.button(self.CancelButton).setDisabled(True)
|
||||||
|
else:
|
||||||
while not self.thread_monitor.is_finished():
|
self.done(0)
|
||||||
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)
|
|
||||||
|
|
||||||
|
|
||||||
def has_selected_vms(self):
|
def has_selected_vms(self):
|
||||||
return self.select_vms_widget.selected_list.count() > 0
|
return self.select_vms_widget.selected_list.count() > 0
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
|
|
||||||
import sys
|
import sys
|
||||||
import os
|
import os
|
||||||
|
import shutil
|
||||||
from PyQt4.QtCore import *
|
from PyQt4.QtCore import *
|
||||||
from PyQt4.QtGui import *
|
from PyQt4.QtGui import *
|
||||||
|
|
||||||
@ -67,6 +68,8 @@ class RestoreVMsWindow(Ui_Restore, QWizard):
|
|||||||
self.vms_to_restore = None
|
self.vms_to_restore = None
|
||||||
self.func_output = []
|
self.func_output = []
|
||||||
self.feedback_queue = Queue()
|
self.feedback_queue = Queue()
|
||||||
|
self.canceled = False
|
||||||
|
self.tmpdir_to_remove = None
|
||||||
|
|
||||||
self.excluded = {}
|
self.excluded = {}
|
||||||
|
|
||||||
@ -189,12 +192,19 @@ class RestoreVMsWindow(Ui_Restore, QWizard):
|
|||||||
print_callback=self.restore_output,
|
print_callback=self.restore_output,
|
||||||
error_callback=self.restore_error_output,
|
error_callback=self.restore_error_output,
|
||||||
progress_callback=self.update_progress_bar)
|
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:
|
except Exception as ex:
|
||||||
print "Exception:", ex
|
print "Exception:", ex
|
||||||
err_msg.append(str(ex))
|
err_msg.append(str(ex))
|
||||||
|
|
||||||
self.qvm_collection.unlock_db()
|
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))
|
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!"))
|
self.emit(SIGNAL("restore_progress(QString)"),'<b><font color="red">{0}</font></b>'.format("Finished with errors!"))
|
||||||
else:
|
else:
|
||||||
@ -229,7 +239,6 @@ class RestoreVMsWindow(Ui_Restore, QWizard):
|
|||||||
self.confirm_page.emit(SIGNAL("completeChanged()"))
|
self.confirm_page.emit(SIGNAL("completeChanged()"))
|
||||||
|
|
||||||
elif self.currentPage() is self.commit_page:
|
elif self.currentPage() is self.commit_page:
|
||||||
self.button(self.CancelButton).setDisabled(True)
|
|
||||||
self.button(self.FinishButton).setDisabled(True)
|
self.button(self.FinishButton).setDisabled(True)
|
||||||
|
|
||||||
self.thread_monitor = ThreadMonitor()
|
self.thread_monitor = ThreadMonitor()
|
||||||
@ -247,7 +256,18 @@ class RestoreVMsWindow(Ui_Restore, QWizard):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
if not self.thread_monitor.success:
|
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:
|
if self.dev_mount_path != None:
|
||||||
umount_device(self.dev_mount_path)
|
umount_device(self.dev_mount_path)
|
||||||
@ -257,6 +277,7 @@ class RestoreVMsWindow(Ui_Restore, QWizard):
|
|||||||
|
|
||||||
self.progress_bar.setValue(100)
|
self.progress_bar.setValue(100)
|
||||||
self.button(self.FinishButton).setEnabled(True)
|
self.button(self.FinishButton).setEnabled(True)
|
||||||
|
self.button(self.CancelButton).setEnabled(False)
|
||||||
|
|
||||||
signal.signal(signal.SIGCHLD, old_sigchld_handler)
|
signal.signal(signal.SIGCHLD, old_sigchld_handler)
|
||||||
|
|
||||||
@ -269,11 +290,19 @@ class RestoreVMsWindow(Ui_Restore, QWizard):
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
def reject(self):
|
def reject(self):
|
||||||
if self.dev_mount_path != None:
|
if self.currentPage() is self.commit_page:
|
||||||
umount_device(self.dev_mount_path)
|
if backup.backup_cancel():
|
||||||
detach_device(self, str(self.dev_combobox.itemData(
|
self.emit(SIGNAL("restore_progress(QString)"),'<font '
|
||||||
self.dev_combobox.currentIndex()).toString()))
|
'color="red">{'
|
||||||
self.done(0)
|
'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):
|
def has_selected_dir(self):
|
||||||
backup_location = unicode(self.dir_line_edit.text())
|
backup_location = unicode(self.dir_line_edit.text())
|
||||||
|
Loading…
Reference in New Issue
Block a user