global_settings.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445
  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 <marmarek@mimuw.edu.pl>
  7. #
  8. # This program is free software; you can redistribute it and/or
  9. # modify it under the terms of the GNU General Public License
  10. # as published by the Free Software Foundation; either version 2
  11. # of the License, or (at your option) any later version.
  12. #
  13. # This program is distributed in the hope that it will be useful,
  14. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. # GNU General Public License for more details.
  17. #
  18. # You should have received a copy of the GNU Lesser General Public License along
  19. # with this program; if not, see <http://www.gnu.org/licenses/>.
  20. #
  21. #
  22. import sys
  23. import os
  24. import os.path
  25. import traceback
  26. import subprocess
  27. from PyQt5 import QtWidgets # pylint: disable=import-error
  28. from qubesadmin import Qubes
  29. from qubesadmin.utils import parse_size
  30. from . import ui_globalsettingsdlg # pylint: disable=no-name-in-module
  31. from . import utils
  32. from configparser import ConfigParser
  33. qmemman_config_path = '/etc/qubes/qmemman.conf'
  34. def _run_qrexec_repo(service, arg=''):
  35. # Fake up a "qrexec call" to dom0 because dom0 can't qrexec to itself yet
  36. cmd = '/etc/qubes-rpc/' + service
  37. p = subprocess.run(
  38. ['sudo', cmd, arg],
  39. stdout=subprocess.PIPE,
  40. stderr=subprocess.PIPE
  41. )
  42. if p.stderr:
  43. raise RuntimeError('qrexec call stderr was not empty',
  44. {'stderr': p.stderr.decode('utf-8')})
  45. if p.returncode != 0:
  46. raise RuntimeError('qrexec call exited with non-zero return code',
  47. {'returncode': p.returncode})
  48. return p.stdout.decode('utf-8')
  49. # pylint: disable=too-many-instance-attributes
  50. class GlobalSettingsWindow(ui_globalsettingsdlg.Ui_GlobalSettings,
  51. QtWidgets.QDialog):
  52. def __init__(self, app, qvm_collection, parent=None):
  53. super(GlobalSettingsWindow, self).__init__(parent)
  54. self.app = app
  55. self.qvm_collection = qvm_collection
  56. self.setupUi(self)
  57. self.buttonBox.accepted.connect(self.save_and_apply)
  58. self.buttonBox.rejected.connect(self.reject)
  59. self.__init_system_defaults__()
  60. self.__init_kernel_defaults__()
  61. self.__init_mem_defaults__()
  62. self.__init_updates__()
  63. def __init_system_defaults__(self):
  64. # set up updatevm choice
  65. self.update_vm_vmlist, self.update_vm_idx = utils.prepare_vm_choice(
  66. self.update_vm_combo, self.qvm_collection, 'updatevm',
  67. None, allow_none=True,
  68. filter_function=(lambda vm: vm.klass != 'TemplateVM')
  69. )
  70. # set up clockvm choice
  71. self.clock_vm_vmlist, self.clock_vm_idx = utils.prepare_vm_choice(
  72. self.clock_vm_combo, self.qvm_collection, 'clockvm',
  73. None, allow_none=True,
  74. filter_function=(lambda vm: vm.klass != 'TemplateVM')
  75. )
  76. # set up default netvm
  77. self.default_netvm_vmlist, self.default_netvm_idx = \
  78. utils.prepare_vm_choice(
  79. self.default_netvm_combo,
  80. self.qvm_collection, 'default_netvm',
  81. None,
  82. filter_function=(lambda vm: vm.provides_network),
  83. allow_none=True)
  84. # default template
  85. self.default_template_vmlist, self.default_template_idx = \
  86. utils.prepare_vm_choice(
  87. self.default_template_combo,
  88. self.qvm_collection, 'default_template',
  89. None,
  90. filter_function=(lambda vm: vm.klass == 'TemplateVM'),
  91. allow_none=True
  92. )
  93. # default dispvm
  94. self.default_dispvm_vmlist, self.default_dispvm_idx = \
  95. utils.prepare_vm_choice(
  96. self.default_dispvm_combo,
  97. self.qvm_collection, 'default_dispvm',
  98. None,
  99. (lambda vm: getattr(vm, 'template_for_dispvms', False)),
  100. allow_none=True
  101. )
  102. def __apply_system_defaults__(self):
  103. # updatevm
  104. if self.qvm_collection.updatevm != \
  105. self.update_vm_vmlist[self.update_vm_combo.currentIndex()]:
  106. self.qvm_collection.updatevm = \
  107. self.update_vm_vmlist[self.update_vm_combo.currentIndex()]
  108. # clockvm
  109. if self.qvm_collection.clockvm !=\
  110. self.clock_vm_vmlist[self.clock_vm_combo.currentIndex()]:
  111. self.qvm_collection.clockvm = \
  112. self.clock_vm_vmlist[self.clock_vm_combo.currentIndex()]
  113. # default netvm
  114. if self.qvm_collection.default_netvm !=\
  115. self.default_netvm_vmlist[
  116. self.default_netvm_combo.currentIndex()]:
  117. self.qvm_collection.default_netvm = \
  118. self.default_netvm_vmlist[
  119. self.default_netvm_combo.currentIndex()]
  120. # default template
  121. if self.qvm_collection.default_template != \
  122. self.default_template_vmlist[
  123. self.default_template_combo.currentIndex()]:
  124. self.qvm_collection.default_template = \
  125. self.default_template_vmlist[
  126. self.default_template_combo.currentIndex()]
  127. # default_dispvm
  128. if self.qvm_collection.default_dispvm != \
  129. self.default_dispvm_vmlist[
  130. self.default_dispvm_combo.currentIndex()]:
  131. self.qvm_collection.default_dispvm = \
  132. self.default_dispvm_vmlist[
  133. self.default_dispvm_combo.currentIndex()]
  134. def __init_kernel_defaults__(self):
  135. self.kernels_list, self.kernels_idx = utils.prepare_kernel_choice(
  136. self.default_kernel_combo, self.qvm_collection, 'default_kernel',
  137. None,
  138. allow_none=True
  139. )
  140. def __apply_kernel_defaults__(self):
  141. if self.qvm_collection.default_kernel != \
  142. self.kernels_list[self.default_kernel_combo.currentIndex()]:
  143. self.qvm_collection.default_kernel = \
  144. self.kernels_list[self.default_kernel_combo.currentIndex()]
  145. def __init_mem_defaults__(self):
  146. # qmemman settings
  147. self.qmemman_config = ConfigParser()
  148. self.vm_min_mem_val = '200MiB' # str(qmemman_algo.MIN_PREFMEM)
  149. self.dom0_mem_boost_val = '350MiB' # str(qmemman_algo.DOM0_MEM_BOOST)
  150. self.qmemman_config.read(qmemman_config_path)
  151. if self.qmemman_config.has_section('global'):
  152. self.vm_min_mem_val = \
  153. self.qmemman_config.get('global', 'vm-min-mem')
  154. self.dom0_mem_boost_val = \
  155. self.qmemman_config.get('global', 'dom0-mem-boost')
  156. self.vm_min_mem_val = parse_size(self.vm_min_mem_val)
  157. self.dom0_mem_boost_val = parse_size(self.dom0_mem_boost_val)
  158. self.min_vm_mem.setValue(self.vm_min_mem_val/1024/1024)
  159. self.dom0_mem_boost.setValue(self.dom0_mem_boost_val/1024/1024)
  160. def __apply_mem_defaults__(self):
  161. # qmemman settings
  162. current_min_vm_mem = self.min_vm_mem.value()
  163. current_dom0_mem_boost = self.dom0_mem_boost.value()
  164. if current_min_vm_mem*1024*1024 != self.vm_min_mem_val \
  165. or current_dom0_mem_boost*1024*1024 != self.dom0_mem_boost_val:
  166. current_min_vm_mem = str(current_min_vm_mem)+'MiB'
  167. current_dom0_mem_boost = str(current_dom0_mem_boost)+'MiB'
  168. if not self.qmemman_config.has_section('global'):
  169. # add the whole section
  170. self.qmemman_config.add_section('global')
  171. self.qmemman_config.set(
  172. 'global', 'vm-min-mem', current_min_vm_mem)
  173. self.qmemman_config.set(
  174. 'global', 'dom0-mem-boost', current_dom0_mem_boost)
  175. self.qmemman_config.set(
  176. 'global', 'cache-margin-factor', str(1.3))
  177. # removed qmemman_algo.CACHE_FACTOR
  178. qmemman_config_file = open(qmemman_config_path, 'a')
  179. self.qmemman_config.write(qmemman_config_file)
  180. qmemman_config_file.close()
  181. else:
  182. # If there already is a 'global' section, we don't use
  183. # SafeConfigParser.write() - it would get rid of
  184. # all the comments...
  185. lines_to_add = {}
  186. lines_to_add['vm-min-mem'] = \
  187. "vm-min-mem = " + current_min_vm_mem + "\n"
  188. lines_to_add['dom0-mem-boost'] = \
  189. "dom0-mem-boost = " + current_dom0_mem_boost + "\n"
  190. config_lines = []
  191. qmemman_config_file = open(qmemman_config_path, 'r')
  192. for line in qmemman_config_file:
  193. if line.strip().startswith('vm-min-mem'):
  194. config_lines.append(lines_to_add['vm-min-mem'])
  195. del lines_to_add['vm-min-mem']
  196. elif line.strip().startswith('dom0-mem-boost'):
  197. config_lines.append(lines_to_add['dom0-mem-boost'])
  198. del lines_to_add['dom0-mem-boost']
  199. else:
  200. config_lines.append(line)
  201. qmemman_config_file.close()
  202. for line in lines_to_add:
  203. config_lines.append(line)
  204. qmemman_config_file = open(qmemman_config_path, 'w')
  205. qmemman_config_file.writelines(config_lines)
  206. qmemman_config_file.close()
  207. def __init_updates__(self):
  208. # TODO: remove workaround when it is no longer needed
  209. self.dom0_updates_file_path = '/var/lib/qubes/updates/disable-updates'
  210. try:
  211. self.updates_dom0_val = bool(self.qvm_collection.domains[
  212. 'dom0'].features['service.qubes-update-check'])
  213. except KeyError:
  214. self.updates_dom0_val =\
  215. not os.path.isfile(self.dom0_updates_file_path)
  216. self.updates_dom0.setChecked(self.updates_dom0_val)
  217. self.updates_vm.setChecked(self.qvm_collection.check_updates_vm)
  218. self.enable_updates_all.clicked.connect(self.__enable_updates_all)
  219. self.disable_updates_all.clicked.connect(self.__disable_updates_all)
  220. self.repos = repos = dict()
  221. for i in _run_qrexec_repo('qubes.repos.List').split('\n'):
  222. lst = i.split('\0')
  223. # Keyed by repo name
  224. dct = repos[lst[0]] = dict()
  225. dct['prettyname'] = lst[1]
  226. dct['enabled'] = lst[2] == 'enabled'
  227. if repos['qubes-dom0-unstable']['enabled']:
  228. self.dom0_updates_repo.setCurrentIndex(3)
  229. elif repos['qubes-dom0-current-testing']['enabled']:
  230. self.dom0_updates_repo.setCurrentIndex(2)
  231. elif repos['qubes-dom0-security-testing']['enabled']:
  232. self.dom0_updates_repo.setCurrentIndex(1)
  233. elif repos['qubes-dom0-current']['enabled']:
  234. self.dom0_updates_repo.setCurrentIndex(0)
  235. else:
  236. raise Exception('Cannot detect enabled dom0 update repositories')
  237. if repos['qubes-templates-itl-testing']['enabled']:
  238. self.itl_tmpl_updates_repo.setCurrentIndex(1)
  239. elif repos['qubes-templates-itl']['enabled']:
  240. self.itl_tmpl_updates_repo.setCurrentIndex(0)
  241. else:
  242. raise Exception('Cannot detect enabled ITL template update '
  243. 'repositories')
  244. if repos['qubes-templates-community-testing']['enabled']:
  245. self.comm_tmpl_updates_repo.setCurrentIndex(2)
  246. elif repos['qubes-templates-community']['enabled']:
  247. self.comm_tmpl_updates_repo.setCurrentIndex(1)
  248. else:
  249. self.comm_tmpl_updates_repo.setCurrentIndex(0)
  250. def __enable_updates_all(self):
  251. reply = QtWidgets.QMessageBox.question(
  252. self, self.tr("Change state of all qubes"),
  253. self.tr("Are you sure you want to set all qubes to check "
  254. "for updates?"),
  255. QtWidgets.QMessageBox.Yes | QtWidgets.QMessageBox.Cancel)
  256. if reply == QtWidgets.QMessageBox.Cancel:
  257. return
  258. self.__set_updates_all(True)
  259. def __disable_updates_all(self):
  260. reply = QtWidgets.QMessageBox.question(
  261. self, self.tr("Change state of all qubes"),
  262. self.tr("Are you sure you want to set all qubes to not check "
  263. "for updates?"),
  264. QtWidgets.QMessageBox.Yes | QtWidgets.QMessageBox.Cancel)
  265. if reply == QtWidgets.QMessageBox.Cancel:
  266. return
  267. self.__set_updates_all(False)
  268. def __set_updates_all(self, state):
  269. for vm in self.qvm_collection.domains:
  270. if vm.klass != "AdminVM":
  271. vm.features['service.qubes-update-check'] = state
  272. def __apply_updates__(self):
  273. if self.updates_dom0.isChecked() != self.updates_dom0_val:
  274. self.qvm_collection.domains['dom0'].features[
  275. 'service.qubes-update-check'] = \
  276. self.updates_dom0.isChecked()
  277. if self.qvm_collection.check_updates_vm != self.updates_vm.isChecked():
  278. self.qvm_collection.check_updates_vm = self.updates_vm.isChecked()
  279. def _manage_repos(self, repolist, action):
  280. for name in repolist:
  281. if self.repos[name]['enabled'] and action == 'Enable' or \
  282. not self.repos[name]['enabled'] and action == 'Disable':
  283. continue
  284. try:
  285. result = _run_qrexec_repo('qubes.repos.' + action, name)
  286. if result != 'ok\n':
  287. raise RuntimeError(
  288. 'qrexec call stdout did not contain "ok" as expected',
  289. {'stdout': result})
  290. except RuntimeError as ex:
  291. msg = '{desc}; {args}'.format(desc=ex.args[0], args=', '.join(
  292. # This is kind of hard to mentally parse but really all
  293. # it does is pretty-print args[1], which is a dictionary
  294. ['{key}: {val}'.format(key=i[0], val=i[1]) for i in
  295. ex.args[1].items()]
  296. ))
  297. QtWidgets.QMessageBox.warning(
  298. None,
  299. self.tr("ERROR!"),
  300. self.tr("Error managing {repo} repository settings:"
  301. " {msg}".format(repo=name, msg=msg)))
  302. def _handle_dom0_updates_combobox(self, idx):
  303. idx += 1
  304. repolist = ['qubes-dom0-current', 'qubes-dom0-security-testing',
  305. 'qubes-dom0-current-testing', 'qubes-dom0-unstable']
  306. enable = repolist[:idx]
  307. disable = repolist[idx:]
  308. self._manage_repos(enable, 'Enable')
  309. self._manage_repos(disable, 'Disable')
  310. # pylint: disable=invalid-name
  311. def _handle_itl_tmpl_updates_combobox(self, idx):
  312. idx += 1
  313. repolist = ['qubes-templates-itl', 'qubes-templates-itl-testing']
  314. enable = repolist[:idx]
  315. disable = repolist[idx:]
  316. self._manage_repos(enable, 'Enable')
  317. self._manage_repos(disable, 'Disable')
  318. # pylint: disable=invalid-name
  319. def _handle_comm_tmpl_updates_combobox(self, idx):
  320. # We don't increment idx by 1 because this is the only combobox that
  321. # has an explicit "disable this repository entirely" option
  322. repolist = ['qubes-templates-community',
  323. 'qubes-templates-community-testing']
  324. enable = repolist[:idx]
  325. disable = repolist[idx:]
  326. self._manage_repos(enable, 'Enable')
  327. self._manage_repos(disable, 'Disable')
  328. def __apply_repos__(self):
  329. self._handle_dom0_updates_combobox(
  330. self.dom0_updates_repo.currentIndex())
  331. self._handle_itl_tmpl_updates_combobox(
  332. self.itl_tmpl_updates_repo.currentIndex())
  333. self._handle_comm_tmpl_updates_combobox(
  334. self.comm_tmpl_updates_repo.currentIndex())
  335. def reject(self):
  336. self.done(0)
  337. def save_and_apply(self):
  338. self.__apply_system_defaults__()
  339. self.__apply_kernel_defaults__()
  340. self.__apply_mem_defaults__()
  341. self.__apply_updates__()
  342. self.__apply_repos__()
  343. # Bases on the original code by:
  344. # Copyright (c) 2002-2007 Pascal Varet <p.varet@gmail.com>
  345. def handle_exception(exc_type, exc_value, exc_traceback):
  346. filename, line, dummy, dummy = traceback.extract_tb(exc_traceback).pop()
  347. filename = os.path.basename(filename)
  348. error = "%s: %s" % (exc_type.__name__, exc_value)
  349. QtWidgets.QMessageBox.critical(
  350. None,
  351. "Houston, we have a problem...",
  352. "Whoops. A critical error has occured. This is most likely a bug "
  353. "in Qubes Global Settings application.<br><br><b><i>%s</i></b>" %
  354. error + "at <b>line %d</b> of file <b>%s</b>.<br/><br/>"
  355. % (line, filename))
  356. def main():
  357. qtapp = QtWidgets.QApplication(sys.argv)
  358. qtapp.setOrganizationName("The Qubes Project")
  359. qtapp.setOrganizationDomain("http://qubes-os.org")
  360. qtapp.setApplicationName("Qubes Global Settings")
  361. sys.excepthook = handle_exception
  362. app = Qubes()
  363. global_window = GlobalSettingsWindow(qtapp, app)
  364. global_window.show()
  365. qtapp.exec_()
  366. qtapp.exit()
  367. if __name__ == "__main__":
  368. main()