qubes-installer-qubes-os/anaconda/tests/gui/inside/__init__.py
Marek Marczykowski-Górecki 3e63d1dd37 anaconda: update to 21.48.21-1
Apply diff anaconda-20.25.16-1..anaconda-21.48.21-1
2016-03-22 02:27:15 +13:00

183 lines
7.5 KiB
Python

#!/usr/bin/python
#
# Copyright (C) 2014 Red Hat, Inc.
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published
# by the Free Software Foundation; either version 2.1 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 Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
# Author: Chris Lumens <clumens@redhat.com>
from dogtail.config import config
from dogtail.predicate import GenericPredicate
from dogtail.tree import SearchError, root
from dogtail.utils import doDelay, screenshot
import glob
import os
import unittest
class UITestCase(unittest.TestCase):
"""A UITestCase is a class that incorporates all the test functions for a
single anaconda Hub or Spoke window. Moving to the window, and dealing
with what happens after we move away is left up to whatever wraps up
all the test cases. A single Hub or Spoke may have multiple instances
of this class - one may test the assumption that everything works as it
should, while another may test that a specific initial setup fails in
an expected way, and another may attempt fuzz testing on all entries on
the screen. However, a single TestSuite will only have one instance
for each anaconda window.
Some basic assumptions about the existence of UI elements are made in
UITestCase subclasses. If an element is required to be there (for
instance, a button we are going to click) we simply grab the element
with the exception-raising dogtail functions and call click on it.
If the element does not exist, the exception will be propagated up and
fail the test. If we are testing for the existence of some element
(for instance, that a dialog was displayed after a button was clicked)
then we use UITestCase.find and the unittest.assert* functions.
This is kind of subtle, but look at existing test cases for examples.
Due to the unusual nature of running such a large program as anaconda
in a test suite, combined with the special environment, tests are
organized inside a UITestCase as follows:
* Testing one little piece of the window is handled by a check_*
function. These are explicitly not named test_* as that is special
to unittest.
* Multiple check_* functions are called linearly in the _run method,
which is used instead of runTest for error handling purposes.
"""
###
### OVERRIDES OF THINGS FROM TestCase
###
def runTest(self):
"""A version of TestCase.runTest that attempts to take a screenshot of
anaconda should a test fail. Subclasses should not override this.
See the documentation for _run.
"""
config.load({"scratchDir": "/mnt/anactest/result/"})
try:
self._run()
except (AssertionError, SearchError):
# Try to take a screenshot of whatever screen anaconda's on, so
# we can attempt to figure out what went wrong.
screenshot()
raise
def _run(self):
"""Do all the tests for this test case. This is like the TestCase.runTest
method, but we've overridden that to do more specialized error handling.
Thus, all testing should be done in this method.
"""
pass
def setUp(self):
self.ana = root.application("anaconda")
###
### METHODS FOR FINDING WIDGETS
###
def find(self, name, roleName=None):
"""Wrap findChild, returning None if no widget is found instead of
raising an exception. This method also allows for checking if
anaconda has hit a traceback and if so, fails the test
immediately.
"""
if len(glob.glob("/tmp/anaconda-tb-*")) > 0:
self.fail("anaconda encountered a traceback")
try:
return self.ana.child(name=name, roleName=roleName)
except SearchError:
return None
###
### METHODS FOR CHECKING A SINGLE WIDGET
###
def check_window_displayed(self, name):
"""Verify that a window (such as a hub or spoke) given by the
provided name is currently displayed on the screen. If not,
the current test case will be failed.
"""
w = self.find(name)
self.assertIsNotNone(w, msg="%s not found" % name)
self.assertTrue(w.showing, msg="%s is not displayed" % name)
def check_dialog_displayed(self, name):
"""Verify that a dialog given by the provided name is currently
displayed on the screen. If not, the current test case will
be failed.
"""
w = self.find(name, "dialog")
self.assertIsNotNone(w, msg="%s not found" % name)
self.assertTrue(w.showing, msg="%s is not displayed" % name)
def check_keyboard_layout_indicator(self, layout):
"""Verify that the keyboard layout indicator is present and that
the currently enabled layout is what we expect. If not, the
current test case will be failed.
"""
indicator = self.find("Keyboard Layout")
self.assertIsNotNone(indicator, msg="keyboard layout indicator not found")
self.assertEqual(indicator.description, layout,
msg="keyboard layout indicator not set to %s" % layout)
def check_no_warning_bar(self):
"""Verify that the warning bar is not currently displayed."""
self.assertIsNone(self.find("Warning"), msg="Warning bar should not be displayed")
def check_warning_bar(self, msg=None):
"""Verify that the warning bar is currently displayed. If msg is given,
verify that it is contained in whatever message the warning bar is
showing.
"""
bar = self.find("Warning")
self.assertTrue(bar.showing, msg="Warning bar should be displayed")
if msg:
self.assertIn(msg, bar.child(roleName="label").text)
def click_button(self, name):
"""Verify that a button with the given name exists and is sensitive,
and then click it.
"""
b = self.find(name, "push button")
self.assertIsNotNone(b, msg="%s button not found" % name)
self.assertTrue(b.sensitive, msg="%s button should be sensitive" % name)
b.click()
def enter_spoke(self, spokeSelectorName):
"""Click on the spoke selector for the given spoke, then wait a moment
to make sure it has appeared.
"""
selector = self.find(spokeSelectorName, "spoke selector")
self.assertIsNotNone(selector, msg="Selector %s not found" % spokeSelectorName)
selector.click()
def exit_spoke(self, hubName="INSTALLATION SUMMARY"):
"""Leave a spoke by clicking the Done button in the upper left corner,
then verify we have returned to the proper hub. Since most spokes
are off the summary hub, that's the default. If we are not back
on the hub, the current test case will be failed.
"""
button = self.find("_Done", "push button")
self.assertIsNotNone(button, msg="Done button not found")
button.click()
doDelay(5)
self.check_window_displayed(hubName)