From f5d94c2bf762acd4bf9eaa429b26976084e6169f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marta=20Marczykowska-G=C3=B3recka?= Date: Fri, 26 Jan 2018 22:38:16 +0100 Subject: [PATCH 1/8] Tests for global settings Unit tests for global settings window; based on QtTest --- qubesmanager/tests/__init__.py | 0 qubesmanager/tests/test_global_settings.py | 331 +++++++++++++++++++++ setup.py | 2 +- 3 files changed, 332 insertions(+), 1 deletion(-) create mode 100644 qubesmanager/tests/__init__.py create mode 100644 qubesmanager/tests/test_global_settings.py diff --git a/qubesmanager/tests/__init__.py b/qubesmanager/tests/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/qubesmanager/tests/test_global_settings.py b/qubesmanager/tests/test_global_settings.py new file mode 100644 index 0000000..e489d11 --- /dev/null +++ b/qubesmanager/tests/test_global_settings.py @@ -0,0 +1,331 @@ +#!/usr/bin/python3 +# +# The Qubes OS Project, https://www.qubes-os.org/ +# +# Copyright (C) 2016 Marek Marczykowski-Górecki +# +# +# This program is free software; you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 2 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License along +# with this program; if not, write to the Free Software Foundation, Inc., +# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. +# +import logging.handlers +import sys +import unittest +import unittest.mock + +import time +from PyQt4 import QtGui, QtTest, QtCore +from qubesadmin import Qubes +from qubes.tests import SystemTestCase +import qubesmanager.global_settings as global_settings +import concurrent.futures + + +def wrap_in_loop(func): + def wrapped(self): + self.loop.run_until_complete( + self.loop.run_in_executor(self.executor, + func, self)) + return wrapped + + +class GlobalSettingsTest(SystemTestCase): + def setUp(self): + super(GlobalSettingsTest, self).setUp() + + self.qtapp = QtGui.QApplication(sys.argv) + self.executor = concurrent.futures.ThreadPoolExecutor(max_workers=1) + self.setUpInExecutor() + + @wrap_in_loop + def setUpInExecutor(self): + self.qapp = Qubes() + self.dialog = global_settings.GlobalSettingsWindow( + self.qtapp, self.qapp) + + def tearDown(self): + self.tearDownInExecutor() + super(GlobalSettingsTest, self).tearDown() + + @wrap_in_loop + def tearDownInExecutor(self): + del self.dialog + del self.qtapp + + @wrap_in_loop + def test_settings_started(self): + # non-empty drop-downs + self.assertNotEqual( + self.dialog.default_kernel_combo.currentText(), "") + self.assertNotEqual( + self.dialog.default_netvm_combo.currentText(), "") + self.assertNotEqual( + self.dialog.default_template_combo.currentText(), + "") + self.assertNotEqual( + self.dialog.clock_vm_combo.currentText(), "") + self.assertNotEqual( + self.dialog.update_vm_combo.currentText(), "") + + @wrap_in_loop + def test_load_correct_defs(self): + # correctly selected default template + selected_default_template = \ + self.dialog.default_template_combo.currentText() + self.assertTrue( + selected_default_template.startswith( + self.app.default_template.name)) + + # correctly selected default NetVM + selected_default_netvm = \ + self.dialog.default_netvm_combo.currentText() + self.assertTrue(selected_default_netvm.startswith( + self.app.default_netvm.name)) + + # correctly selected default kernel + selected_default_kernel = \ + self.dialog.default_kernel_combo.currentText() + self.assertTrue(selected_default_kernel.startswith( + self.app.default_kernel)) + + # correct ClockVM + selected_clockvm = \ + self.dialog.clock_vm_combo.currentText() + correct_clockvm = self.app.clockvm.name if self.app.clockvm \ + else "(none)" + self.assertTrue(selected_clockvm.startswith(correct_clockvm)) + + # correct updateVM + selected_updatevm = \ + self.dialog.update_vm_combo.currentText() + correct_updatevm = \ + self.app.updatevm.name if self.app.updatevm else "(none)" + self.assertTrue(selected_updatevm.startswith(correct_updatevm)) + + # update vm status + self.assertEqual(self.app.check_updates_vm, + self.dialog.updates_vm.isChecked()) + + def test_dom0_updates_load(self): + # check dom0 updates + try: + dom0_updates = self.app.check_updates_dom0 + except AttributeError: + self.skipTest("check_updates_dom0 property not implemented") + return + + self.assertEqual(dom0_updates, self.dialog.updates_dom0.isChecked()) + + def __set_noncurrent(self, widget): + if widget.count() < 2: + self.skipTest("not enough choices for " + widget.objectName()) + + widget.setCurrentIndex(0) + while widget.currentText().endswith("(current)") \ + or widget.currentText().startswith("(none)"): + widget.setCurrentIndex(widget.currentIndex() + 1) + + return widget.currentText() + + def __set_none(self, widget): + widget.setCurrentIndex(0) + while not widget.currentText().startswith("(none)"): + if widget.currentIndex() == widget.count(): + self.skipTest("none not available for " + widget.objectName()) + widget.setCurrentIndex(widget.currentIndex() + 1) + + def __click_ok(self): + okwidget = self.dialog.buttonBox.button( + self.dialog.buttonBox.Ok) + + QtTest.QTest.mouseClick(okwidget, + QtCore.Qt.LeftButton) + + @wrap_in_loop + def test_set_update_vm(self): + new_updatevm_name = self.__set_noncurrent(self.dialog.update_vm_combo) + self.__click_ok() + + self.assertEqual(self.app.updatevm.name, new_updatevm_name) + + @wrap_in_loop + def test_set_update_vm_to_none(self): + self.__set_none(self.dialog.update_vm_combo) + self.__click_ok() + + self.assertIsNone(self.app.updatevm) + + @wrap_in_loop + def test_set_update_vm_to_none2(self): + self.app.updatevm = None + self.dialog = global_settings.GlobalSettingsWindow( + self.qtapp, self.qapp) + + self.assertEqual(self.dialog.update_vm_combo.currentText(), + "(none) (current)") + + @wrap_in_loop + def test_set_clock_vm(self): + new_clockvm_name = self.__set_noncurrent(self.dialog.clock_vm_combo) + self.__click_ok() + + self.assertEqual(self.app.clockvm.name, new_clockvm_name) + + @wrap_in_loop + def test_set_clock_vm_to_none(self): + self.__set_none(self.dialog.clock_vm_combo) + self.__click_ok() + + self.assertIsNone(self.app.clockvm) + + @wrap_in_loop + def test_set_clock_vm_to_none2(self): + self.app.clockvm = None + self.dialog = global_settings.GlobalSettingsWindow( + self.qtapp, self.qapp) + + self.assertEqual(self.dialog.clock_vm_combo.currentText(), + "(none) (current)") + + @wrap_in_loop + def test_set_default_netvm(self): + new_netvm_name = self.__set_noncurrent(self.dialog.default_netvm_combo) + self.__click_ok() + + self.assertEqual(self.app.default_netvm.name, new_netvm_name) + + @wrap_in_loop + def test_set_default_netvm_to_none(self): + self.__set_none(self.dialog.default_netvm_combo) + self.__click_ok() + + self.assertIsNone(self.app.default_netvm) + + @wrap_in_loop + def test_set_default_netvm_to_none2(self): + self.app.default_netvm = None + self.dialog = global_settings.GlobalSettingsWindow( + self.qtapp, self.qapp) + + self.assertEqual(self.dialog.default_netvm_combo.currentText(), + "(none) (current)") + + @wrap_in_loop + def test_set_default_template(self): + new_def_template_name = self.__set_noncurrent( + self.dialog.default_template_combo) + self.__click_ok() + + self.assertEqual(self.app.default_template.name, new_def_template_name) + + @wrap_in_loop + def test_set_default_kernel(self): + new_def_kernel_name = self.__set_noncurrent( + self.dialog.default_kernel_combo) + self.__click_ok() + + self.assertEqual(self.app.default_kernel, new_def_kernel_name) + + @wrap_in_loop + def test_set_default_kernel_to_none(self): + self.__set_none(self.dialog.default_kernel_combo) + self.__click_ok() + + self.assertEqual(self.app.default_kernel, '') + + @wrap_in_loop + def test_set_default_kernel_to_none2(self): + self.app.default_kernel = None + self.dialog = global_settings.GlobalSettingsWindow( + self.qtapp, self.qapp) + + self.assertEqual(self.dialog.default_kernel_combo.currentText(), + "(none) (current)") + + @wrap_in_loop + def test_set_dom0_updates_true(self): + self.dialog.updates_dom0.setChecked(True) + self.__click_ok() + + if not hasattr(self.app, 'check_updates_dom0'): + self.skipTest("check_updates_dom0 property not implemented") + + self.assertTrue(self.app.check_updates_dom0) + + @wrap_in_loop + def test_set_dom0_updates_false(self): + self.dialog.updates_dom0.setChecked(False) + self.__click_ok() + + if not hasattr(self.app, 'check_updates_dom0'): + self.skipTest("check_updates_dom0 property not implemented") + + self.assertFalse(self.app.check_updates_dom0) + + @wrap_in_loop + def test_set_vm_updates_true(self): + self.dialog.updates_vm.setChecked(True) + self.__click_ok() + + self.assertTrue(self.app.check_updates_vm) + + @wrap_in_loop + def test_set_vm_updates_false(self): + self.dialog.updates_vm.setChecked(False) + self.__click_ok() + + self.assertFalse(self.app.check_updates_vm) + + @wrap_in_loop + def test_00set_all_vms_true(self): + + with unittest.mock.patch("PyQt4.QtGui.QMessageBox.question", + return_value=QtGui.QMessageBox.Yes) as msgbox: + + QtTest.QTest.mouseClick(self.dialog.enable_updates_all, + QtCore.Qt.LeftButton) + + msgbox.assert_called_once_with( + self.dialog, + "Change state of all qubes", + "Are you sure you want to set all qubes to check for updates?", + unittest.mock.ANY) + + for vm in self.app.domains: + self.assertTrue(vm.features['check-updates']) + + @wrap_in_loop + def test_00set_all_vms_false(self): + with unittest.mock.patch("PyQt4.QtGui.QMessageBox.question", + return_value=QtGui.QMessageBox.Yes) as msgbox: + QtTest.QTest.mouseClick(self.dialog.disable_updates_all, + QtCore.Qt.LeftButton) + + msgbox.assert_called_once_with( + self.dialog, + "Change state of all qubes", + "Are you sure you want to set all qubes to not check " + "for updates?", + unittest.mock.ANY) + + for vm in self.app.domains: + self.assertFalse(vm.features['check-updates']) + + +if __name__ == "__main__": + ha_syslog = logging.handlers.SysLogHandler('/dev/log') + ha_syslog.setFormatter( + logging.Formatter('%(name)s[%(process)d]: %(message)s')) + logging.root.addHandler(ha_syslog) + unittest.main() diff --git a/setup.py b/setup.py index 721284c..f0388a8 100644 --- a/setup.py +++ b/setup.py @@ -12,7 +12,7 @@ if __name__ == '__main__': description='Qubes OS Manager', license='GPL2+', url='https://www.qubes-os.org/', - packages=setuptools.find_packages(), + packages=setuptools.find_packages(exclude=['qubesmanager.tests']), package_data={ 'qubesmanager': ['i18n/*'] }, From 44163be96d33158067ee9cd5073aa32c492a6e27 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marta=20Marczykowska-G=C3=B3recka?= Date: Sat, 27 Jan 2018 18:12:20 +0100 Subject: [PATCH 2/8] Removed unused import from global settings test --- qubesmanager/tests/test_global_settings.py | 1 - 1 file changed, 1 deletion(-) diff --git a/qubesmanager/tests/test_global_settings.py b/qubesmanager/tests/test_global_settings.py index e489d11..f27a3b0 100644 --- a/qubesmanager/tests/test_global_settings.py +++ b/qubesmanager/tests/test_global_settings.py @@ -24,7 +24,6 @@ import sys import unittest import unittest.mock -import time from PyQt4 import QtGui, QtTest, QtCore from qubesadmin import Qubes from qubes.tests import SystemTestCase From fc6c886543246b1541efd3aacc090894aa61168b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marta=20Marczykowska-G=C3=B3recka?= Date: Sat, 27 Jan 2018 18:32:00 +0100 Subject: [PATCH 3/8] Numbered tests for Global Settings --- qubesmanager/tests/test_global_settings.py | 46 +++++++++++----------- 1 file changed, 24 insertions(+), 22 deletions(-) diff --git a/qubesmanager/tests/test_global_settings.py b/qubesmanager/tests/test_global_settings.py index f27a3b0..4be171a 100644 --- a/qubesmanager/tests/test_global_settings.py +++ b/qubesmanager/tests/test_global_settings.py @@ -30,6 +30,7 @@ from qubes.tests import SystemTestCase import qubesmanager.global_settings as global_settings import concurrent.futures +# sudo systemctl stop qubesd; sudo -E python3 test_backup.py -v ; sudo systemctl start qubesd def wrap_in_loop(func): def wrapped(self): @@ -63,7 +64,7 @@ class GlobalSettingsTest(SystemTestCase): del self.qtapp @wrap_in_loop - def test_settings_started(self): + def test_00_settings_started(self): # non-empty drop-downs self.assertNotEqual( self.dialog.default_kernel_combo.currentText(), "") @@ -78,7 +79,7 @@ class GlobalSettingsTest(SystemTestCase): self.dialog.update_vm_combo.currentText(), "") @wrap_in_loop - def test_load_correct_defs(self): + def test_01_load_correct_defs(self): # correctly selected default template selected_default_template = \ self.dialog.default_template_combo.currentText() @@ -116,7 +117,8 @@ class GlobalSettingsTest(SystemTestCase): self.assertEqual(self.app.check_updates_vm, self.dialog.updates_vm.isChecked()) - def test_dom0_updates_load(self): + @wrap_in_loop + def test_02_dom0_updates_load(self): # check dom0 updates try: dom0_updates = self.app.check_updates_dom0 @@ -152,21 +154,21 @@ class GlobalSettingsTest(SystemTestCase): QtCore.Qt.LeftButton) @wrap_in_loop - def test_set_update_vm(self): + def test_10_set_update_vm(self): new_updatevm_name = self.__set_noncurrent(self.dialog.update_vm_combo) self.__click_ok() self.assertEqual(self.app.updatevm.name, new_updatevm_name) @wrap_in_loop - def test_set_update_vm_to_none(self): + def test_11_set_update_vm_to_none(self): self.__set_none(self.dialog.update_vm_combo) self.__click_ok() self.assertIsNone(self.app.updatevm) @wrap_in_loop - def test_set_update_vm_to_none2(self): + def test_12_set_update_vm_to_none2(self): self.app.updatevm = None self.dialog = global_settings.GlobalSettingsWindow( self.qtapp, self.qapp) @@ -175,21 +177,21 @@ class GlobalSettingsTest(SystemTestCase): "(none) (current)") @wrap_in_loop - def test_set_clock_vm(self): + def test_20_set_clock_vm(self): new_clockvm_name = self.__set_noncurrent(self.dialog.clock_vm_combo) self.__click_ok() self.assertEqual(self.app.clockvm.name, new_clockvm_name) @wrap_in_loop - def test_set_clock_vm_to_none(self): + def test_21_set_clock_vm_to_none(self): self.__set_none(self.dialog.clock_vm_combo) self.__click_ok() self.assertIsNone(self.app.clockvm) @wrap_in_loop - def test_set_clock_vm_to_none2(self): + def test_22_set_clock_vm_to_none2(self): self.app.clockvm = None self.dialog = global_settings.GlobalSettingsWindow( self.qtapp, self.qapp) @@ -198,21 +200,21 @@ class GlobalSettingsTest(SystemTestCase): "(none) (current)") @wrap_in_loop - def test_set_default_netvm(self): + def test_30_set_default_netvm(self): new_netvm_name = self.__set_noncurrent(self.dialog.default_netvm_combo) self.__click_ok() self.assertEqual(self.app.default_netvm.name, new_netvm_name) @wrap_in_loop - def test_set_default_netvm_to_none(self): + def test_31_set_default_netvm_to_none(self): self.__set_none(self.dialog.default_netvm_combo) self.__click_ok() self.assertIsNone(self.app.default_netvm) @wrap_in_loop - def test_set_default_netvm_to_none2(self): + def test_32_set_default_netvm_to_none2(self): self.app.default_netvm = None self.dialog = global_settings.GlobalSettingsWindow( self.qtapp, self.qapp) @@ -221,7 +223,7 @@ class GlobalSettingsTest(SystemTestCase): "(none) (current)") @wrap_in_loop - def test_set_default_template(self): + def test_40_set_default_template(self): new_def_template_name = self.__set_noncurrent( self.dialog.default_template_combo) self.__click_ok() @@ -229,7 +231,7 @@ class GlobalSettingsTest(SystemTestCase): self.assertEqual(self.app.default_template.name, new_def_template_name) @wrap_in_loop - def test_set_default_kernel(self): + def test__50_set_default_kernel(self): new_def_kernel_name = self.__set_noncurrent( self.dialog.default_kernel_combo) self.__click_ok() @@ -237,14 +239,14 @@ class GlobalSettingsTest(SystemTestCase): self.assertEqual(self.app.default_kernel, new_def_kernel_name) @wrap_in_loop - def test_set_default_kernel_to_none(self): + def test_51_set_default_kernel_to_none(self): self.__set_none(self.dialog.default_kernel_combo) self.__click_ok() self.assertEqual(self.app.default_kernel, '') @wrap_in_loop - def test_set_default_kernel_to_none2(self): + def test_52_set_default_kernel_to_none2(self): self.app.default_kernel = None self.dialog = global_settings.GlobalSettingsWindow( self.qtapp, self.qapp) @@ -253,7 +255,7 @@ class GlobalSettingsTest(SystemTestCase): "(none) (current)") @wrap_in_loop - def test_set_dom0_updates_true(self): + def test_60_set_dom0_updates_true(self): self.dialog.updates_dom0.setChecked(True) self.__click_ok() @@ -263,7 +265,7 @@ class GlobalSettingsTest(SystemTestCase): self.assertTrue(self.app.check_updates_dom0) @wrap_in_loop - def test_set_dom0_updates_false(self): + def test_61_set_dom0_updates_false(self): self.dialog.updates_dom0.setChecked(False) self.__click_ok() @@ -273,21 +275,21 @@ class GlobalSettingsTest(SystemTestCase): self.assertFalse(self.app.check_updates_dom0) @wrap_in_loop - def test_set_vm_updates_true(self): + def test_70_set_vm_updates_true(self): self.dialog.updates_vm.setChecked(True) self.__click_ok() self.assertTrue(self.app.check_updates_vm) @wrap_in_loop - def test_set_vm_updates_false(self): + def test_71_set_vm_updates_false(self): self.dialog.updates_vm.setChecked(False) self.__click_ok() self.assertFalse(self.app.check_updates_vm) @wrap_in_loop - def test_00set_all_vms_true(self): + def test_72_set_all_vms_true(self): with unittest.mock.patch("PyQt4.QtGui.QMessageBox.question", return_value=QtGui.QMessageBox.Yes) as msgbox: @@ -305,7 +307,7 @@ class GlobalSettingsTest(SystemTestCase): self.assertTrue(vm.features['check-updates']) @wrap_in_loop - def test_00set_all_vms_false(self): + def test_73_set_all_vms_false(self): with unittest.mock.patch("PyQt4.QtGui.QMessageBox.question", return_value=QtGui.QMessageBox.Yes) as msgbox: QtTest.QTest.mouseClick(self.dialog.disable_updates_all, From f6bda2e9f64058388664e3e8e4767ae61c56408e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marta=20Marczykowska-G=C3=B3recka?= Date: Sun, 28 Jan 2018 18:10:34 +0100 Subject: [PATCH 4/8] Tests for Backup Initial tests. --- qubesmanager/tests/test_backup.py | 64 +++++++++++++++++++++++++++++++ 1 file changed, 64 insertions(+) create mode 100644 qubesmanager/tests/test_backup.py diff --git a/qubesmanager/tests/test_backup.py b/qubesmanager/tests/test_backup.py new file mode 100644 index 0000000..a429fcf --- /dev/null +++ b/qubesmanager/tests/test_backup.py @@ -0,0 +1,64 @@ +#!/usr/bin/python3 +# +# The Qubes OS Project, https://www.qubes-os.org/ +# +# Copyright (C) 2016 Marek Marczykowski-Górecki +# +# +# This program is free software; you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 2 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License along +# with this program; if not, write to the Free Software Foundation, Inc., +# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. +# +import logging.handlers +import sys +import unittest +# import unittest.mock + +from PyQt4 import QtGui, QtTest, QtCore +from qubesadmin import Qubes +import qubesmanager.backup as backup_gui + + +class BackupTest(unittest.TestCase): + def setUp(self): + super(BackupTest, self).setUp() + + self.qapp = Qubes() + self.qtapp = QtGui.QApplication(sys.argv) + self.dialog = backup_gui.BackupVMsWindow(self.qtapp, self.qapp) + + def tearDown(self): + del self.dialog + del self.qtapp + del self.qapp + super(BackupTest, self).tearDown() + + def test_window_loads(self): + self.assertTrue(self.dialog.select_vms_widget is not None) + + def test_vms_load_correctly(self): + all_vms = len([vm for vm in self.qapp.domains + if not vm.features.get('internal', False)]) + + selected_vms = self.dialog.select_vms_widget.selected_list.count() + available_vms = self.dialog.select_vms_widget.available_list.count() + + self.assertEqual(all_vms, available_vms + selected_vms) + + +if __name__ == "__main__": + ha_syslog = logging.handlers.SysLogHandler('/dev/log') + ha_syslog.setFormatter( + logging.Formatter('%(name)s[%(process)d]: %(message)s')) + logging.root.addHandler(ha_syslog) + unittest.main() From a69c061b498193439dd9bfd4e25615006c9d23ea Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marta=20Marczykowska-G=C3=B3recka?= Date: Thu, 1 Mar 2018 23:02:30 +0100 Subject: [PATCH 5/8] Backup tests --- .../{test_backup.py => test_backup_01.py} | 32 ++++++++++++++++++- 1 file changed, 31 insertions(+), 1 deletion(-) rename qubesmanager/tests/{test_backup.py => test_backup_01.py} (59%) diff --git a/qubesmanager/tests/test_backup.py b/qubesmanager/tests/test_backup_01.py similarity index 59% rename from qubesmanager/tests/test_backup.py rename to qubesmanager/tests/test_backup_01.py index a429fcf..1aa8901 100644 --- a/qubesmanager/tests/test_backup.py +++ b/qubesmanager/tests/test_backup_01.py @@ -22,7 +22,7 @@ import logging.handlers import sys import unittest -# import unittest.mock +import unittest.mock from PyQt4 import QtGui, QtTest, QtCore from qubesadmin import Qubes @@ -33,6 +33,12 @@ class BackupTest(unittest.TestCase): def setUp(self): super(BackupTest, self).setUp() + # mock up nonexistence of saved backup settings + self.patcher = unittest.mock.patch('builtins.open') + self.mock_open = self.patcher.start() + self.mock_open.side_effect = FileNotFoundError() + self.addCleanup(self.patcher.stop) + self.qapp = Qubes() self.qtapp = QtGui.QApplication(sys.argv) self.dialog = backup_gui.BackupVMsWindow(self.qtapp, self.qapp) @@ -55,6 +61,30 @@ class BackupTest(unittest.TestCase): self.assertEqual(all_vms, available_vms + selected_vms) + def test_correct_defaults(self): + # backup is compressed + self.assertTrue(self.dialog.compress_checkbox.isChecked(), + "Compress backup should be checked by default") + + # correct VMs are selected + include_in_backups_no = len([vm for vm in self.qapp.domains + if not vm.features.get('internal', False) + and getattr(vm, 'include_in_backups', True)]) + selected_no = self.dialog.select_vms_widget.selected_list.count() + self.assertEqual(include_in_backups_no, selected_no, + "Incorrect VMs selected by default") + + # passphrase is empty + self.assertEqual(self.dialog.passphrase_line_edit.text(), "", + "Passphrase should be empty") + + # save defaults + self.assertTrue(self.dialog.save_profile_checkbox.isChecked(), + "By default, profile should be saved") + + # Check if target vms are selected + # Check if no default file loads correctly - another file?? + # TODO: make a separate backup testing file to test various backup defaults if __name__ == "__main__": ha_syslog = logging.handlers.SysLogHandler('/dev/log') From b0c7aed800da6e80554d09d0efff67061f5e469e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marta=20Marczykowska-G=C3=B3recka?= Date: Wed, 28 Mar 2018 17:42:42 +0200 Subject: [PATCH 6/8] Initials tests for qube manager --- qubesmanager/tests/test_qube_manager.py | 119 ++++++++++++++++++++++++ 1 file changed, 119 insertions(+) create mode 100644 qubesmanager/tests/test_qube_manager.py diff --git a/qubesmanager/tests/test_qube_manager.py b/qubesmanager/tests/test_qube_manager.py new file mode 100644 index 0000000..e105423 --- /dev/null +++ b/qubesmanager/tests/test_qube_manager.py @@ -0,0 +1,119 @@ +#!/usr/bin/python3 +# +# The Qubes OS Project, https://www.qubes-os.org/ +# +# Copyright (C) 2016 Marek Marczykowski-Górecki +# +# +# This program is free software; you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 2 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License along +# with this program; if not, write to the Free Software Foundation, Inc., +# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. +# +import logging.handlers +import sys +import unittest +import unittest.mock +import gc + +from PyQt4 import QtGui, QtTest, QtCore +from qubesadmin import Qubes +import qubesmanager.qube_manager as qube_manager + + +class QubeManagerTest(unittest.TestCase): + def setUp(self): + super(QubeManagerTest, self).setUp() + + # # todo: mockup no settings file + # self.patcher = unittest.mock.patch('builtins.open') + # self.mock_open = self.patcher.start() + # self.mock_open.side_effect = FileNotFoundError() + # self.addCleanup(self.patcher.stop) + + self.qapp = Qubes() + self.qtapp = QtGui.QApplication(sys.argv) + self.dialog = qube_manager.VmManagerWindow(self.qtapp, self.qapp) + + def tearDown(self): + del self.dialog + del self.qtapp + del self.qapp + super(QubeManagerTest, self).tearDown() + gc.collect() + + # 0 - Check if the window was displayed and populated correctly + + def test_00_window_loads(self): + self.assertTrue(self.dialog.table is not None) + + @unittest.expectedFailure + def test_01_table_populates_correctly(self): + vms_in_table = [] + for row in range(self.dialog.table.rowCount()): + item = self.dialog.table.item(row, + self.dialog.columns_indices["Name"]) + self.assertIsNotNone(item) + vms_in_table.append(item.text()) + + actual_vms = [vm.name for vm in self.qapp.domains] + + self.assertEqual(len(vms_in_table), len(actual_vms), + "Incorrect number of VMs loaded") + self.assertListEqual(sorted(vms_in_table), sorted(actual_vms), + "Incorrect VMs loaded") +# todos: + # did settings load correctly + # did settings save corectly + + @unittest.mock.patch('qubesmanager.settings.VMSettingsWindow') + def test_20_vm_open_settings(self, mock_window): + selected_vm = self._select_non_admin_vm() + self.assertIsNotNone(selected_vm, "No valid non-admin VM found") + widget = self.dialog.toolbar.widgetForAction( + self.dialog.action_settings) + QtTest.QTest.mouseClick(widget, + QtCore.Qt.LeftButton) + mock_window.assert_called_once_with(selected_vm, self.qtapp, "basic") + + @unittest.mock.patch('qubesmanager.settings.VMSettingsWindow') + def test_21_vm_firewall_settings(self, mock_window): + selected_vm = self._select_non_admin_vm() + self.assertIsNotNone(selected_vm, "No valid non-admin VM found") + widget = self.dialog.toolbar.widgetForAction( + self.dialog.action_editfwrules) + QtTest.QTest.mouseClick(widget, + QtCore.Qt.LeftButton) + mock_window.assert_called_once_with(selected_vm, self.qtapp, "firewall") + + +# test whether pause/start/resume works + @unittest.mock.patch('qubesmanager.qubesadmin.vm.QubesVM.pause') + @unittest.mock.patch('qubesmanager.qubesadmin.vm.QubesVM.is_running') + @unittest.mock.patch('qubesmanager.qubesadmin.vm.QubesVM.get_power_state') + + + def _select_non_admin_vm(self): + for row in range(self.dialog.table.rowCount()): + template = self.dialog.table.item( + row, self.dialog.columns_indices["Template"]) + if template.text() != 'AdminVM': + self.dialog.table.setCurrentItem(template) + return template.vm + return None + +if __name__ == "__main__": + ha_syslog = logging.handlers.SysLogHandler('/dev/log') + ha_syslog.setFormatter( + logging.Formatter('%(name)s[%(process)d]: %(message)s')) + logging.root.addHandler(ha_syslog) + unittest.main() From f99df0d577cbd25caa0b79663e3a669874759c55 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marta=20Marczykowska-G=C3=B3recka?= Date: Fri, 13 Apr 2018 06:44:24 +0200 Subject: [PATCH 7/8] Added instructions for tests --- qubesmanager/tests/instructions | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 qubesmanager/tests/instructions diff --git a/qubesmanager/tests/instructions b/qubesmanager/tests/instructions new file mode 100644 index 0000000..0773293 --- /dev/null +++ b/qubesmanager/tests/instructions @@ -0,0 +1,5 @@ +To run qube manager and backup tests: +python3 test_name.py -v + +To run global settings tests: +sudo systemctl stop qubesd; sudo -E python3 test_global_settings.py -v ; sudo systemctl start qubesd \ No newline at end of file From e18058039f4356148329c143bbfd6077e12db7b6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marta=20Marczykowska-G=C3=B3recka?= Date: Sun, 16 Sep 2018 18:04:56 +0200 Subject: [PATCH 8/8] Changes as requested by @marmarek --- README.md | 12 ++++++++++++ qubesmanager/tests/instructions | 5 ----- qubesmanager/tests/test_backup_01.py | 11 ++++++----- qubesmanager/tests/test_global_settings.py | 6 +++--- qubesmanager/tests/test_qube_manager.py | 6 ++---- rpm_spec/qmgr.spec.in | 6 ++++++ setup.py | 2 +- 7 files changed, 30 insertions(+), 18 deletions(-) delete mode 100644 qubesmanager/tests/instructions diff --git a/README.md b/README.md index 8071c89..dafd105 100644 --- a/README.md +++ b/README.md @@ -19,3 +19,15 @@ This will keep translated strings, but will add new ones. ### Updating translations Commit updated `.ts` files into `i18n` directory. + + +Tests +---------------------- + +Located in the tests/ directory. + +To run qube manager and backup tests: + python3 test_name.py -v + +To run global settings tests: + sudo systemctl stop qubesd; sudo -E python3 test_global_settings.py -v ; sudo systemctl start qubesd diff --git a/qubesmanager/tests/instructions b/qubesmanager/tests/instructions deleted file mode 100644 index 0773293..0000000 --- a/qubesmanager/tests/instructions +++ /dev/null @@ -1,5 +0,0 @@ -To run qube manager and backup tests: -python3 test_name.py -v - -To run global settings tests: -sudo systemctl stop qubesd; sudo -E python3 test_global_settings.py -v ; sudo systemctl start qubesd \ No newline at end of file diff --git a/qubesmanager/tests/test_backup_01.py b/qubesmanager/tests/test_backup_01.py index 1aa8901..3470061 100644 --- a/qubesmanager/tests/test_backup_01.py +++ b/qubesmanager/tests/test_backup_01.py @@ -2,8 +2,8 @@ # # The Qubes OS Project, https://www.qubes-os.org/ # -# Copyright (C) 2016 Marek Marczykowski-Górecki -# +# Copyright (C) 2016 Marta Marczykowska-Górecka +# # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by @@ -49,10 +49,10 @@ class BackupTest(unittest.TestCase): del self.qapp super(BackupTest, self).tearDown() - def test_window_loads(self): + def test_00_window_loads(self): self.assertTrue(self.dialog.select_vms_widget is not None) - def test_vms_load_correctly(self): + def test_01_vms_load_correctly(self): all_vms = len([vm for vm in self.qapp.domains if not vm.features.get('internal', False)]) @@ -61,7 +61,7 @@ class BackupTest(unittest.TestCase): self.assertEqual(all_vms, available_vms + selected_vms) - def test_correct_defaults(self): + def test_02_correct_defaults(self): # backup is compressed self.assertTrue(self.dialog.compress_checkbox.isChecked(), "Compress backup should be checked by default") @@ -86,6 +86,7 @@ class BackupTest(unittest.TestCase): # Check if no default file loads correctly - another file?? # TODO: make a separate backup testing file to test various backup defaults + if __name__ == "__main__": ha_syslog = logging.handlers.SysLogHandler('/dev/log') ha_syslog.setFormatter( diff --git a/qubesmanager/tests/test_global_settings.py b/qubesmanager/tests/test_global_settings.py index 4be171a..dbccd5d 100644 --- a/qubesmanager/tests/test_global_settings.py +++ b/qubesmanager/tests/test_global_settings.py @@ -2,8 +2,8 @@ # # The Qubes OS Project, https://www.qubes-os.org/ # -# Copyright (C) 2016 Marek Marczykowski-Górecki -# +# Copyright (C) 2016 Marta Marczykowska-Górecka +# # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by @@ -231,7 +231,7 @@ class GlobalSettingsTest(SystemTestCase): self.assertEqual(self.app.default_template.name, new_def_template_name) @wrap_in_loop - def test__50_set_default_kernel(self): + def test_50_set_default_kernel(self): new_def_kernel_name = self.__set_noncurrent( self.dialog.default_kernel_combo) self.__click_ok() diff --git a/qubesmanager/tests/test_qube_manager.py b/qubesmanager/tests/test_qube_manager.py index e105423..46cd16c 100644 --- a/qubesmanager/tests/test_qube_manager.py +++ b/qubesmanager/tests/test_qube_manager.py @@ -2,8 +2,8 @@ # # The Qubes OS Project, https://www.qubes-os.org/ # -# Copyright (C) 2016 Marek Marczykowski-Górecki -# +# Copyright (C) 2016 Marta Marczykowska-Górecka +# # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by @@ -100,8 +100,6 @@ class QubeManagerTest(unittest.TestCase): @unittest.mock.patch('qubesmanager.qubesadmin.vm.QubesVM.pause') @unittest.mock.patch('qubesmanager.qubesadmin.vm.QubesVM.is_running') @unittest.mock.patch('qubesmanager.qubesadmin.vm.QubesVM.get_power_state') - - def _select_non_admin_vm(self): for row in range(self.dialog.table.rowCount()): template = self.dialog.table.item( diff --git a/rpm_spec/qmgr.spec.in b/rpm_spec/qmgr.spec.in index 2a44416..c2de6bf 100644 --- a/rpm_spec/qmgr.spec.in +++ b/rpm_spec/qmgr.spec.in @@ -113,6 +113,12 @@ rm -rf $RPM_BUILD_ROOT %{python3_sitelib}/qubesmanager/i18n/qubesmanager_*.qm %{python3_sitelib}/qubesmanager/i18n/qubesmanager_*.ts +%{python3_sitelib}/qubesmanager/tests/__pycache__ +%{python3_sitelib}/qubesmanager/tests/__init__.py +%{python3_sitelib}/qubesmanager/tests/test_backup_01.py +%{python3_sitelib}/qubesmanager/tests/test_global_settings.py +%{python3_sitelib}/qubesmanager/tests/test_qube_manager.py + %dir %{python3_sitelib}/qubesmanager-*.egg-info %{python3_sitelib}/qubesmanager-*.egg-info/* diff --git a/setup.py b/setup.py index f0388a8..721284c 100644 --- a/setup.py +++ b/setup.py @@ -12,7 +12,7 @@ if __name__ == '__main__': description='Qubes OS Manager', license='GPL2+', url='https://www.qubes-os.org/', - packages=setuptools.find_packages(exclude=['qubesmanager.tests']), + packages=setuptools.find_packages(), package_data={ 'qubesmanager': ['i18n/*'] },