backup_utils.py 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  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. import socket
  23. from PyQt4 import QtGui # pylint: disable=import-error
  24. from PyQt4 import QtCore # pylint: disable=import-error
  25. import subprocess
  26. from . import utils
  27. import yaml
  28. path_re = re.compile(r"[a-zA-Z0-9/:.,_+=() -]*")
  29. path_max_len = 512
  30. def fill_appvms_list(dialog):
  31. """
  32. Helper function, designed to fill the destination vm combobox in both backup
  33. and restore GUI tools.
  34. :param dialog: QtGui.QWizard with a combobox called appvm_combobox
  35. """
  36. dialog.appvm_combobox.clear()
  37. dialog.appvm_combobox.addItem("dom0")
  38. dialog.appvm_combobox.setCurrentIndex(0) # current selected is null ""
  39. for vm in dialog.qubes_app.domains:
  40. if vm.features.get('internal', False) or vm.klass == 'TemplateVM':
  41. continue
  42. if vm.is_running() and vm.qid != 0:
  43. dialog.appvm_combobox.addItem(vm.name)
  44. def enable_dir_line_edit(dialog, boolean):
  45. dialog.dir_line_edit.setEnabled(boolean)
  46. dialog.select_path_button.setEnabled(boolean)
  47. def select_path_button_clicked(dialog, select_file=False, read_only=False):
  48. """
  49. Helper function that displays a file/directory selection wizard. Used by
  50. backup and restore GUI tools.
  51. :param dialog: QtGui.QWizard with a dir_line_edit text box that wants to
  52. receive a file/directory path and appvm_combobox with VM to use
  53. :param select_file: True: select file dialog; False: select directory
  54. dialog
  55. :param read_only: should the dir_line_edit be changed after selecting a file
  56. or directory
  57. :return:
  58. """
  59. backup_location = str(dialog.dir_line_edit.text())
  60. new_path = None
  61. new_appvm = str(dialog.appvm_combobox.currentText())
  62. vm = dialog.qubes_app.domains[new_appvm]
  63. try:
  64. if vm.name == socket.gethostname():
  65. file_dialog = QtGui.QFileDialog()
  66. file_dialog.setReadOnly(True)
  67. if select_file:
  68. file_dialog_function = file_dialog.getOpenFileName
  69. else:
  70. file_dialog_function = file_dialog.getExistingDirectory
  71. new_path = file_dialog_function(
  72. dialog,
  73. dialog.tr("Select backup location."),
  74. backup_location if backup_location else '/')
  75. else:
  76. new_path = utils.get_path_from_vm(
  77. vm,
  78. "qubes.SelectFile" if select_file
  79. else "qubes.SelectDirectory")
  80. except subprocess.CalledProcessError:
  81. if not read_only:
  82. QtGui.QMessageBox.warning(
  83. None,
  84. dialog.tr("Nothing selected!"),
  85. dialog.tr("No file or directory selected."))
  86. else:
  87. return
  88. if new_path and not read_only:
  89. dialog.dir_line_edit.setText(new_path)
  90. if new_path and backup_location and not read_only:
  91. dialog.select_dir_page.emit(QtCore.SIGNAL("completeChanged()"))
  92. def get_profile_name(use_temp):
  93. backup_profile_name = 'qubes-manager-backup'
  94. temp_backup_profile_name = 'qubes-manager-backup-tmp'
  95. return temp_backup_profile_name if use_temp else backup_profile_name
  96. def get_profile_path(use_temp):
  97. path = '/etc/qubes/backup/' + get_profile_name(use_temp) + '.conf'
  98. return path
  99. def load_backup_profile(use_temp=False):
  100. path = get_profile_path(use_temp)
  101. with open(path) as profile_file:
  102. profile_data = yaml.safe_load(profile_file)
  103. return profile_data
  104. def write_backup_profile(args, use_temp=False):
  105. acceptable_fields = ['include', 'passphrase_text', 'compression',
  106. 'destination_vm', 'destination_path']
  107. profile_data = {key: value for key, value in args.items()
  108. if key in acceptable_fields}
  109. path = get_profile_path(use_temp)
  110. with open(path, 'w') as profile_file:
  111. yaml.safe_dump(profile_data, profile_file)