qube_manager.py 50 KB

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