test_global_settings.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. #!/usr/bin/python3
  2. #
  3. # The Qubes OS Project, https://www.qubes-os.org/
  4. #
  5. # Copyright (C) 2016 Marta Marczykowska-Górecka
  6. # <marmarta@invisiblethingslab.com>
  7. #
  8. # This program is free software; you can redistribute it and/or modify
  9. # it under the terms of the GNU General Public License as published by
  10. # the Free Software Foundation; either version 2 of the License, or
  11. # (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 General Public License along
  19. # with this program; if not, write to the Free Software Foundation, Inc.,
  20. # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  21. #
  22. import logging.handlers
  23. import unittest
  24. import unittest.mock
  25. from PyQt5 import QtTest, QtCore, QtWidgets
  26. from qubesadmin import Qubes
  27. from qubesmanager.tests import init_qtapp
  28. import qubesmanager.global_settings as global_settings
  29. class GlobalSettingsTest(unittest.TestCase):
  30. def setUp(self):
  31. super(GlobalSettingsTest, self).setUp()
  32. self.qtapp, self.loop = init_qtapp()
  33. self.qapp = Qubes()
  34. self.dialog = global_settings.GlobalSettingsWindow(self.qtapp,
  35. self.qapp)
  36. self.setattr_patcher = unittest.mock.patch.object(
  37. type(self.dialog.qvm_collection), "__setattr__")
  38. self.setattr_mock = self.setattr_patcher.start()
  39. self.addCleanup(self.setattr_patcher.stop)
  40. def tearDown(self):
  41. self.dialog.close()
  42. self.qtapp.processEvents()
  43. yield from self.loop.sleep(1)
  44. super(GlobalSettingsTest, self).tearDown()
  45. def test_00_settings_started(self):
  46. # non-empty drop-downs
  47. self.assertNotEqual(
  48. self.dialog.default_kernel_combo.currentText(), "",
  49. "Default kernel not listed")
  50. self.assertNotEqual(
  51. self.dialog.default_netvm_combo.currentText(), "",
  52. "Default netVM not listed")
  53. self.assertNotEqual(
  54. self.dialog.default_template_combo.currentText(),
  55. "", "Default template not listed")
  56. self.assertNotEqual(
  57. self.dialog.clock_vm_combo.currentText(), "",
  58. "ClockVM not listed")
  59. self.assertNotEqual(
  60. self.dialog.update_vm_combo.currentText(), "",
  61. "UpdateVM for dom0 not listed")
  62. self.assertNotEqual(
  63. self.dialog.default_dispvm_combo.currentText(), "",
  64. "Default DispVM not listed")
  65. # not empty memory settings
  66. self.assertTrue(len(self.dialog.min_vm_mem.text()) > 4,
  67. "Too short min mem value")
  68. self.assertTrue(len(self.dialog.dom0_mem_boost.text()) > 4,
  69. "Too short dom0 mem boost value")
  70. def test_01_load_correct_defs(self):
  71. # correctly selected default template
  72. selected_default_template = \
  73. self.dialog.default_template_combo.currentText()
  74. self.assertTrue(
  75. selected_default_template.startswith(
  76. str(getattr(self.qapp, 'default_template', '(none)'))),
  77. "Incorrect default template loaded")
  78. # correctly selected default NetVM
  79. selected_default_netvm = self.dialog.default_netvm_combo.currentText()
  80. self.assertTrue(selected_default_netvm.startswith(
  81. str(getattr(self.qapp, 'default_netvm', '(none)'))),
  82. "Incorrect default netVM loaded")
  83. # correctly selected default kernel
  84. selected_default_kernel = self.dialog.default_kernel_combo.currentText()
  85. self.assertTrue(selected_default_kernel.startswith(
  86. str(getattr(self.qapp, 'default_kernel', '(none)'))),
  87. "Incorrect default kernel loaded")
  88. # correct ClockVM
  89. selected_clockvm = self.dialog.clock_vm_combo.currentText()
  90. correct_clockvm = str(getattr(self.qapp, 'clockvm', "(none)"))
  91. self.assertTrue(selected_clockvm.startswith(correct_clockvm),
  92. "Incorrect clockVM loaded")
  93. # correct updateVM
  94. selected_updatevm = self.dialog.update_vm_combo.currentText()
  95. correct_updatevm = str(getattr(self.qapp, 'updatevm', "(none)"))
  96. self.assertTrue(selected_updatevm.startswith(correct_updatevm),
  97. "Incorrect updateVm loaded")
  98. # correct defaultDispVM
  99. selected_default_dispvm = self.dialog.default_dispvm_combo.currentText()
  100. correct_default_dispvm = \
  101. str(getattr(self.qapp, 'default_dispvm', "(none)"))
  102. self.assertTrue(
  103. selected_default_dispvm.startswith(correct_default_dispvm),
  104. "Incorrect defaultDispVM loaded")
  105. # update vm status
  106. self.assertEqual(self.qapp.check_updates_vm,
  107. self.dialog.updates_vm.isChecked(),
  108. "Incorrect check qube updates value loaded")
  109. def test_02_dom0_updates_load(self):
  110. # check dom0 updates
  111. try:
  112. dom0_updates = self.qapp.domains[
  113. 'dom0'].features['service.qubes-update-check']
  114. except KeyError:
  115. self.skipTest("check_updates_dom0 property not implemented")
  116. return
  117. self.assertEqual(bool(dom0_updates),
  118. self.dialog.updates_dom0.isChecked(),
  119. "Incorrect dom0 updates value")
  120. def __set_noncurrent(self, widget):
  121. if widget.count() < 2:
  122. self.skipTest("not enough choices for " + widget.objectName())
  123. widget.setCurrentIndex(0)
  124. while widget.currentText().endswith("(current)") \
  125. or widget.currentText().startswith("(none)"):
  126. widget.setCurrentIndex(widget.currentIndex() + 1)
  127. return widget.currentText()
  128. def __set_none(self, widget):
  129. widget.setCurrentIndex(0)
  130. while not widget.currentText().startswith("(none)"):
  131. if widget.currentIndex() == widget.count():
  132. self.skipTest("none not available for " + widget.objectName())
  133. widget.setCurrentIndex(widget.currentIndex() + 1)
  134. def __click_ok(self):
  135. okwidget = self.dialog.buttonBox.button(
  136. self.dialog.buttonBox.Ok)
  137. QtTest.QTest.mouseClick(okwidget, QtCore.Qt.LeftButton)
  138. def __click_cancel(self):
  139. cancelwidget = self.dialog.buttonBox.button(
  140. self.dialog.buttonBox.Cancel)
  141. QtTest.QTest.mouseClick(cancelwidget, QtCore.Qt.LeftButton)
  142. def test_03_nothing_changed_ok(self):
  143. self.__click_ok()
  144. self.assertEqual(self.setattr_mock.call_count, 0,
  145. "Changes occurred despite no changes being made")
  146. def test_04_nothing_changed_cancel(self):
  147. self.__click_cancel()
  148. self.assertEqual(self.setattr_mock.call_count, 0,
  149. "Changes occurred despite no changes being made")
  150. def test_10_set_update_vm(self):
  151. new_updatevm_name = self.__set_noncurrent(self.dialog.update_vm_combo)
  152. self.__click_ok()
  153. self.setattr_mock.assert_called_once_with('updatevm', new_updatevm_name)
  154. def test_11_set_update_vm_to_none(self):
  155. self.__set_none(self.dialog.update_vm_combo)
  156. self.__click_ok()
  157. self.setattr_mock.assert_called_once_with('updatevm', None)
  158. def test_20_set_clock_vm(self):
  159. new_clockvm_name = self.__set_noncurrent(self.dialog.clock_vm_combo)
  160. self.__click_ok()
  161. self.setattr_mock.assert_called_once_with('clockvm', new_clockvm_name)
  162. def test_21_set_clock_vm_to_none(self):
  163. self.__set_none(self.dialog.clock_vm_combo)
  164. self.__click_ok()
  165. self.setattr_mock.assert_called_once_with('clockvm', None)
  166. def test_30_set_default_netvm(self):
  167. new_netvm_name = self.__set_noncurrent(self.dialog.default_netvm_combo)
  168. self.__click_ok()
  169. self.setattr_mock.assert_called_once_with('default_netvm',
  170. new_netvm_name)
  171. def test_31_set_default_netvm_to_none(self):
  172. self.__set_none(self.dialog.default_netvm_combo)
  173. self.__click_ok()
  174. self.setattr_mock.assert_called_once_with('default_netvm', None)
  175. def test_40_set_default_template(self):
  176. new_def_template_name = self.__set_noncurrent(
  177. self.dialog.default_template_combo)
  178. self.__click_ok()
  179. self.setattr_mock.assert_called_once_with('default_template',
  180. new_def_template_name)
  181. def test_50_set_default_kernel(self):
  182. new_def_kernel_name = self.__set_noncurrent(
  183. self.dialog.default_kernel_combo)
  184. self.__click_ok()
  185. self.setattr_mock.assert_called_once_with('default_kernel',
  186. new_def_kernel_name)
  187. def test_51_set_default_kernel_to_none(self):
  188. self.__set_none(self.dialog.default_kernel_combo)
  189. self.__click_ok()
  190. self.setattr_mock.assert_called_once_with('default_kernel',
  191. None)
  192. def test_60_set_dom0_updates_true(self):
  193. current_state = self.dialog.updates_dom0.isChecked()
  194. self.dialog.updates_dom0.setChecked(not current_state)
  195. with unittest.mock.patch.object(
  196. type(self.dialog.qvm_collection.domains['dom0'].features),
  197. '__setitem__') as mock_features:
  198. self.__click_ok()
  199. mock_features.assert_called_once_with('service.qubes-update-check',
  200. not current_state)
  201. def test_70_change_vm_updates(self):
  202. current_state = self.dialog.updates_vm.isChecked()
  203. self.dialog.updates_vm.setChecked(not current_state)
  204. self.__click_ok()
  205. self.setattr_mock.assert_called_once_with('check_updates_vm',
  206. not current_state)
  207. @unittest.mock.patch("PyQt5.QtWidgets.QMessageBox.question",
  208. return_value=QtWidgets.QMessageBox.Yes)
  209. @unittest.mock.patch('qubesadmin.features.Features.__setitem__')
  210. def test_72_set_all_vms_true(self, mock_features, msgbox):
  211. QtTest.QTest.mouseClick(self.dialog.enable_updates_all,
  212. QtCore.Qt.LeftButton)
  213. self.assertEqual(msgbox.call_count, 1,
  214. "Wrong number of confirmation window calls")
  215. call_list_expected = \
  216. [unittest.mock.call('service.qubes-update-check', True) for vm
  217. in self.qapp.domains if vm.klass != 'AdminVM']
  218. self.assertListEqual(call_list_expected,
  219. mock_features.call_args_list)
  220. @unittest.mock.patch("PyQt5.QtWidgets.QMessageBox.question",
  221. return_value=QtWidgets.QMessageBox.Yes)
  222. @unittest.mock.patch('qubesadmin.features.Features.__setitem__')
  223. def test_73_set_all_vms_false(self, mock_features, msgbox):
  224. QtTest.QTest.mouseClick(self.dialog.disable_updates_all,
  225. QtCore.Qt.LeftButton)
  226. self.assertEqual(msgbox.call_count, 1,
  227. "Wrong number of confirmation window calls")
  228. call_list_expected = \
  229. [unittest.mock.call('service.qubes-update-check', False) for vm
  230. in self.qapp.domains if vm.klass != 'AdminVM']
  231. self.assertListEqual(call_list_expected,
  232. mock_features.call_args_list)
  233. def test_80_set_default_dispvm(self):
  234. new_dispvm_name = self.__set_noncurrent(
  235. self.dialog.default_dispvm_combo)
  236. self.__click_ok()
  237. self.setattr_mock.assert_called_once_with('default_dispvm',
  238. new_dispvm_name)
  239. def test_81_set_default_dispvm_to_none(self):
  240. self.__set_none(self.dialog.default_dispvm_combo)
  241. self.__click_ok()
  242. self.setattr_mock.assert_called_once_with('default_dispvm', None)
  243. @unittest.mock.patch.object(
  244. type(Qubes()), '__getattr__',
  245. side_effect=(lambda x: False if x == 'check_updates_vm' else None))
  246. def test_90_test_all_set_none(self, mock_qubes):
  247. mock_qubes.configure_mock()
  248. self.dialog = global_settings.GlobalSettingsWindow(
  249. self.qtapp, self.qapp)
  250. self.assertEqual(self.dialog.update_vm_combo.currentText(),
  251. "(none) (current)",
  252. "UpdateVM displays as none incorrectly")
  253. self.assertEqual(self.dialog.clock_vm_combo.currentText(),
  254. "(none) (current)",
  255. "ClockVM displays as none incorrectly")
  256. self.assertEqual(self.dialog.default_netvm_combo.currentText(),
  257. "(none) (current)",
  258. "Default NetVM displays as none incorrectly")
  259. self.assertEqual(self.dialog.default_template_combo.currentText(),
  260. "(none) (current)",
  261. "Default template displays as none incorrectly")
  262. self.assertEqual(self.dialog.default_kernel_combo.currentText(),
  263. "(none) (current)",
  264. "Defautl kernel displays as none incorrectly")
  265. self.assertEqual(self.dialog.default_dispvm_combo.currentText(),
  266. "(none) (current)",
  267. "Default DispVM displays as none incorrectly")
  268. if __name__ == "__main__":
  269. ha_syslog = logging.handlers.SysLogHandler('/dev/log')
  270. ha_syslog.setFormatter(
  271. logging.Formatter('%(name)s[%(process)d]: %(message)s'))
  272. logging.root.addHandler(ha_syslog)
  273. unittest.main()
  274. # TODO: add tests for memory settings once memory is handled better