mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-17 18:52:44 +01:00
cedfb63a88
The aim is to allow faster test cycles on native for unit test style apps (that don't need interaction) by bypassing the python test framework using e.g.: make RIOT_TERMINAL=native all term This would work already before, but now is more convenient as no manual press of the `s` key is needed to start the test. For non-native boards we need the sync, as otherwise the board may finish booting before the python test automation framework can capture output. For `native` and `native64`, we actually control when the RIOT app is started and do not need to sync. On a typical machine this can reduce the test cycle by more than 4 seconds. With this change: $ time sh -c 'make BOARD=native -C tests/unittests tests-nanocoap -j && make BOARD=native RIOT_TERMINAL=native -C tests/unittests term' [...] main(): This is RIOT! (Version: 2024.10-devel-394-gd65dec-tests/no-sync-control) ................................... OK (35 tests) [...] make: Leaving directory '/home/marian.buschsieweke@ml-pa.loc/Repos/software/RIOT/master/tests/unittests' sh -c 0.30s user 0.24s system 113% cpu 0.476 total Before t his change: $ time sh -c 'make BOARD=native -C tests/unittests tests-nanocoap -j && make BOARD=native -C tests/unittests test' [...] main(): This is RIOT! (Version: 2024.10-devel-394-gd65dec-tests/no-sync-control) Help: Press s to start test, r to print it is ready READY s START ................................... OK (35 tests) [...] make: Leaving directory '/home/marian.buschsieweke@ml-pa.loc/Repos/software/RIOT/master/tests/unittests' sh -c 0.50s user 0.37s system 17% cpu 4.863 total
204 lines
8.6 KiB
Markdown
204 lines
8.6 KiB
Markdown
# Running and creating tests {#running-and-creating-tests}
|
|
|
|
There are a number of tests included in RIOT. They are located in the
|
|
[tests directory](https://github.com/RIOT-OS/RIOT/tree/master/tests). These tests
|
|
allow basic functionality to be verified as well as provide an example of
|
|
usage.
|
|
|
|
# Directory Structure
|
|
|
|
The [tests directory](https://github.com/RIOT-OS/RIOT/tree/master/tests) in RIOT
|
|
is further divided into a number of subdirectories.
|
|
|
|
- [*bench*](https://github.com/RIOT-OS/RIOT/tree/master/tests/bench): Benchmark
|
|
tests, these provide numbers on how RIOT performs on the used hardware.
|
|
- [*build_system*](https://github.com/RIOT-OS/RIOT/tree/master/tests/build_system): Tests
|
|
the RIOT build system functionality, such as blob, external board/module/package dirs,
|
|
and kconfig.
|
|
- [*core*](https://github.com/RIOT-OS/RIOT/tree/master/tests/core): Tests the
|
|
RIOT core functionality such as threading and IPC.
|
|
- [*cpu*](https://github.com/RIOT-OS/RIOT/tree/master/tests/cpu): Tests
|
|
RIOT cpu specific features such as efm32, stm32, native and AVR.
|
|
- [*drivers*](https://github.com/RIOT-OS/RIOT/tree/master/tests/drivers): Tests
|
|
individual drivers. The tests for sensors print the measured values to the
|
|
console, others demonstrate the functionality of the driver and attached
|
|
hardware.
|
|
- [*net*](https://github.com/RIOT-OS/RIOT/tree/master/tests/net): Tests the
|
|
networking features provided in RIOT, such as CoAP, emcute, GNRC, IEEE 802.15.4
|
|
and sntp.
|
|
- [*periph*](https://github.com/RIOT-OS/RIOT/tree/master/tests/periph): Tests the
|
|
low level peripherals in RIOT, such as interacting with SPI and I2C
|
|
peripherals.
|
|
- [*pkg*](https://github.com/RIOT-OS/RIOT/tree/master/tests/pkg): Tests the
|
|
external packages available in RIOT, such as lvgl, lwip, nanocbor, and tinyusb.
|
|
- [*sys*](https://github.com/RIOT-OS/RIOT/tree/master/tests/sys): Collection
|
|
of tests for the utilities in
|
|
[*sys*](https://github.com/RIOT-OS/RIOT/tree/master/sys) directory of RIOT.
|
|
- [*unittests*](https://github.com/RIOT-OS/RIOT/tree/master/tests/unittests):
|
|
Collection of very simple test applications that test simple modules and do
|
|
not rely on extra hardware. Can be flashed and run as single application to
|
|
test all unit tests at once.
|
|
|
|
# Running automated tests
|
|
|
|
Some tests can be performed automatically. The test automation scripts are
|
|
defined in the `<test_application>/tests/` folder. They are written in python
|
|
and interact through the serial (typically UART) with the test application code running on a
|
|
board to do the validation. It is recommended to flash the board with the
|
|
test just before running it because some platforms cannot be reset while
|
|
testing.
|
|
|
|
## Running single test
|
|
|
|
From the test application directory run:
|
|
|
|
BOARD=<board_of_your_choice> make flash test
|
|
|
|
|
|
An automated way of knowing if a test is available is to execute the
|
|
'test/available' target from the test application directory.
|
|
It executes without error if tests run by 'make test' are present.
|
|
|
|
make test/available
|
|
|
|
## Running all test for particular board
|
|
|
|
If you would like execute all tests for given board, you could use dedicated
|
|
script `compile_and_test_for_board.py`
|
|
|
|
Go to main RIOT directory and execute command:
|
|
|
|
./dist/tools/compile_and_test_for_board/compile_and_test_for_board.py . <board_of_your_choice> --with-test-only --jobs=4
|
|
|
|
More details concerning other available parameters provided by this tool can be found in
|
|
[README.md file](https://github.com/RIOT-OS/RIOT/tree/master/dist/tools/compile_and_test_for_board)
|
|
and directly in [compile_and_test_for_board.py](https://github.com/RIOT-OS/RIOT/tree/master/dist/tools/compile_and_test_for_board/compile_and_test_for_board.py) script.
|
|
|
|
## Running tests that require a preliminary manual configuration
|
|
|
|
Some tests need active monitoring or manual setup steps but still have some
|
|
automated scripts. The test automation scripts are defined in the
|
|
`<test_application>/tests-with-config/` folder.
|
|
For running them, follow the setup or analysis documentation and use the
|
|
`test-with-config` target.
|
|
|
|
## Running tests that require root privileges
|
|
|
|
Some tests require root privileges to launch their automated script. In this
|
|
case, the test automation scripts are defined in the
|
|
`<test_application>/tests-as-root/` folder.
|
|
For running them, follow the setup or analysis documentation and use the
|
|
`test-as-root` target.
|
|
|
|
## Cleaning intermediate files
|
|
|
|
After test execution intermediate files are not automatically deleted.
|
|
Execution of multiple tests, especially all for particular board could generate
|
|
many files. For example, after execution of all test for stm32f469i-disco board
|
|
(more than 230 tests) around 7.5 GB of intermediate files are created.
|
|
|
|
There are few methods for cleaning intermediate files.
|
|
|
|
If you would like to clean intermediate file only for particular board you should
|
|
go to main RIOT directory and execute one from these commands:
|
|
|
|
./dist/tools/compile_and_test_for_board/compile_and_test_for_board.py . <board_of_your_choice> --compile-targets clean
|
|
|
|
or
|
|
|
|
make BOARD=<board_of_your_choice> clean
|
|
|
|
|
|
If you would like to clean intermediate files for all boards go to main RIOT
|
|
directory and use this command.
|
|
|
|
@warning This command cleans all local files, for example, pkg downloads and
|
|
locally generared docs.
|
|
|
|
|
|
make distclean
|
|
|
|
# Implementing automated tests
|
|
|
|
The goal is to be able to run all tests in a sequential way for as many targets
|
|
as possible.
|
|
|
|
As some board can't be reset without a manual trigger tests should be implemented
|
|
with some kind of `synchronization`. This can be done in two ways:
|
|
|
|
- use `test_utils_interactive_sync` when uart input/output does not need to be
|
|
disabled for the test. This is enabled by default, except for `native` and
|
|
`native64`.
|
|
- set up the test in a loop so the test script will be able so sync with some kind
|
|
of start condition in the test.
|
|
|
|
The module for the first option is `test_utils_interactive_sync` and is set as a
|
|
default module in `Makefile.tests_common`. It can be disabled by setting in the
|
|
application makefile `DISABLE_MODULE += test_utils_interactive_sync`. The python
|
|
test script will adapt to it automatically.
|
|
|
|
When using the `shell` module, `test_utils_interactive_sync` will use the shell
|
|
itself to synchronize, and will not use `test_utils_interactive_sync();` function
|
|
to synchronize. Some times you will want to synchronize before the start of the
|
|
script and use `test_utils_interactive_sync();` function (e.g.:
|
|
[tests/ps_schedstatistics](tests/ps_schedstatistics/main.c)). For these cases
|
|
you can disable `test_utils_interactive_sync_shell` module in the application
|
|
`Makefile`: `DISABLE_MODULE += test_utils_interactive_sync_shell`.
|
|
|
|
## Automated Tests Guidelines
|
|
|
|
When using `pexpect` `$` is useless for matching the end of a line, instead use
|
|
`\r\n`([pexpect end-of-line](https://pexpect.readthedocs.io/en/stable/overview.html#find-the-end-of-line-cr-lf-conventions)).
|
|
|
|
Beware of `+` and `*` at the end of patterns. These patterns will always get
|
|
a minimal match (non-greedy).([pexpect end-of-patterns](https://pexpect.readthedocs.io/en/stable/overview.html#beware-of-and-at-the-end-of-patterns))
|
|
This can be an issue when matching groups and using the matched groups to verify
|
|
some kind of behavior since `*` could return an empty match and `+` only a subset.
|
|
|
|
This is especially prevalent since `printf()` is buffered so the output might not
|
|
arrive in a single read to `pexpect`.
|
|
|
|
To avoid this make sure to match a non-ambiguous character at the end of the
|
|
pattern like `\r\n`, `\s`, `\)`, etc..
|
|
|
|
**don't**:
|
|
|
|
~~~~
|
|
child.expect(r'some string: (\d+)')
|
|
~~~~
|
|
|
|
**do**:
|
|
|
|
~~~
|
|
child.expect(r'some string: (\d+)\r\n')
|
|
~~~
|
|
~~~
|
|
child.expect(r'some string: (\d+)\s')
|
|
~~~
|
|
~~~
|
|
child.expect(r'some string: (\d+) ,')
|
|
~~~
|
|
|
|
## Use expect() instead of assert()
|
|
|
|
In order to make a test application functional in all cases, use `expect()`
|
|
instead of `assert()`. The former works like the latter, but will still be
|
|
compiled in if `NDEBUG` is defined. This is useful to keep a test application
|
|
working even when compiling with -DNDEBUG, allowing for the code-under-test to
|
|
be compiled with that flag. Otherwise, the application would force compiling
|
|
all tested code with assertions enabled.
|
|
`expect()` is defined in the header `test_utils/expect.h`.
|
|
|
|
## Interaction through the uart
|
|
|
|
Tests implemented with `testrunner` use the `cleanterm` target that
|
|
provides an interaction without adding extra text output or input handling.
|
|
It can currently be expected to have unmodified line based interaction with the
|
|
board.
|
|
|
|
The expected behavior is verified with the test in `tests/test_tools`.
|
|
|
|
Tests cannot rely on having on all boards and terminal programs:
|
|
* unbuffered input
|
|
* allowing sending special characters like `ctrl+c/ctrl+d`
|