qube_manager.py 44 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191
  1. #!/usr/bin/python3
  2. #
  3. # The Qubes OS Project, http://www.qubes-os.org
  4. #
  5. # Copyright (C) 2012 Agnieszka Kostrzewa <agnieszka.kostrzewa@gmail.com>
  6. # Copyright (C) 2012 Marek Marczykowski-Górecki
  7. # <marmarek@invisiblethingslab.com>
  8. # Copyright (C) 2017 Wojtek Porczyk <woju@invisiblethingslab.com>
  9. #
  10. # This program is free software; you can redistribute it and/or
  11. # modify it under the terms of the GNU General Public License
  12. # as published by the Free Software Foundation; either version 2
  13. # of the License, or (at your option) any later version.
  14. #
  15. # This program is distributed in the hope that it will be useful,
  16. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. # GNU General Public License for more details.
  19. #
  20. # You should have received a copy of the GNU Lesser General Public License along
  21. # with this program; if not, see <http://www.gnu.org/licenses/>.
  22. #
  23. #
  24. import sys
  25. import os
  26. import os.path
  27. import subprocess
  28. import time
  29. from datetime import datetime, timedelta
  30. import traceback
  31. from qubesadmin import Qubes
  32. from qubesadmin import exc
  33. from PyQt4 import QtGui # pylint: disable=import-error
  34. from PyQt4 import QtCore # pylint: disable=import-error
  35. from . import ui_qubemanager # pylint: disable=no-name-in-module
  36. from . import thread_monitor
  37. from . import table_widgets
  38. from . import settings
  39. from . import global_settings
  40. from . import restore
  41. from . import backup
  42. from . import log_dialog
  43. import threading
  44. from qubesmanager.about import AboutDialog
  45. class SearchBox(QtGui.QLineEdit):
  46. def __init__(self, parent=None):
  47. super(SearchBox, self).__init__(parent)
  48. self.focusing = False
  49. def focusInEvent(self, e): # pylint: disable=invalid-name
  50. super(SearchBox, self).focusInEvent(e)
  51. self.selectAll()
  52. self.focusing = True
  53. def mousePressEvent(self, e): # pylint: disable=invalid-name
  54. super(SearchBox, self).mousePressEvent(e)
  55. if self.focusing:
  56. self.selectAll()
  57. self.focusing = False
  58. class VmRowInTable(object):
  59. # pylint: disable=too-few-public-methods
  60. def __init__(self, vm, row_no, table):
  61. self.vm = vm
  62. self.row_no = row_no
  63. # TODO: replace a various different widgets with a more generic
  64. # VmFeatureWidget or VMPropertyWidget
  65. table_widgets.row_height = VmManagerWindow.row_height
  66. table.setRowHeight(row_no, VmManagerWindow.row_height)
  67. self.type_widget = table_widgets.VmTypeWidget(vm)
  68. table.setCellWidget(row_no, VmManagerWindow.columns_indices['Type'],
  69. self.type_widget)
  70. table.setItem(row_no, VmManagerWindow.columns_indices['Type'],
  71. self.type_widget.table_item)
  72. self.label_widget = table_widgets.VmLabelWidget(vm)
  73. table.setCellWidget(row_no, VmManagerWindow.columns_indices['Label'],
  74. self.label_widget)
  75. table.setItem(row_no, VmManagerWindow.columns_indices['Label'],
  76. self.label_widget.table_item)
  77. self.name_widget = table_widgets.VmNameItem(vm)
  78. table.setItem(row_no, VmManagerWindow.columns_indices['Name'],
  79. self.name_widget)
  80. self.info_widget = table_widgets.VmInfoWidget(vm)
  81. table.setCellWidget(row_no, VmManagerWindow.columns_indices['State'],
  82. self.info_widget)
  83. table.setItem(row_no, VmManagerWindow.columns_indices['State'],
  84. self.info_widget.table_item)
  85. self.template_widget = table_widgets.VmTemplateItem(vm)
  86. table.setItem(row_no, VmManagerWindow.columns_indices['Template'],
  87. self.template_widget)
  88. self.netvm_widget = table_widgets.VmNetvmItem(vm)
  89. table.setItem(row_no, VmManagerWindow.columns_indices['NetVM'],
  90. self.netvm_widget)
  91. self.size_widget = table_widgets.VmSizeOnDiskItem(vm)
  92. table.setItem(row_no, VmManagerWindow.columns_indices['Size'],
  93. self.size_widget)
  94. self.internal_widget = table_widgets.VmInternalItem(vm)
  95. table.setItem(row_no, VmManagerWindow.columns_indices['Internal'],
  96. self.internal_widget)
  97. self.ip_widget = table_widgets.VmIPItem(vm)
  98. table.setItem(row_no, VmManagerWindow.columns_indices['IP'],
  99. self.ip_widget)
  100. self.include_in_backups_widget = \
  101. table_widgets.VmIncludeInBackupsItem(vm)
  102. table.setItem(row_no, VmManagerWindow.columns_indices[
  103. 'Backups'], self.include_in_backups_widget)
  104. self.last_backup_widget = table_widgets.VmLastBackupItem(vm)
  105. table.setItem(row_no, VmManagerWindow.columns_indices[
  106. 'Last backup'], self.last_backup_widget)
  107. def update(self, update_size_on_disk=False):
  108. """
  109. Update info in a single VM row
  110. :param update_size_on_disk: should disk utilization be updated? the
  111. widget will extract the data from VM object
  112. :return: None
  113. """
  114. self.info_widget.update_vm_state(self.vm)
  115. if update_size_on_disk:
  116. self.size_widget.update()
  117. vm_shutdown_timeout = 20000 # in msec
  118. vm_restart_check_timeout = 1000 # in msec
  119. class VmShutdownMonitor(QtCore.QObject):
  120. def __init__(self, vm, shutdown_time=vm_shutdown_timeout,
  121. check_time=vm_restart_check_timeout,
  122. and_restart=False, caller=None):
  123. QtCore.QObject.__init__(self)
  124. self.vm = vm
  125. self.shutdown_time = shutdown_time
  126. self.check_time = check_time
  127. self.and_restart = and_restart
  128. self.shutdown_started = datetime.now()
  129. self.caller = caller
  130. def restart_vm_if_needed(self):
  131. if self.and_restart and self.caller:
  132. self.caller.start_vm(self.vm)
  133. def check_again_later(self):
  134. # noinspection PyTypeChecker,PyCallByClass
  135. QtCore.QTimer.singleShot(self.check_time, self.check_if_vm_has_shutdown)
  136. def timeout_reached(self):
  137. actual = datetime.now() - self.shutdown_started
  138. allowed = timedelta(milliseconds=self.shutdown_time)
  139. return actual > allowed
  140. def check_if_vm_has_shutdown(self):
  141. vm = self.vm
  142. vm_is_running = vm.is_running()
  143. try:
  144. vm_start_time = datetime.fromtimestamp(float(vm.start_time))
  145. except AttributeError:
  146. vm_start_time = None
  147. if vm_is_running and vm_start_time \
  148. and vm_start_time < self.shutdown_started:
  149. if self.timeout_reached():
  150. reply = QtGui.QMessageBox.question(
  151. None, self.tr("Qube Shutdown"),
  152. self.tr(
  153. "The Qube <b>'{0}'</b> hasn't shutdown within the last "
  154. "{1} seconds, do you want to kill it?<br>").format(
  155. vm.name, self.shutdown_time / 1000),
  156. self.tr("Kill it!"),
  157. self.tr("Wait another {0} seconds...").format(
  158. self.shutdown_time / 1000))
  159. if reply == 0:
  160. vm.force_shutdown()
  161. self.restart_vm_if_needed()
  162. else:
  163. self.shutdown_started = datetime.now()
  164. self.check_again_later()
  165. else:
  166. self.check_again_later()
  167. else:
  168. if vm_is_running:
  169. # Due to unknown reasons, Xen sometimes reports that a domain
  170. # is running even though its start-up timestamp is not valid.
  171. # Make sure that "restart_vm_if_needed" is not called until
  172. # the domain has been completely shut down according to Xen.
  173. self.check_again_later()
  174. return
  175. self.restart_vm_if_needed()
  176. class VmManagerWindow(ui_qubemanager.Ui_VmManagerWindow, QtGui.QMainWindow):
  177. # pylint: disable=too-many-instance-attributes
  178. row_height = 30
  179. column_width = 200
  180. min_visible_rows = 10
  181. search = ""
  182. # suppress saving settings while initializing widgets
  183. settings_loaded = False
  184. columns_indices = {"Type": 0,
  185. "Label": 1,
  186. "Name": 2,
  187. "State": 3,
  188. "Template": 4,
  189. "NetVM": 5,
  190. "Size": 6,
  191. "Internal": 7,
  192. "IP": 8,
  193. "Backups": 9,
  194. "Last backup": 10,
  195. }
  196. def __init__(self, qubes_app, qt_app, parent=None):
  197. # pylint: disable=unused-argument
  198. super(VmManagerWindow, self).__init__()
  199. self.setupUi(self)
  200. self.manager_settings = QtCore.QSettings(self)
  201. self.qubes_app = qubes_app
  202. self.qt_app = qt_app
  203. self.searchbox = SearchBox()
  204. self.searchbox.setValidator(QtGui.QRegExpValidator(
  205. QtCore.QRegExp("[a-zA-Z0-9-]*", QtCore.Qt.CaseInsensitive), None))
  206. self.searchContainer.addWidget(self.searchbox)
  207. self.connect(self.table, QtCore.SIGNAL("itemSelectionChanged()"),
  208. self.table_selection_changed)
  209. self.table.setColumnWidth(0, self.column_width)
  210. self.sort_by_column = "Type"
  211. self.sort_order = QtCore.Qt.AscendingOrder
  212. self.vms_list = []
  213. self.vms_in_table = {}
  214. self.reload_table = False
  215. self.frame_width = 0
  216. self.frame_height = 0
  217. self.move(self.x(), 0)
  218. self.columns_actions = {
  219. self.columns_indices["Type"]: self.action_vm_type,
  220. self.columns_indices["Label"]: self.action_label,
  221. self.columns_indices["Name"]: self.action_name,
  222. self.columns_indices["State"]: self.action_state,
  223. self.columns_indices["Template"]: self.action_template,
  224. self.columns_indices["NetVM"]: self.action_netvm,
  225. self.columns_indices["Size"]: self.action_size_on_disk,
  226. self.columns_indices["Internal"]: self.action_internal,
  227. self.columns_indices["IP"]: self
  228. .action_ip, self.columns_indices["Backups"]: self
  229. .action_backups, self.columns_indices["Last backup"]: self
  230. .action_last_backup
  231. }
  232. self.visible_columns_count = len(self.columns_indices)
  233. self.table.setColumnWidth(self.columns_indices["State"], 80)
  234. self.table.setColumnWidth(self.columns_indices["Name"], 150)
  235. self.table.setColumnWidth(self.columns_indices["Label"], 40)
  236. self.table.setColumnWidth(self.columns_indices["Type"], 40)
  237. self.table.setColumnWidth(self.columns_indices["Size"], 100)
  238. self.table.setColumnWidth(self.columns_indices["Internal"], 60)
  239. self.table.setColumnWidth(self.columns_indices["IP"], 100)
  240. self.table.setColumnWidth(self.columns_indices["Backups"], 60)
  241. self.table.setColumnWidth(self.columns_indices["Last backup"], 90)
  242. self.table.horizontalHeader().setResizeMode(
  243. QtGui.QHeaderView.Interactive)
  244. self.table.horizontalHeader().setStretchLastSection(True)
  245. self.table.sortItems(self.columns_indices[self.sort_by_column],
  246. self.sort_order)
  247. self.context_menu = QtGui.QMenu(self)
  248. self.context_menu.addAction(self.action_settings)
  249. self.context_menu.addAction(self.action_editfwrules)
  250. self.context_menu.addAction(self.action_appmenus)
  251. self.context_menu.addAction(self.action_set_keyboard_layout)
  252. self.context_menu.addSeparator()
  253. self.context_menu.addAction(self.action_updatevm)
  254. self.context_menu.addAction(self.action_run_command_in_vm)
  255. self.context_menu.addAction(self.action_resumevm)
  256. self.context_menu.addAction(self.action_startvm_tools_install)
  257. self.context_menu.addAction(self.action_pausevm)
  258. self.context_menu.addAction(self.action_shutdownvm)
  259. self.context_menu.addAction(self.action_restartvm)
  260. self.context_menu.addAction(self.action_killvm)
  261. self.context_menu.addSeparator()
  262. self.context_menu.addAction(self.action_clonevm)
  263. self.context_menu.addAction(self.action_removevm)
  264. self.context_menu.addSeparator()
  265. self.context_menu.addMenu(self.logs_menu)
  266. self.context_menu.addSeparator()
  267. self.tools_context_menu = QtGui.QMenu(self)
  268. self.tools_context_menu.addAction(self.action_toolbar)
  269. self.tools_context_menu.addAction(self.action_menubar)
  270. self.connect(
  271. self.table.horizontalHeader(),
  272. QtCore.SIGNAL("sortIndicatorChanged(int, Qt::SortOrder)"),
  273. self.sort_indicator_changed)
  274. self.connect(self.table,
  275. QtCore.SIGNAL("customContextMenuRequested(const QPoint&)"),
  276. self.open_context_menu)
  277. self.connect(self.menubar,
  278. QtCore.SIGNAL("customContextMenuRequested(const QPoint&)"),
  279. lambda pos: self.open_tools_context_menu(self.menubar,
  280. pos))
  281. self.connect(self.toolbar,
  282. QtCore.SIGNAL("customContextMenuRequested(const QPoint&)"),
  283. lambda pos: self.open_tools_context_menu(self.toolbar,
  284. pos))
  285. self.connect(self.logs_menu, QtCore.SIGNAL("triggered(QAction *)"),
  286. self.show_log)
  287. self.connect(self.searchbox,
  288. QtCore.SIGNAL("textChanged(const QString&)"),
  289. self.do_search)
  290. self.table.setContentsMargins(0, 0, 0, 0)
  291. self.centralwidget.layout().setContentsMargins(0, 0, 0, 0)
  292. self.layout().setContentsMargins(0, 0, 0, 0)
  293. self.connect(self.action_menubar, QtCore.SIGNAL("toggled(bool)"),
  294. self.showhide_menubar)
  295. self.connect(self.action_toolbar, QtCore.SIGNAL("toggled(bool)"),
  296. self.showhide_toolbar)
  297. self.load_manager_settings()
  298. self.fill_table()
  299. self.counter = 0
  300. self.update_size_on_disk = False
  301. self.shutdown_monitor = {}
  302. def load_manager_settings(self):
  303. # visible columns
  304. self.visible_columns_count = 0
  305. for col in self.columns_indices:
  306. col_no = self.columns_indices[col]
  307. visible = self.manager_settings.value(
  308. 'columns/%s' % col,
  309. defaultValue="true")
  310. self.columns_actions[col_no].setChecked(visible == "true")
  311. self.visible_columns_count += 1
  312. self.sort_by_column = str(
  313. self.manager_settings.value("view/sort_column",
  314. defaultValue=self.sort_by_column))
  315. self.sort_order = QtCore.Qt.SortOrder(
  316. self.manager_settings.value("view/sort_order",
  317. defaultValue=self.sort_order))
  318. self.table.sortItems(self.columns_indices[self.sort_by_column],
  319. self.sort_order)
  320. if not self.manager_settings.value("view/menubar_visible",
  321. defaultValue=True):
  322. self.action_menubar.setChecked(False)
  323. if not self.manager_settings.value("view/toolbar_visible",
  324. defaultValue=True):
  325. self.action_toolbar.setChecked(False)
  326. self.settings_loaded = True
  327. def get_vms_list(self):
  328. return [vm for vm in self.qubes_app.domains]
  329. def fill_table(self):
  330. # save current selection
  331. row_index = self.table.currentRow()
  332. selected_qid = -1
  333. if row_index != -1:
  334. vm_item = self.table.item(row_index, self.columns_indices["Name"])
  335. if vm_item:
  336. selected_qid = vm_item.qid
  337. self.table.setSortingEnabled(False)
  338. self.table.clearContents()
  339. vms_list = self.get_vms_list()
  340. vms_in_table = {}
  341. row_no = 0
  342. for vm in vms_list:
  343. vm_row = VmRowInTable(vm, row_no, self.table)
  344. vms_in_table[vm.qid] = vm_row
  345. row_no += 1
  346. self.table.setRowCount(row_no)
  347. self.vms_list = vms_list
  348. self.vms_in_table = vms_in_table
  349. self.reload_table = False
  350. if selected_qid in vms_in_table.keys():
  351. self.table.setCurrentItem(
  352. self.vms_in_table[selected_qid].name_widget)
  353. self.table.setSortingEnabled(True)
  354. self.showhide_vms()
  355. def showhide_vms(self):
  356. if not self.search:
  357. for row_no in range(self.table.rowCount()):
  358. self.table.setRowHidden(row_no, False)
  359. else:
  360. for row_no in range(self.table.rowCount()):
  361. widget = self.table.cellWidget(row_no,
  362. self.columns_indices["State"])
  363. show = (self.search in widget.vm.name)
  364. self.table.setRowHidden(row_no, not show)
  365. @QtCore.pyqtSlot(str)
  366. def do_search(self, search):
  367. self.search = str(search)
  368. self.showhide_vms()
  369. # noinspection PyArgumentList
  370. @QtCore.pyqtSlot(name='on_action_search_triggered')
  371. def action_search_triggered(self):
  372. self.searchbox.setFocus()
  373. def mark_table_for_update(self):
  374. self.reload_table = True
  375. def update_table(self):
  376. self.fill_table()
  377. # TODO: instead of manually refreshing the entire table, use dbus events
  378. # reapply sorting
  379. if self.sort_by_column:
  380. self.table.sortByColumn(self.columns_indices[self.sort_by_column])
  381. self.table_selection_changed()
  382. # noinspection PyPep8Naming
  383. def sort_indicator_changed(self, column, order):
  384. self.sort_by_column = [name for name in self.columns_indices if
  385. self.columns_indices[name] == column][0]
  386. self.sort_order = order
  387. if self.settings_loaded:
  388. self.manager_settings.setValue('view/sort_column',
  389. self.sort_by_column)
  390. self.manager_settings.setValue('view/sort_order', self.sort_order)
  391. self.manager_settings.sync()
  392. def table_selection_changed(self):
  393. vm = self.get_selected_vm()
  394. if vm is not None and vm in self.qubes_app.domains:
  395. # TODO: add boot from device to menu and add windows tools there
  396. # Update available actions:
  397. self.action_settings.setEnabled(vm.klass != 'AdminVM')
  398. self.action_removevm.setEnabled(
  399. vm.klass != 'AdminVM' and not vm.is_running())
  400. self.action_clonevm.setEnabled(vm.klass != 'AdminVM')
  401. self.action_resumevm.setEnabled(
  402. not vm.is_running() or vm.get_power_state() == "Paused")
  403. self.action_pausevm.setEnabled(
  404. vm.is_running() and vm.get_power_state() != "Paused"
  405. and vm.klass != 'AdminVM')
  406. self.action_shutdownvm.setEnabled(
  407. vm.is_running() and vm.get_power_state() != "Paused"
  408. and vm.klass != 'AdminVM')
  409. self.action_restartvm.setEnabled(
  410. vm.is_running() and vm.get_power_state() != "Paused"
  411. and vm.klass != 'AdminVM' and vm.klass != 'DispVM')
  412. self.action_killvm.setEnabled(
  413. (vm.get_power_state() == "Paused" or vm.is_running())
  414. and vm.klass != 'AdminVM')
  415. self.action_appmenus.setEnabled(
  416. vm.klass != 'AdminVM' and vm.klass != 'DispVM'
  417. and not vm.features.get('internal', False))
  418. self.action_editfwrules.setEnabled(vm.klass != 'AdminVM')
  419. self.action_updatevm.setEnabled(getattr(vm, 'updateable', False)
  420. or vm.qid == 0)
  421. self.action_run_command_in_vm.setEnabled(
  422. not vm.get_power_state() == "Paused" and vm.qid != 0)
  423. self.action_set_keyboard_layout.setEnabled(
  424. vm.qid != 0 and
  425. vm.get_power_state() != "Paused" and vm.is_running())
  426. else:
  427. self.action_settings.setEnabled(False)
  428. self.action_removevm.setEnabled(False)
  429. self.action_clonevm.setEnabled(False)
  430. self.action_resumevm.setEnabled(False)
  431. self.action_pausevm.setEnabled(False)
  432. self.action_shutdownvm.setEnabled(False)
  433. self.action_restartvm.setEnabled(False)
  434. self.action_killvm.setEnabled(False)
  435. self.action_appmenus.setEnabled(False)
  436. self.action_editfwrules.setEnabled(False)
  437. self.action_updatevm.setEnabled(False)
  438. self.action_run_command_in_vm.setEnabled(False)
  439. self.action_set_keyboard_layout.setEnabled(False)
  440. # noinspection PyArgumentList
  441. @QtCore.pyqtSlot(name='on_action_createvm_triggered')
  442. def action_createvm_triggered(self): # pylint: disable=no-self-use
  443. subprocess.check_call('qubes-vm-create')
  444. def get_selected_vm(self):
  445. # vm selection relies on the VmInfo widget's value used
  446. # for sorting by VM name
  447. row_index = self.table.currentRow()
  448. if row_index != -1:
  449. vm_item = self.table.item(row_index, self.columns_indices["Name"])
  450. # here is possible race with update_table timer so check
  451. # if really got the item
  452. if vm_item is None:
  453. return None
  454. qid = vm_item.qid
  455. assert self.vms_in_table[qid] is not None
  456. vm = self.vms_in_table[qid].vm
  457. return vm
  458. else:
  459. return None
  460. # noinspection PyArgumentList
  461. @QtCore.pyqtSlot(name='on_action_removevm_triggered')
  462. def action_removevm_triggered(self):
  463. vm = self.get_selected_vm()
  464. if vm.klass == 'TemplateVM':
  465. dependent_vms = 0
  466. for single_vm in self.qubes_app.domains:
  467. if getattr(single_vm, 'template', None) == vm:
  468. dependent_vms += 1
  469. if dependent_vms > 0:
  470. QtGui.QMessageBox.warning(
  471. None, self.tr("Warning!"),
  472. self.tr("This Template Qube cannot be removed, "
  473. "because there is at least one Qube that is based "
  474. "on it.<br><small>If you want to remove this "
  475. "Template Qube and all the Qubes based on it, you "
  476. "should first remove each individual Qube that "
  477. "uses this template.</small>"))
  478. return
  479. (requested_name, ok) = QtGui.QInputDialog.getText(
  480. None, self.tr("Qube Removal Confirmation"),
  481. self.tr("Are you sure you want to remove the Qube <b>'{0}'</b>"
  482. "?<br> All data on this Qube's private storage will be "
  483. "lost!<br><br>Type the name of the Qube (<b>{1}</b>) below "
  484. "to confirm:").format(vm.name, vm.name))
  485. if not ok:
  486. # user clicked cancel
  487. return
  488. elif requested_name != vm.name:
  489. # name did not match
  490. QtGui.QMessageBox.warning(
  491. None,
  492. self.tr("Qube removal confirmation failed"),
  493. self.tr(
  494. "Entered name did not match! Not removing "
  495. "{0}.").format(vm.name))
  496. return
  497. else:
  498. # remove the VM
  499. t_monitor = thread_monitor.ThreadMonitor()
  500. thread = threading.Thread(target=self.do_remove_vm,
  501. args=(vm, self.qubes_app, t_monitor))
  502. thread.daemon = True
  503. thread.start()
  504. progress = QtGui.QProgressDialog(
  505. self.tr(
  506. "Removing Qube: <b>{0}</b>...").format(vm.name), "", 0, 0)
  507. progress.setCancelButton(None)
  508. progress.setModal(True)
  509. progress.show()
  510. while not t_monitor.is_finished():
  511. self.qt_app.processEvents()
  512. time.sleep(0.1)
  513. progress.hide()
  514. if t_monitor.success:
  515. pass
  516. else:
  517. QtGui.QMessageBox.warning(None, self.tr("Error removing Qube!"),
  518. self.tr("ERROR: {0}").format(
  519. t_monitor.error_msg))
  520. self.update_table()
  521. @staticmethod
  522. def do_remove_vm(vm, qubes_app, t_monitor):
  523. try:
  524. del qubes_app.domains[vm.name]
  525. except exc.QubesException as ex:
  526. t_monitor.set_error_msg(str(ex))
  527. t_monitor.set_finished()
  528. # noinspection PyArgumentList
  529. @QtCore.pyqtSlot(name='on_action_clonevm_triggered')
  530. def action_clonevm_triggered(self):
  531. vm = self.get_selected_vm()
  532. name_number = 1
  533. name_format = vm.name + '-clone-%d'
  534. while name_format % name_number in self.qubes_app.domains.keys():
  535. name_number += 1
  536. (clone_name, ok) = QtGui.QInputDialog.getText(
  537. self, self.tr('Qubes clone Qube'),
  538. self.tr('Enter name for Qube <b>{}</b> clone:').format(vm.name),
  539. text=(name_format % name_number))
  540. if not ok or clone_name == "":
  541. return
  542. t_monitor = thread_monitor.ThreadMonitor()
  543. thread = threading.Thread(target=self.do_clone_vm,
  544. args=(vm, self.qubes_app,
  545. clone_name, t_monitor))
  546. thread.daemon = True
  547. thread.start()
  548. progress = QtGui.QProgressDialog(
  549. self.tr("Cloning Qube <b>{0}</b> to <b>{1}</b>...").format(
  550. vm.name, clone_name), "", 0, 0)
  551. progress.setCancelButton(None)
  552. progress.setModal(True)
  553. progress.show()
  554. while not t_monitor.is_finished():
  555. self.qt_app.processEvents()
  556. time.sleep(0.2)
  557. progress.hide()
  558. if not t_monitor.success:
  559. QtGui.QMessageBox.warning(
  560. None,
  561. self.tr("Error while cloning Qube"),
  562. self.tr("Exception while cloning:<br>{0}").format(
  563. t_monitor.error_msg))
  564. self.update_table()
  565. @staticmethod
  566. def do_clone_vm(src_vm, qubes_app, dst_name, t_monitor):
  567. dst_vm = None
  568. try:
  569. dst_vm = qubes_app.clone_vm(src_vm, dst_name)
  570. except exc.QubesException as ex:
  571. t_monitor.set_error_msg(str(ex))
  572. if dst_vm:
  573. pass
  574. t_monitor.set_finished()
  575. # noinspection PyArgumentList
  576. @QtCore.pyqtSlot(name='on_action_resumevm_triggered')
  577. def action_resumevm_triggered(self):
  578. vm = self.get_selected_vm()
  579. if vm.get_power_state() in ["Paused", "Suspended"]:
  580. try:
  581. vm.unpause()
  582. except exc.QubesException as ex:
  583. QtGui.QMessageBox.warning(
  584. None, self.tr("Error unpausing Qube!"),
  585. self.tr("ERROR: {0}").format(ex))
  586. return
  587. self.start_vm(vm)
  588. self.update_table()
  589. def start_vm(self, vm):
  590. if vm.is_running():
  591. return
  592. t_monitor = thread_monitor.ThreadMonitor()
  593. thread = threading.Thread(target=self.do_start_vm,
  594. args=(vm, t_monitor))
  595. thread.daemon = True
  596. thread.start()
  597. while not t_monitor.is_finished():
  598. self.qt_app.processEvents()
  599. time.sleep(0.1)
  600. if not t_monitor.success:
  601. QtGui.QMessageBox.warning(
  602. None,
  603. self.tr("Error starting Qube!"),
  604. self.tr("ERROR: {0}").format(t_monitor.error_msg))
  605. self.update_table()
  606. @staticmethod
  607. def do_start_vm(vm, t_monitor):
  608. try:
  609. vm.start()
  610. except exc.QubesException as ex:
  611. t_monitor.set_error_msg(str(ex))
  612. t_monitor.set_finished()
  613. return
  614. t_monitor.set_finished()
  615. # noinspection PyArgumentList
  616. @QtCore.pyqtSlot(name='on_action_startvm_tools_install_triggered')
  617. # TODO: replace with boot from device
  618. def action_startvm_tools_install_triggered(self):
  619. # pylint: disable=invalid-name
  620. pass
  621. @QtCore.pyqtSlot(name='on_action_pausevm_triggered')
  622. def action_pausevm_triggered(self):
  623. vm = self.get_selected_vm()
  624. assert vm.is_running()
  625. try:
  626. vm.pause()
  627. self.update_table()
  628. except exc.QubesException as ex:
  629. QtGui.QMessageBox.warning(
  630. None,
  631. self.tr("Error pausing Qube!"),
  632. self.tr("ERROR: {0}").format(ex))
  633. return
  634. # noinspection PyArgumentList
  635. @QtCore.pyqtSlot(name='on_action_shutdownvm_triggered')
  636. def action_shutdownvm_triggered(self):
  637. vm = self.get_selected_vm()
  638. assert vm.is_running()
  639. reply = QtGui.QMessageBox.question(
  640. None, self.tr("Qube Shutdown Confirmation"),
  641. self.tr("Are you sure you want to power down the Qube"
  642. " <b>'{0}'</b>?<br><small>This will shutdown all the "
  643. "running applications within this Qube.</small>").format(
  644. vm.name), QtGui.QMessageBox.Yes | QtGui.QMessageBox.Cancel)
  645. self.qt_app.processEvents()
  646. if reply == QtGui.QMessageBox.Yes:
  647. self.shutdown_vm(vm)
  648. self.update_table()
  649. def shutdown_vm(self, vm, shutdown_time=vm_shutdown_timeout,
  650. check_time=vm_restart_check_timeout, and_restart=False):
  651. try:
  652. vm.shutdown()
  653. except exc.QubesException as ex:
  654. QtGui.QMessageBox.warning(
  655. None,
  656. self.tr("Error shutting down Qube!"),
  657. self.tr("ERROR: {0}").format(ex))
  658. return
  659. self.shutdown_monitor[vm.qid] = VmShutdownMonitor(vm, shutdown_time,
  660. check_time,
  661. and_restart, self)
  662. # noinspection PyCallByClass,PyTypeChecker
  663. QtCore.QTimer.singleShot(check_time, self.shutdown_monitor[
  664. vm.qid].check_if_vm_has_shutdown)
  665. # noinspection PyArgumentList
  666. @QtCore.pyqtSlot(name='on_action_restartvm_triggered')
  667. def action_restartvm_triggered(self):
  668. vm = self.get_selected_vm()
  669. assert vm.is_running()
  670. reply = QtGui.QMessageBox.question(
  671. None, self.tr("Qube Restart Confirmation"),
  672. self.tr("Are you sure you want to restart the Qube <b>'{0}'</b>?"
  673. "<br><small>This will shutdown all the running "
  674. "applications within this Qube.</small>").format(vm.name),
  675. QtGui.QMessageBox.Yes | QtGui.QMessageBox.Cancel)
  676. self.qt_app.processEvents()
  677. if reply == QtGui.QMessageBox.Yes:
  678. self.shutdown_vm(vm, and_restart=True)
  679. self.update_table()
  680. # noinspection PyArgumentList
  681. @QtCore.pyqtSlot(name='on_action_killvm_triggered')
  682. def action_killvm_triggered(self):
  683. vm = self.get_selected_vm()
  684. assert vm.is_running() or vm.is_paused()
  685. reply = QtGui.QMessageBox.question(
  686. None, self.tr("Qube Kill Confirmation"),
  687. self.tr("Are you sure you want to kill the Qube <b>'{0}'</b>?<br>"
  688. "<small>This will end <b>(not shutdown!)</b> all the "
  689. "running applications within this Qube.</small>").format(
  690. vm.name),
  691. QtGui.QMessageBox.Yes | QtGui.QMessageBox.Cancel,
  692. QtGui.QMessageBox.Cancel)
  693. self.qt_app.processEvents()
  694. if reply == QtGui.QMessageBox.Yes:
  695. try:
  696. vm.force_shutdown()
  697. except exc.QubesException as ex:
  698. QtGui.QMessageBox.critical(
  699. None, self.tr("Error while killing Qube!"),
  700. self.tr(
  701. "<b>An exception ocurred while killing {0}.</b><br>"
  702. "ERROR: {1}").format(vm.name, ex))
  703. return
  704. # noinspection PyArgumentList
  705. @QtCore.pyqtSlot(name='on_action_settings_triggered')
  706. def action_settings_triggered(self):
  707. vm = self.get_selected_vm()
  708. if vm:
  709. settings_window = settings.VMSettingsWindow(
  710. vm, self.qt_app, "basic")
  711. settings_window.exec_()
  712. self.update_table()
  713. # noinspection PyArgumentList
  714. @QtCore.pyqtSlot(name='on_action_appmenus_triggered')
  715. def action_appmenus_triggered(self):
  716. vm = self.get_selected_vm()
  717. if vm:
  718. settings_window = settings.VMSettingsWindow(
  719. vm, self.qt_app, "applications")
  720. settings_window.exec_()
  721. # noinspection PyArgumentList
  722. @QtCore.pyqtSlot(name='on_action_refresh_list_triggered')
  723. def action_refresh_list_triggered(self):
  724. self.update_table()
  725. # noinspection PyArgumentList
  726. @QtCore.pyqtSlot(name='on_action_updatevm_triggered')
  727. def action_updatevm_triggered(self):
  728. vm = self.get_selected_vm()
  729. if not vm.is_running():
  730. reply = QtGui.QMessageBox.question(
  731. None, self.tr("Qube Update Confirmation"),
  732. self.tr(
  733. "<b>{0}</b><br>The Qube has to be running to be updated."
  734. "<br>Do you want to start it?<br>").format(vm.name),
  735. QtGui.QMessageBox.Yes | QtGui.QMessageBox.Cancel)
  736. if reply != QtGui.QMessageBox.Yes:
  737. return
  738. self.qt_app.processEvents()
  739. t_monitor = thread_monitor.ThreadMonitor()
  740. thread = threading.Thread(target=self.do_update_vm,
  741. args=(vm, t_monitor))
  742. thread.daemon = True
  743. thread.start()
  744. progress = QtGui.QProgressDialog(
  745. self.tr(
  746. "<b>{0}</b><br>Please wait for the updater to "
  747. "launch...").format(vm.name), "", 0, 0)
  748. progress.setCancelButton(None)
  749. progress.setModal(True)
  750. progress.show()
  751. while not t_monitor.is_finished():
  752. self.qt_app.processEvents()
  753. time.sleep(0.2)
  754. progress.hide()
  755. if vm.qid != 0:
  756. if not t_monitor.success:
  757. QtGui.QMessageBox.warning(
  758. None,
  759. self.tr("Error on Qube update!"),
  760. self.tr("ERROR: {0}").format(t_monitor.error_msg))
  761. self.update_table()
  762. @staticmethod
  763. def do_update_vm(vm, t_monitor):
  764. try:
  765. if vm.qid == 0:
  766. subprocess.check_call(
  767. ["/usr/bin/qubes-dom0-update", "--clean", "--gui"])
  768. else:
  769. if not vm.is_running():
  770. vm.start()
  771. vm.run_service("qubes.InstallUpdatesGUI",
  772. user="root", wait=False)
  773. except (ChildProcessError, exc.QubesException) as ex:
  774. t_monitor.set_error_msg(str(ex))
  775. t_monitor.set_finished()
  776. return
  777. t_monitor.set_finished()
  778. # noinspection PyArgumentList
  779. @QtCore.pyqtSlot(name='on_action_run_command_in_vm_triggered')
  780. def action_run_command_in_vm_triggered(self):
  781. # pylint: disable=invalid-name
  782. vm = self.get_selected_vm()
  783. (command_to_run, ok) = QtGui.QInputDialog.getText(
  784. self, self.tr('Qubes command entry'),
  785. self.tr('Run command in <b>{}</b>:').format(vm.name))
  786. if not ok or command_to_run == "":
  787. return
  788. t_monitor = thread_monitor.ThreadMonitor()
  789. thread = threading.Thread(target=self.do_run_command_in_vm, args=(
  790. vm, command_to_run, t_monitor))
  791. thread.daemon = True
  792. thread.start()
  793. while not t_monitor.is_finished():
  794. self.qt_app.processEvents()
  795. time.sleep(0.2)
  796. if not t_monitor.success:
  797. QtGui.QMessageBox.warning(
  798. None, self.tr("Error while running command"),
  799. self.tr("Exception while running command:<br>{0}").format(
  800. t_monitor.error_msg))
  801. @staticmethod
  802. def do_run_command_in_vm(vm, command_to_run, t_monitor):
  803. try:
  804. vm.run(command_to_run)
  805. except (ChildProcessError, exc.QubesException) as ex:
  806. t_monitor.set_error_msg(str(ex))
  807. t_monitor.set_finished()
  808. # noinspection PyArgumentList
  809. @QtCore.pyqtSlot(name='on_action_set_keyboard_layout_triggered')
  810. def action_set_keyboard_layout_triggered(self):
  811. # pylint: disable=invalid-name
  812. vm = self.get_selected_vm()
  813. vm.run('qubes-change-keyboard-layout')
  814. # noinspection PyArgumentList
  815. @QtCore.pyqtSlot(name='on_action_editfwrules_triggered')
  816. def action_editfwrules_triggered(self):
  817. vm = self.get_selected_vm()
  818. settings_window = settings.VMSettingsWindow(vm, self.qt_app, "firewall")
  819. settings_window.exec_()
  820. # noinspection PyArgumentList
  821. @QtCore.pyqtSlot(name='on_action_global_settings_triggered')
  822. def action_global_settings_triggered(self): # pylint: disable=invalid-name
  823. global_settings_window = global_settings.GlobalSettingsWindow(
  824. self.qt_app,
  825. self.qubes_app)
  826. global_settings_window.exec_()
  827. # noinspection PyArgumentList
  828. @QtCore.pyqtSlot(name='on_action_show_network_triggered')
  829. def action_show_network_triggered(self):
  830. pass
  831. # TODO: revive for 4.1
  832. # network_notes_dialog = NetworkNotesDialog()
  833. # network_notes_dialog.exec_()
  834. # noinspection PyArgumentList
  835. @QtCore.pyqtSlot(name='on_action_restore_triggered')
  836. def action_restore_triggered(self):
  837. restore_window = restore.RestoreVMsWindow(self.qt_app, self.qubes_app)
  838. restore_window.exec_()
  839. # noinspection PyArgumentList
  840. @QtCore.pyqtSlot(name='on_action_backup_triggered')
  841. def action_backup_triggered(self):
  842. backup_window = backup.BackupVMsWindow(self.qt_app, self.qubes_app)
  843. backup_window.exec_()
  844. def showhide_menubar(self, checked):
  845. self.menubar.setVisible(checked)
  846. if not checked:
  847. self.context_menu.addAction(self.action_menubar)
  848. else:
  849. self.context_menu.removeAction(self.action_menubar)
  850. if self.settings_loaded:
  851. self.manager_settings.setValue('view/menubar_visible', checked)
  852. self.manager_settings.sync()
  853. def showhide_toolbar(self, checked):
  854. self.toolbar.setVisible(checked)
  855. if not checked:
  856. self.context_menu.addAction(self.action_toolbar)
  857. else:
  858. self.context_menu.removeAction(self.action_toolbar)
  859. if self.settings_loaded:
  860. self.manager_settings.setValue('view/toolbar_visible', checked)
  861. self.manager_settings.sync()
  862. def showhide_column(self, col_num, show):
  863. self.table.setColumnHidden(col_num, not show)
  864. val = 1 if show else -1
  865. self.visible_columns_count += val
  866. if self.visible_columns_count == 1:
  867. # disable hiding the last one
  868. for col in self.columns_actions:
  869. if self.columns_actions[col].isChecked():
  870. self.columns_actions[col].setEnabled(False)
  871. break
  872. elif self.visible_columns_count == 2 and val == 1:
  873. # enable hiding previously disabled column
  874. for col in self.columns_actions:
  875. if not self.columns_actions[col].isEnabled():
  876. self.columns_actions[col].setEnabled(True)
  877. break
  878. if self.settings_loaded:
  879. col_name = [name for name in self.columns_indices if
  880. self.columns_indices[name] == col_num][0]
  881. self.manager_settings.setValue('columns/%s' % col_name, show)
  882. self.manager_settings.sync()
  883. def on_action_vm_type_toggled(self, checked):
  884. self.showhide_column(self.columns_indices['Type'], checked)
  885. def on_action_label_toggled(self, checked):
  886. self.showhide_column(self.columns_indices['Label'], checked)
  887. def on_action_name_toggled(self, checked):
  888. self.showhide_column(self.columns_indices['Name'], checked)
  889. def on_action_state_toggled(self, checked):
  890. self.showhide_column(self.columns_indices['State'], checked)
  891. def on_action_internal_toggled(self, checked):
  892. self.showhide_column(self.columns_indices['Internal'], checked)
  893. def on_action_ip_toggled(self, checked):
  894. self.showhide_column(self.columns_indices['IP'], checked)
  895. def on_action_backups_toggled(self, checked):
  896. self.showhide_column(self.columns_indices['Backups'], checked)
  897. def on_action_last_backup_toggled(self, checked):
  898. self.showhide_column(self.columns_indices['Last backup'], checked)
  899. def on_action_template_toggled(self, checked):
  900. self.showhide_column(self.columns_indices['Template'], checked)
  901. def on_action_netvm_toggled(self, checked):
  902. self.showhide_column(self.columns_indices['NetVM'], checked)
  903. def on_action_size_on_disk_toggled(self, checked):
  904. self.showhide_column(self.columns_indices['Size'], checked)
  905. # noinspection PyArgumentList
  906. @QtCore.pyqtSlot(name='on_action_about_qubes_triggered')
  907. def action_about_qubes_triggered(self): # pylint: disable=no-self-use
  908. about = AboutDialog()
  909. about.exec_()
  910. def createPopupMenu(self): # pylint: disable=invalid-name
  911. menu = QtGui.QMenu()
  912. menu.addAction(self.action_toolbar)
  913. menu.addAction(self.action_menubar)
  914. return menu
  915. def open_tools_context_menu(self, widget, point):
  916. self.tools_context_menu.exec_(widget.mapToGlobal(point))
  917. @QtCore.pyqtSlot('const QPoint&')
  918. def open_context_menu(self, point):
  919. vm = self.get_selected_vm()
  920. # logs menu
  921. self.logs_menu.clear()
  922. if vm.qid == 0:
  923. logfiles = ["/var/log/xen/console/hypervisor.log"]
  924. else:
  925. logfiles = [
  926. "/var/log/xen/console/guest-" + vm.name + ".log",
  927. "/var/log/xen/console/guest-" + vm.name + "-dm.log",
  928. "/var/log/qubes/guid." + vm.name + ".log",
  929. "/var/log/qubes/qrexec." + vm.name + ".log",
  930. ]
  931. menu_empty = True
  932. for logfile in logfiles:
  933. if os.path.exists(logfile):
  934. action = self.logs_menu.addAction(QtGui.QIcon(":/log.png"),
  935. logfile)
  936. action.setData(logfile)
  937. menu_empty = False
  938. self.logs_menu.setEnabled(not menu_empty)
  939. self.context_menu.exec_(self.table.mapToGlobal(point))
  940. @QtCore.pyqtSlot('QAction *')
  941. def show_log(self, action):
  942. log = str(action.data())
  943. log_dlg = log_dialog.LogDialog(self.qt_app, log)
  944. log_dlg.exec_()
  945. # Bases on the original code by:
  946. # Copyright (c) 2002-2007 Pascal Varet <p.varet@gmail.com>
  947. def handle_exception(exc_type, exc_value, exc_traceback):
  948. filename, line, dummy, dummy = traceback.extract_tb(exc_traceback).pop()
  949. filename = os.path.basename(filename)
  950. error = "%s: %s" % (exc_type.__name__, exc_value)
  951. strace = ""
  952. stacktrace = traceback.extract_tb(exc_traceback)
  953. while stacktrace:
  954. (filename, line, func, txt) = stacktrace.pop()
  955. strace += "----\n"
  956. strace += "line: %s\n" % txt
  957. strace += "func: %s\n" % func
  958. strace += "line no.: %d\n" % line
  959. strace += "file: %s\n" % filename
  960. msg_box = QtGui.QMessageBox()
  961. msg_box.setDetailedText(strace)
  962. msg_box.setIcon(QtGui.QMessageBox.Critical)
  963. msg_box.setWindowTitle("Houston, we have a problem...")
  964. msg_box.setText("Whoops. A critical error has occured. "
  965. "This is most likely a bug in Qubes Manager.<br><br>"
  966. "<b><i>%s</i></b>" % error +
  967. "<br/>at line <b>%d</b><br/>of file %s.<br/><br/>"
  968. % (line, filename))
  969. msg_box.exec_()
  970. def main():
  971. qt_app = QtGui.QApplication(sys.argv)
  972. qt_app.setOrganizationName("The Qubes Project")
  973. qt_app.setOrganizationDomain("http://qubes-os.org")
  974. qt_app.setApplicationName("Qube Manager")
  975. qt_app.setWindowIcon(QtGui.QIcon.fromTheme("qubes-manager"))
  976. sys.excepthook = handle_exception
  977. qubes_app = Qubes()
  978. manager_window = VmManagerWindow(qubes_app, qt_app)
  979. manager_window.show()
  980. manager_window.update_table()
  981. qt_app.exec_()
  982. if __name__ == "__main__":
  983. main()