2011-10-14 14:14:24 +00:00
|
|
|
Unit tests
|
|
|
|
==========
|
|
|
|
|
2017-10-13 21:48:04 +00:00
|
|
|
Tests are located in the `kivy/tests` folder. If you find a bug in Kivy, a good
|
2013-05-12 18:00:01 +00:00
|
|
|
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
|
|
|
|
as a `unittest <http://docs.python.org/2/library/unittest.html>`_
|
2017-10-13 21:48:04 +00:00
|
|
|
, it will prevent the bug from coming back unnoticed in the future, and wil
|
|
|
|
make Kivy a better, stronger project. Writing a unittest may be a really good
|
|
|
|
way to get familiar with Kivy while doing something useful.
|
2012-05-25 16:10:49 +00:00
|
|
|
|
2015-12-14 03:44:31 +00:00
|
|
|
Unit tests are separated into two cases:
|
2011-10-14 14:14:24 +00:00
|
|
|
|
2015-12-09 21:29:25 +00:00
|
|
|
* Non graphical unit tests: these are standard unit tests that can run in a
|
2013-05-12 18:00:01 +00:00
|
|
|
console
|
2017-10-13 21:48:04 +00:00
|
|
|
* Graphical unit tests: these need a GL context, and if requested, work via
|
|
|
|
image comparison
|
2011-10-14 14:14:24 +00:00
|
|
|
|
2019-03-09 12:48:56 +00:00
|
|
|
To be able to run unit tests, you need to install pytest (https://pytest.org/),
|
|
|
|
and coverage (http://nedbatchelder.com/code/coverage/). You can use pip for
|
|
|
|
that::
|
2011-10-14 14:14:24 +00:00
|
|
|
|
2019-03-09 12:48:56 +00:00
|
|
|
sudo pip install pytest coverage
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
Then, in the kivy directory::
|
|
|
|
|
|
|
|
make test
|
|
|
|
|
2013-03-01 13:31:05 +00:00
|
|
|
How it works
|
|
|
|
------------
|
2011-10-14 14:14:24 +00:00
|
|
|
|
2013-02-19 18:19:36 +00:00
|
|
|
All the tests are located in `kivy/tests`, and the filename starts with
|
2019-03-09 12:48:56 +00:00
|
|
|
`test_<name>.py`. Pytest will automatically gather all the files and classes
|
2013-02-19 18:19:36 +00:00
|
|
|
inside this folder, and use them to generate test cases.
|
2011-10-14 14:14:24 +00:00
|
|
|
|
2013-02-19 18:19:36 +00:00
|
|
|
To write a test, create a file that respects the previous naming, then
|
|
|
|
start with this template::
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
import unittest
|
|
|
|
|
|
|
|
class XXXTestCase(unittest.TestCase):
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
# import class and prepare everything here.
|
|
|
|
pass
|
|
|
|
|
|
|
|
def test_YYY(self):
|
|
|
|
# place your test case here
|
|
|
|
a = 1
|
|
|
|
self.assertEqual(a, 1)
|
|
|
|
|
2013-02-19 18:19:36 +00:00
|
|
|
Replace `XXX` with an appropriate name that covers your tests cases, then
|
2013-05-12 18:00:01 +00:00
|
|
|
replace 'YYY' with the name of your test. If you have any doubts, check how
|
2013-05-18 17:54:41 +00:00
|
|
|
the other tests have been written.
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
Then, to execute them, just run::
|
|
|
|
|
|
|
|
make test
|
|
|
|
|
|
|
|
If you want to execute that file only, you can run::
|
|
|
|
|
2019-03-09 12:48:56 +00:00
|
|
|
pytest kivy/tests/test_yourtestcase.py
|
2011-10-14 14:14:24 +00:00
|
|
|
|
2017-10-13 21:48:04 +00:00
|
|
|
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
|
2019-03-09 12:48:56 +00:00
|
|
|
with `pytest` or via `unittest.main()`.
|
2017-10-13 21:48:04 +00:00
|
|
|
|
|
|
|
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.
|
|
|
|
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
GL unit tests
|
2017-10-13 21:48:04 +00:00
|
|
|
~~~~~~~~~~~~~
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
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
|
2013-05-12 18:00:01 +00:00
|
|
|
used. For these tests, the goal is to save the output of the rendering at
|
2011-10-14 14:14:24 +00:00
|
|
|
frame X, and compare it to a reference image.
|
|
|
|
|
2013-05-12 18:00:01 +00:00
|
|
|
Currently, images are generated at 320x240 pixels, in *png* format.
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
.. note::
|
|
|
|
|
2012-07-08 09:36:49 +00:00
|
|
|
Currently, image comparison is done per-pixel. This means the reference
|
|
|
|
image that you generate will only be correct for your GPU/driver. If
|
|
|
|
somebody can implement image comparison with "delta" support, patches
|
|
|
|
are welcome :)
|
2011-10-14 14:14:24 +00:00
|
|
|
|
2013-05-12 18:00:01 +00:00
|
|
|
To execute GL unit tests, you need to create a directory::
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
mkdir kivy/tests/results
|
2017-10-13 21:48:04 +00:00
|
|
|
KIVY_UNITTEST_SCREENSHOTS=1 make test
|
2011-10-14 14:14:24 +00:00
|
|
|
|
2013-05-12 18:00:01 +00:00
|
|
|
The results directory will contain all the reference images and the
|
2013-02-19 18:19:36 +00:00
|
|
|
generated images. After the first execution, if the results directory is empty,
|
|
|
|
no comparison will be done. It will use the generated images as reference.
|
|
|
|
After the second execution, all the images will be compared to the reference
|
|
|
|
images.
|
2011-10-14 14:14:24 +00:00
|
|
|
|
2013-02-19 18:19:36 +00:00
|
|
|
A html file is available to show the comparison before/after the test, and a
|
2011-10-14 14:14:24 +00:00
|
|
|
snippet of the associated unit test. It will be generated at:
|
|
|
|
|
|
|
|
kivy/tests/build/index.html
|
|
|
|
|
|
|
|
.. note::
|
|
|
|
|
|
|
|
The build directory is cleaned after each call to `make test`. If you don't
|
2019-03-09 12:48:56 +00:00
|
|
|
want that, just use pytest command.
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
Writing GL Unit tests
|
|
|
|
---------------------
|
|
|
|
|
|
|
|
The idea is to create a root widget, as you would do in
|
2014-01-08 14:23:35 +00:00
|
|
|
:meth:`~kivy.app.App.build`, or in :func:`kivy.base.runTouchApp`.
|
2013-05-12 18:00:01 +00:00
|
|
|
You'll give that root widget to a rendering function which will capture the
|
2011-10-14 14:14:24 +00:00
|
|
|
output in X frames.
|
|
|
|
|
|
|
|
Here is an example::
|
|
|
|
|
2017-10-13 21:48:04 +00:00
|
|
|
from kivy.tests.common import GraphicUnitTest
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
class VertexInstructionTestCase(GraphicUnitTest):
|
|
|
|
|
|
|
|
def test_ellipse(self):
|
|
|
|
from kivy.uix.widget import Widget
|
|
|
|
from kivy.graphics import Ellipse, Color
|
|
|
|
r = self.render
|
|
|
|
|
|
|
|
# create a root widget
|
|
|
|
wid = Widget()
|
|
|
|
|
|
|
|
# put some graphics instruction on it
|
|
|
|
with wid.canvas:
|
|
|
|
Color(1, 1, 1)
|
|
|
|
self.e = Ellipse(pos=(100, 100), size=(200, 100))
|
|
|
|
|
|
|
|
# render, and capture it directly
|
|
|
|
r(wid)
|
|
|
|
|
|
|
|
# as alternative, you can capture in 2 frames:
|
|
|
|
r(wid, 2)
|
|
|
|
|
|
|
|
# or in 10 frames
|
|
|
|
r(wid, 10)
|
|
|
|
|
2013-02-19 18:19:36 +00:00
|
|
|
Each call to `self.render` (or `r` in our example) will generate an image named
|
2013-05-12 18:00:01 +00:00
|
|
|
as follows::
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
<classname>_<funcname>-<r-call-count>.png
|
|
|
|
|
2013-05-12 18:00:01 +00:00
|
|
|
`r-call-count` represents the number of times that `self.render` is called
|
|
|
|
inside the test function.
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
The reference images are named::
|
|
|
|
|
|
|
|
ref_<classname>_<funcname>-<r-call-count>.png
|
|
|
|
|
2013-05-12 18:00:01 +00:00
|
|
|
You can easily replace the reference image with a new one if you wish.
|
2011-10-14 14:14:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
Coverage reports
|
|
|
|
----------------
|
|
|
|
|
2013-05-12 18:00:01 +00:00
|
|
|
Coverage is based on the execution of previous tests. Statistics on code
|
|
|
|
coverage are automatically calculated during execution. You can generate an html
|
2011-10-14 14:14:24 +00:00
|
|
|
report of the coverage with the command::
|
|
|
|
|
|
|
|
make cover
|
|
|
|
|
|
|
|
Then, open `kivy/htmlcov/index.html` with your favorite web browser.
|