bootfromdevice.py 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  1. #!/usr/bin/python3
  2. #
  3. # The Qubes OS Project, http://www.qubes-os.org
  4. #
  5. # This program is free software; you can redistribute it and/or
  6. # modify it under the terms of the GNU General Public License
  7. # as published by the Free Software Foundation; either version 2
  8. # of the License, or (at your option) any later version.
  9. #
  10. # This program is distributed in the hope that it will be useful,
  11. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. # GNU General Public License for more details.
  14. #
  15. # You should have received a copy of the GNU Lesser General Public License along
  16. # with this program; if not, see <http://www.gnu.org/licenses/>.
  17. #
  18. #
  19. import functools
  20. import subprocess
  21. from . import utils
  22. from . import ui_bootfromdevice # pylint: disable=no-name-in-module
  23. from PyQt5 import QtWidgets, QtGui # pylint: disable=import-error
  24. from qubesadmin import tools
  25. from qubesadmin import exc
  26. class VMBootFromDeviceWindow(ui_bootfromdevice.Ui_BootDialog,
  27. QtWidgets.QDialog):
  28. def __init__(self, vm, qapp, qubesapp=None, parent=None, new_vm = False):
  29. super().__init__(parent)
  30. self.vm = vm
  31. self.qapp = qapp
  32. self.qubesapp = qubesapp
  33. self.cdrom_location = None
  34. self.new_vm = new_vm
  35. self.setupUi(self)
  36. self.setWindowTitle(
  37. self.tr("Boot {vm} from device").format(vm=self.vm))
  38. self.buttonBox.accepted.connect(self.save_and_apply)
  39. # populate buttons and such
  40. self.__init_buttons__()
  41. # warn user if the VM is currently running
  42. self.__warn_if_running__()
  43. def setup_application(self):
  44. self.qapp.setApplicationName(self.tr("Boot Qube From Device"))
  45. self.qapp.setWindowIcon(QtGui.QIcon.fromTheme("qubes-manager"))
  46. def save_and_apply(self):
  47. if self.blockDeviceRadioButton.isChecked():
  48. self.cdrom_location = self.blockDeviceComboBox.currentText()
  49. elif self.fileRadioButton.isChecked():
  50. self.cdrom_location = str(self.fileVM.currentData()) + \
  51. ":" + self.pathText.text()
  52. else:
  53. QtWidgets.QMessageBox.warning(
  54. self,
  55. self.tr("ERROR!"),
  56. self.tr("No file or block device selected; please select one."))
  57. return
  58. # warn user if the VM is currently running
  59. self.__warn_if_running__()
  60. self.accept()
  61. def __warn_if_running__(self):
  62. if self.new_vm:
  63. return
  64. try:
  65. if self.qubesapp.domains[self.vm].is_running():
  66. QtWidgets.QMessageBox.warning(
  67. self,
  68. self.tr("Warning!"),
  69. self.tr("Qube must be turned off before booting it from "
  70. "device. Please turn off the qube."))
  71. except exc.QubesDaemonAccessError:
  72. QtWidgets.QMessageBox.warning(
  73. self,
  74. self.tr("Warning!"),
  75. self.tr("Insufficient permissions to determine if qube is "
  76. "running. It must be turned off before booting it from "
  77. "device."))
  78. def __init_buttons__(self):
  79. self.fileVM.setEnabled(False)
  80. self.selectFileButton.setEnabled(False)
  81. self.blockDeviceComboBox.setEnabled(False)
  82. self.blockDeviceRadioButton.clicked.connect(self.radio_button_clicked)
  83. self.fileRadioButton.clicked.connect(self.radio_button_clicked)
  84. self.selectFileButton.clicked.connect(self.select_file_dialog)
  85. utils.initialize_widget_with_vms(
  86. widget=self.fileVM,
  87. qubes_app=self.qubesapp,
  88. allow_internal=True
  89. )
  90. device_choice = []
  91. for domain in self.qubesapp.domains:
  92. try:
  93. for device in domain.devices["block"]:
  94. device_choice.append((str(device), device))
  95. except exc.QubesDaemonAccessError:
  96. # insufficient permissions
  97. pass
  98. if device_choice:
  99. utils.initialize_widget(
  100. widget=self.blockDeviceComboBox,
  101. choices=device_choice,
  102. selected_value=device_choice[0][1],
  103. add_current_label=False
  104. )
  105. else:
  106. self.blockDeviceRadioButton.setEnabled(False)
  107. self.blockDeviceComboBox.setEnabled(False)
  108. self.blockDeviceComboBox.addItem("no block devices found!")
  109. self.blockDeviceComboBox.setCurrentIndex(0)
  110. def radio_button_clicked(self):
  111. self.blockDeviceComboBox.setEnabled(
  112. self.blockDeviceRadioButton.isChecked())
  113. self.fileVM.setEnabled(self.fileRadioButton.isChecked())
  114. self.selectFileButton.setEnabled(self.fileRadioButton.isChecked())
  115. self.pathText.setEnabled(self.fileRadioButton.isChecked())
  116. def select_file_dialog(self):
  117. backend_vm = self.fileVM.currentData()
  118. error_occurred = False
  119. try:
  120. new_path = utils.get_path_from_vm(backend_vm, "qubes.SelectFile")
  121. except subprocess.CalledProcessError as ex:
  122. if ex.returncode != 1:
  123. # Error other than 'user did not select a file'
  124. error_occurred = True
  125. new_path = None
  126. except Exception: # pylint: disable=broad-except
  127. error_occurred = True
  128. new_path = None
  129. if error_occurred:
  130. QtWidgets.QMessageBox.warning(
  131. None,
  132. self.tr("Failed to display file selection dialog"),
  133. self.tr("Check if the qube {0} can be started and has a file"
  134. " manager installed.").format(backend_vm)
  135. )
  136. if new_path:
  137. self.pathText.setText(new_path)
  138. parser = tools.QubesArgumentParser(vmname_nargs=1)
  139. def main(args=None):
  140. args = parser.parse_args(args)
  141. vm = args.domains.pop()
  142. utils.run_synchronous(functools.partial(VMBootFromDeviceWindow, vm))
  143. if __name__ == "__main__":
  144. main()