Docs: Add simple guide for GraphicUnitTest

This commit is contained in:
Peter Badida 2017-10-13 23:48:04 +02:00
parent b918f02abb
commit 04b7f068d0
1 changed files with 115 additions and 8 deletions

View File

@ -1,19 +1,20 @@
Unit tests Unit tests
========== ==========
Tests are located in the kivy/tests folder. If you find a bug in Kivy, a good Tests are located in the `kivy/tests` folder. If you find a bug in Kivy, a good
thing to do can be to write a minimal case showing the issue and to ask core thing to do can be to write a minimal case showing the issue and to ask core
devs if the behaviour shown is intended or a real bug. If you write your code devs if the behaviour shown is intended or a real bug. If you write your code
as a `unittest <http://docs.python.org/2/library/unittest.html>`_ as a `unittest <http://docs.python.org/2/library/unittest.html>`_
, it will prevent the bug from coming back unnoticed in the future, , it will prevent the bug from coming back unnoticed in the future, and wil
and will make Kivy a better, stronger project. Writing a unittest may be a make Kivy a better, stronger project. Writing a unittest may be a really good
really good way to get familiar with Kivy while doing something useful. way to get familiar with Kivy while doing something useful.
Unit tests are separated into two cases: Unit tests are separated into two cases:
* Non graphical unit tests: these are standard unit tests that can run in a * Non graphical unit tests: these are standard unit tests that can run in a
console console
* Graphical unit tests: these need a GL context, and work via image comparison * Graphical unit tests: these need a GL context, and if requested, work via
image comparison
To be able to run unit tests, you need to install nose To be able to run unit tests, you need to install nose
(http://code.google.com/p/python-nose/), and coverage (http://code.google.com/p/python-nose/), and coverage
@ -60,9 +61,115 @@ If you want to execute that file only, you can run::
nosetests kivy/tests/test_yourtestcase.py nosetests kivy/tests/test_yourtestcase.py
or include this simple `unittest.main()` call at the end of the file and run
the test with `python test_yourtestcase.py`::
if __name__ == '__main__':
unittest.main()
Graphical unit tests
--------------------
While simple unit tests are fine and useful to keep things granular, in certain
cases we need to test Kivy after the GL Window is created to interact with the
graphics, widgets and to test more advanced stuff such as widget, modules,
various cases of input and interaction with everything that becomes available
only after the Window is created and Kivy properly initialized.
These tests are executed the same way like the ordinary unit tests i.e. either
with `nosetests` or via `unittest.main()`.
Here are two similar examples with different approaches of running the app.
In the first one you are setting up the required stuff manually and the
`tearDown()` of the `GraphicUnitTest` may only attempt to clean it after you::
from kivy.tests.common import GraphicUnitTest
class MyTestCase(GraphicUnitTest):
def test_runtouchapp(self):
# non-integrated approach
from kivy.app import runTouchApp
from kivy.uix.button import Button
button = Button()
runTouchApp(button)
# get your Window instance safely
from kivy.base import EventLoop
EventLoop.ensure_window()
window = EventLoop.window
# your asserts
self.assertEqual(window.children[0], button)
self.assertEqual(
window.children[0].height,
window.height
)
In the second test case both `setUp()` and `tearDown()` work together with
`GraphicUnitTest.render()`. This is the basic setup it does automatically:
* Window is sized to 320 x 240 px
* Only the default Config is used during the test, it's restricted with the
`KIVY_USE_DEFAULTCONFIG` environment variable
* Any input (mouse/touch/...) is *removed* and if you need to test it, either
mock it or manually add it
* Window's canvas is cleared before displaying any widget tree
.. warning::
Do NOT use absolute numbers in your tests to preserve the functionality
across the all resolutions. Instead, use e.g. relative position or size and
multiply it by the `Window.size` in your test.
::
from kivy.tests.common import GraphicUnitTest, UnitTestTouch
class MyTestCase(GraphicUnitTest):
def test_render(self):
from kivy.uix.button import Button
# with GraphicUnitTest.render() you basically do this:
# runTouchApp(Button()) + some setup before
button = Button()
self.render(button)
# get your Window instance safely
from kivy.base import EventLoop
EventLoop.ensure_window()
window = EventLoop.window
touch = UnitTestTouch(
*[s / 2.0 for s in window.size]
)
# bind something to test the touch with
button.bind(
on_release=lambda instance: setattr(
instance, 'test_released', True
)
)
# then let's touch the Window's center
touch.touch_down()
touch.touch_up()
self.assertTrue(button.test_released)
if __name__ == '__main__':
import unittest
unittest.main()
.. note::
Make sure you check the source of `kivy.tests.common` before writing
comprehensive test cases.
GL unit tests GL unit tests
------------- ~~~~~~~~~~~~~
GL unit test are more difficult. You must know that even if OpenGL is a GL unit test are more difficult. You must know that even if OpenGL is a
standard, the output/rendering is not. It depends on your GPU and the driver standard, the output/rendering is not. It depends on your GPU and the driver
@ -81,7 +188,7 @@ Currently, images are generated at 320x240 pixels, in *png* format.
To execute GL unit tests, you need to create a directory:: To execute GL unit tests, you need to create a directory::
mkdir kivy/tests/results mkdir kivy/tests/results
make test KIVY_UNITTEST_SCREENSHOTS=1 make test
The results directory will contain all the reference images and the The results directory will contain all the reference images and the
generated images. After the first execution, if the results directory is empty, generated images. After the first execution, if the results directory is empty,
@ -109,7 +216,7 @@ output in X frames.
Here is an example:: Here is an example::
from common import GraphicUnitTest from kivy.tests.common import GraphicUnitTest
class VertexInstructionTestCase(GraphicUnitTest): class VertexInstructionTestCase(GraphicUnitTest):