backup_utils.py 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. #!/usr/bin/python2
  2. #
  3. # The Qubes OS Project, http://www.qubes-os.org
  4. #
  5. # Copyright (C) 2012 Agnieszka Kostrzewa <agnieszka.kostrzewa@gmail.com>
  6. #
  7. # This program is free software; you can redistribute it and/or
  8. # modify it under the terms of the GNU General Public License
  9. # as published by the Free Software Foundation; either version 2
  10. # of the License, or (at your option) any later version.
  11. #
  12. # This program is distributed in the hope that it will be useful,
  13. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. # GNU General Public License for more details.
  16. #
  17. # You should have received a copy of the GNU Lesser General Public License along
  18. # with this program; if not, see <http://www.gnu.org/licenses/>.
  19. #
  20. #
  21. import re
  22. from PyQt4 import QtGui # pylint: disable=import-error
  23. from PyQt4 import QtCore # pylint: disable=import-error
  24. import subprocess
  25. from . import utils
  26. import yaml
  27. path_re = re.compile(r"[a-zA-Z0-9/:.,_+=() -]*")
  28. path_max_len = 512
  29. def fill_appvms_list(dialog):
  30. """
  31. Helper function, designed to fill the destination vm combobox in both backup
  32. and restore GUI tools.
  33. :param dialog: QtGui.QWizard with a combobox called appvm_combobox
  34. """
  35. dialog.appvm_combobox.clear()
  36. dialog.appvm_combobox.addItem("dom0")
  37. dialog.appvm_combobox.setCurrentIndex(0) # current selected is null ""
  38. for vm in dialog.qubes_app.domains:
  39. if vm.features.get('internal', False) or vm.klass == 'TemplateVM':
  40. continue
  41. if vm.is_running() and vm.qid != 0:
  42. dialog.appvm_combobox.addItem(vm.name)
  43. def enable_dir_line_edit(dialog, boolean):
  44. dialog.dir_line_edit.setEnabled(boolean)
  45. dialog.select_path_button.setEnabled(boolean)
  46. def select_path_button_clicked(dialog, select_file=False, read_only=False):
  47. """
  48. Helper function that displays a file/directory selection wizard. Used by
  49. backup and restore GUI tools.
  50. :param dialog: QtGui.QWizard with a dir_line_edit text box that wants to
  51. receive a file/directory path and appvm_combobox with VM to use
  52. :param select_file: True: select file dialog; False: select directory
  53. dialog
  54. :param read_only: should the dir_line_edit be changed after selecting a file
  55. or directory
  56. :return:
  57. """
  58. backup_location = str(dialog.dir_line_edit.text())
  59. file_dialog = QtGui.QFileDialog()
  60. file_dialog.setReadOnly(True)
  61. new_path = None
  62. new_appvm = str(dialog.appvm_combobox.currentText())
  63. vm = dialog.qubes_app.domains[new_appvm]
  64. try:
  65. new_path = utils.get_path_from_vm(
  66. vm,
  67. "qubes.SelectFile" if select_file
  68. else "qubes.SelectDirectory")
  69. except subprocess.CalledProcessError:
  70. if not read_only:
  71. QtGui.QMessageBox.warning(
  72. None,
  73. dialog.tr("Nothing selected!"),
  74. dialog.tr("No file or directory selected."))
  75. else:
  76. return
  77. if new_path and not read_only:
  78. dialog.dir_line_edit.setText(new_path)
  79. if new_path and backup_location and not read_only:
  80. dialog.select_dir_page.emit(QtCore.SIGNAL("completeChanged()"))
  81. def get_profile_name(use_temp):
  82. backup_profile_name = 'qubes-manager-backup'
  83. temp_backup_profile_name = 'qubes-manager-backup-tmp'
  84. return temp_backup_profile_name if use_temp else backup_profile_name
  85. def get_profile_path(use_temp):
  86. path = '/etc/qubes/backup/' + get_profile_name(use_temp) + '.conf'
  87. return path
  88. def load_backup_profile(use_temp=False):
  89. path = get_profile_path(use_temp)
  90. with open(path) as profile_file:
  91. profile_data = yaml.safe_load(profile_file)
  92. return profile_data
  93. def write_backup_profile(args, use_temp=False):
  94. acceptable_fields = ['include', 'passphrase_text', 'compression',
  95. 'destination_vm', 'destination_path']
  96. profile_data = {key: value for key, value in args.items()
  97. if key in acceptable_fields}
  98. path = get_profile_path(use_temp)
  99. with open(path, 'w') as profile_file:
  100. yaml.safe_dump(profile_data, profile_file)