qube_manager.py 51 KB

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