|
@@ -36,61 +36,6 @@ from string import replace
|
|
|
|
|
|
path_re = re.compile(r"[a-zA-Z0-9/:.,_+=() -]*")
|
|
|
path_max_len = 512
|
|
|
-mount_for_backup_path = '/usr/libexec/qubes-manager/mount_for_backup.sh'
|
|
|
-
|
|
|
-def check_if_mounted(dev_path):
|
|
|
- mounts_file = open("/proc/mounts")
|
|
|
- for m in list(mounts_file):
|
|
|
- if m.startswith(dev_path):
|
|
|
- return m.split(" ")[1]
|
|
|
- return None
|
|
|
-
|
|
|
-
|
|
|
-def mount_device(dev_path):
|
|
|
- try:
|
|
|
- mount_dir_name = "backup" + replace(str(datetime.now()),' ', '-').split(".")[0]
|
|
|
- pmount_cmd = [mount_for_backup_path, dev_path, mount_dir_name]
|
|
|
- res = subprocess.check_call(pmount_cmd)
|
|
|
- except Exception as ex:
|
|
|
- QMessageBox.warning (None, "Error mounting selected device!", "<b>Could not mount {0}.</b><br><br>ERROR: {1}".format(dev_path, ex))
|
|
|
- return None
|
|
|
- if res == 0:
|
|
|
- dev_mount_path = "/media/"+mount_dir_name
|
|
|
- return dev_mount_path
|
|
|
- return None
|
|
|
-
|
|
|
-def umount_device(dev_mount_path):
|
|
|
- while True:
|
|
|
- try:
|
|
|
- pumount_cmd = ["/usr/bin/sudo", "/usr/bin/pumount", "--luks-force", dev_mount_path]
|
|
|
- res = subprocess.check_call(pumount_cmd)
|
|
|
- if res == 0:
|
|
|
- dev_mount_path = None
|
|
|
- return dev_mount_path
|
|
|
- except Exception as ex:
|
|
|
- title = "Error unmounting backup device!"
|
|
|
- text = "<b>Could not unmount {0}.</b><br>\
|
|
|
- <b>Please retry or unmount it manually using</b><br> pumount {0}.<br><br>\
|
|
|
- ERROR: {1}".format(dev_mount_path, ex)
|
|
|
- button = QMessageBox.warning (None, title, text, QMessageBox.Ok | QMessageBox.Retry, QMessageBox.Retry)
|
|
|
- if button == QMessageBox.Ok:
|
|
|
- return dev_mount_path
|
|
|
-
|
|
|
-def detach_device(dialog, dev_name):
|
|
|
- """ Detach device from dom0, if device was attached from some VM"""
|
|
|
- if not dev_name.startswith(dialog.vm.name+":"):
|
|
|
- with dialog.blk_manager.blk_lock:
|
|
|
- dialog.blk_manager.detach_device(dialog.vm, dev_name)
|
|
|
- dialog.blk_manager.update()
|
|
|
- else:
|
|
|
- # umount/LUKS remove do not trigger udev event on underlying device,
|
|
|
- # so trigger it manually - to publish back as available device
|
|
|
- subprocess.call(["/usr/bin/sudo", "/bin/udevadm", "trigger", "--action=change",
|
|
|
- "--subsystem-match=block",
|
|
|
- "--sysname-match=%s" % dev_name.split(":")[1]])
|
|
|
- with dialog.blk_manager.blk_lock:
|
|
|
- dialog.blk_manager.update()
|
|
|
-
|
|
|
|
|
|
def fill_appvms_list(dialog):
|
|
|
dialog.appvm_combobox.clear()
|
|
@@ -107,100 +52,10 @@ def fill_appvms_list(dialog):
|
|
|
if vm.is_running() and vm.qid != 0:
|
|
|
dialog.appvm_combobox.addItem(vm.name)
|
|
|
|
|
|
-def fill_devs_list(dialog):
|
|
|
- dialog.dev_combobox.clear()
|
|
|
- dialog.dev_combobox.addItem("None")
|
|
|
-
|
|
|
- dialog.blk_manager.blk_lock.acquire()
|
|
|
- for a in dialog.blk_manager.attached_devs:
|
|
|
- if dialog.blk_manager.attached_devs[a]['attached_to']['vm'] == dialog.vm.name :
|
|
|
- att = a + " " + unicode(dialog.blk_manager.attached_devs[a]['size']) + " " + dialog.blk_manager.attached_devs[a]['desc']
|
|
|
- dialog.dev_combobox.addItem(att, QVariant(a))
|
|
|
- for a in dialog.blk_manager.free_devs:
|
|
|
- att = a + " " + unicode(dialog.blk_manager.free_devs[a]['size']) + " " + dialog.blk_manager.free_devs[a]['desc']
|
|
|
- dialog.dev_combobox.addItem(att, QVariant(a))
|
|
|
- dialog.blk_manager.blk_lock.release()
|
|
|
-
|
|
|
- dialog.dev_combobox.setCurrentIndex(0) #current selected is null ""
|
|
|
- dialog.prev_dev_idx = 0
|
|
|
- dialog.dir_line_edit.clear()
|
|
|
- enable_dir_line_edit(dialog, True)
|
|
|
-
|
|
|
-
|
|
|
def enable_dir_line_edit(dialog, boolean):
|
|
|
dialog.dir_line_edit.setEnabled(boolean)
|
|
|
dialog.select_path_button.setEnabled(boolean)
|
|
|
|
|
|
-def update_device_appvm_enabled(dialog, idx):
|
|
|
- # Only one of those can be used
|
|
|
- dialog.dev_combobox.setEnabled(dialog.appvm_combobox.currentIndex() == 0)
|
|
|
- dialog.appvm_combobox.setEnabled(dialog.dev_combobox.currentIndex() == 0)
|
|
|
-
|
|
|
-def dev_combobox_activated(dialog, idx):
|
|
|
- if idx == dialog.prev_dev_idx: #nothing has changed
|
|
|
- return
|
|
|
- #there was a change
|
|
|
-
|
|
|
- dialog.dir_line_edit.setText("")
|
|
|
-
|
|
|
- # umount old device if any
|
|
|
- if dialog.dev_mount_path != None:
|
|
|
- dialog.dev_mount_path = umount_device(dialog.dev_mount_path)
|
|
|
- if dialog.dev_mount_path != None:
|
|
|
- dialog.dev_combobox.setCurrentIndex(dialog.prev_dev_idx)
|
|
|
- return
|
|
|
- else:
|
|
|
- detach_device(dialog,
|
|
|
- str(dialog.dev_combobox.itemData(dialog.prev_dev_idx).toString()))
|
|
|
-
|
|
|
- # then attach new one
|
|
|
- if dialog.dev_combobox.currentIndex() != 0: #An existing device chosen
|
|
|
- dev_name = str(dialog.dev_combobox.itemData(idx).toString())
|
|
|
-
|
|
|
- if dev_name.startswith(dialog.vm.name+":"):
|
|
|
- # originally attached to dom0
|
|
|
- dev_path = "/dev/"+dev_name.split(":")[1]
|
|
|
- else:
|
|
|
- try:
|
|
|
- with dialog.blk_manager.blk_lock:
|
|
|
- if dev_name in dialog.blk_manager.free_devs:
|
|
|
- #attach it to dom0, then treat it as an attached device
|
|
|
- dialog.blk_manager.attach_device(dialog.vm, dev_name)
|
|
|
- dialog.blk_manager.update()
|
|
|
-
|
|
|
- if dev_name in dialog.blk_manager.attached_devs: #is attached to dom0
|
|
|
- assert dialog.blk_manager.attached_devs[dev_name]['attached_to']['vm'] == dialog.vm.name
|
|
|
- dev_path = "/dev/" + dialog.blk_manager.attached_devs[dev_name]['attached_to']['frontend']
|
|
|
- else:
|
|
|
- raise QubesException("device not attached?!")
|
|
|
- except QubesException as ex:
|
|
|
- QMessageBox.warning (None, "Error attaching selected device!",
|
|
|
- "<b>Could not attach {0}.</b><br><br>ERROR: {1}".format(dev_name, ex))
|
|
|
- dialog.dev_combobox.setCurrentIndex(0) #if couldn't mount - set current device to "None"
|
|
|
- dialog.prev_dev_idx = 0
|
|
|
- return
|
|
|
-
|
|
|
- #check if device mounted
|
|
|
- dialog.dev_mount_path = check_if_mounted(dev_path)
|
|
|
- if dialog.dev_mount_path == None:
|
|
|
- dialog.dev_mount_path = mount_device(dev_path)
|
|
|
- if dialog.dev_mount_path == None:
|
|
|
- dialog.dev_combobox.setCurrentIndex(0) #if couldn't mount - set current device to "None"
|
|
|
- dialog.prev_dev_idx = 0
|
|
|
- detach_device(dialog,
|
|
|
- str(dialog.dev_combobox.itemData(idx).toString()))
|
|
|
- return
|
|
|
-
|
|
|
- dialog.prev_dev_idx = idx
|
|
|
-
|
|
|
- if hasattr(dialog, 'selected_vms'):
|
|
|
- # backup window
|
|
|
- if dialog.dev_mount_path != None:
|
|
|
- # Initialize path with root of mounted device
|
|
|
- dialog.dir_line_edit.setText(dialog.dev_mount_path)
|
|
|
- dialog.select_dir_page.emit(SIGNAL("completeChanged()"))
|
|
|
-
|
|
|
-
|
|
|
def get_path_for_vm(vm, service_name):
|
|
|
if not vm:
|
|
|
return None
|
|
@@ -234,8 +89,6 @@ def select_path_button_clicked(dialog, select_file = False):
|
|
|
if vm:
|
|
|
new_path = get_path_for_vm(vm, "qubes.SelectFile" if select_file
|
|
|
else "qubes.SelectDirectory")
|
|
|
- elif dialog.dev_mount_path != None:
|
|
|
- new_path = file_dialog_function(dialog, "Select backup location.", dialog.dev_mount_path)
|
|
|
else:
|
|
|
new_path = file_dialog_function(dialog, "Select backup location.",
|
|
|
backup_location if backup_location
|
|
@@ -258,5 +111,4 @@ def simulate_long_lasting_proces(period, progress_callback):
|
|
|
time.sleep(1)
|
|
|
|
|
|
progress_callback(100)
|
|
|
- return 0
|
|
|
-
|
|
|
+ return 0
|