Модульні Тести
Розробникам рекомендується писати модульні тести на всіх етапах розробки, включаючи додавання нових функцій, виправлення помилок і рефакторинг
PX4 надає декілька методів для написання юніт тестів:
- Модульні тести з Google Test ("GTest") – тести, які мають мінімальні внутрішні залежності
- Функціональні тести з GTest - тести, які залежать від параметрів та uORB повідомлень
- SITL unit tests. This is for tests that need to run in full SITL. Ці тести виконуються набагато повільніше та важче налагодити, тому, якщо можливо, замість них рекомендується використовувати GTest.
Написання GTest Unit Test
Порада: загалом, якщо вам потрібен доступ до розширених утиліт GTest, структур даних із STL або потрібно зв’язатися з параметрами
чи бібліотеками uorb
, натомість слід використовувати функціональні тести.
Кроки для створення нових функціональних тестів такі:
- Unit tests should be arranged in three sections: setup, run, check results. Each test should test one very specific behavior or setup case, so if a test fails it is obvious what is wrong. Please try to follow these standards when possible.
- Copy and rename the example unit test AttitudeControlTest to the directory the code to be tested is in.
- Add the new file to the directory's
CMakeLists.txt
. It should look something likepx4_add_unit_gtest(SRC MyNewUnitTest.cpp LINKLIBS <library_to_be_tested>)
- Add the desired test functionality. This will mean including the header files required for your specific tests, adding new tests (each with an individual name) and putting the logic for the setup, running the code to be tested and verifying that it behaves as expected.
- If additional library dependencies are required, they should also be added to the CMakeLists after the
LINKLIBS
as shown above.
Тести можна запустити за допомогою make tests
, після чого ви знайдете двійковий файл у build/px4_sitl_test/unit-MyNewUnit
. Він може бути запущений безпосередньо в налагоджувачі.
Написання GTest Functional Test
Функціональні тести GTest слід використовувати, коли тест або компоненти, що тестуються, залежать від параметрів, повідомлень uORB та/або розширеної функціональності GTest. Крім того, функціональні тести можуть містити локальне використання структур даних STL (хоча і будьте обережні відмінності платформ між такими як macOS і Linux).
Кроки для створення нових функціональних тестів такі:
- In general (and similar to unit tests), functional tests should be arranged in three sections: setup, run, check results. Each test should test one very specific behavior or setup case, so if a test fails it is obvious what is wrong. Please try to follow these standards when possible.
- Copy and rename the example functional test ParameterTest to the directory the code to be tested is in.
- Перейменуйте клас з ParameterTest на те, що краще представляє код, що тестується
- Add the new file to the directory's
CMakeLists.txt
. It should look something likepx4_add_functional_gtest(SRC MyNewFunctionalTest.cpp LINKLIBS <library_to_be_tested>)
- Add the desired test functionality. This will mean including the header files required for your specific tests, adding new tests (each with an individual name) and putting the logic for the test setup, running the code to be tested and verifying that it behaves as expected.
- If additional library dependencies are required, they should also be added to the CMakeLists after the
LINKLIBS
as shown above.
Tests can be run via make tests
, after which you will find the binary in build/px4_sitl_test/functional-MyNewFunctional
. It can be run directly in a debugger, however be careful to only run one test per executable invocation using the --gtest_filter=<regex> arguments, as some parts of the uORB and parameter libraries don't clean themselves up perfectly and may result in undefined behavior if set up multiple times.
Написання SITL Unit Test
SITL unit tests should be used when you specifically need all of the flight controller components - drivers, time, and more. These tests are slower to run (1s+ for each new module), and harder to debug, so in general they should only be used when necessary.
Кроки для створення нових модульних тестів SITL такі:
Examine the sample Unittest-class.
Create a new .cpp file within tests with name test_[description].cpp.
Within test_[description].cpp include the base unittest-class
<unit_test.h>
and all files required to write a test for the new feature.Within test_[description].cpp create a class
[Description]Test
that inherits fromUnitTest
.Within
[Description]Test
class declare the public methodvirtual bool run_tests()
.Within
[Description]Test
class declare all private methods required to test the feature in question (test1()
,test2()
,...).Within test_[description].cpp implement the
run_tests()
method where each test[1,2,...] will be run.Within test_[description].cpp, implement the various tests.
At the bottom within test_[description].cpp declare the test.
cpput_declare_test_c(test_[description], [Description]Test)
Here is a template:
cpp#include <unit_test.h> #include "[new feature].h" ... class [Description]Test : public UnitTest { public: virtual bool run_tests(); private: bool test1(); bool test2(); ... }; bool [Description]Test::run_tests() { ut_run_test(test1) ut_run_test(test2) ... return (_tests_failed == 0); } bool [Description]Test::test1() { ut_[name of one of the unit test functions](... ut_[name of one of the unit test functions](... ... return true; } bool [Description]Test::test2() { ut_[name of one of the unit test functions](... ut_[name of one of the unit test functions](... ... return true; } ... ut_declare_test_c(test_[description], [Description]Test)
Note that
ut_[name of one of the unit test functions]
corresponds to one of the unittest functions defined within unit_test.h.Within tests_main.h define the new test:
cppextern int test_[description](int argc, char *argv[]);
Within tests_main.c add description name, test function and option:
cpp... } tests[] = { {... {"[description]", test_[description], OPTION}, ... }
OPTION
can beOPT_NOALLTEST
,OPT_NOJIGTEST
or0
and is considered if within px4 shell one of the two commands are called:shpxh> tests all
or
shpxh> tests jig
If a test has option
OPT_NOALLTEST
, then that test will be excluded when callingtests all
. The same is true forOPT_NOJITEST
when commandtest jig
is called. Option0
means that the test is never excluded, which is what most developer want to use.Add the test
test_[description].cpp
to the CMakeLists.txt.
Тестування на локальній машині
Запустіть повний список модульних тестів GTest, функціональних тестів GTest і модульних тестів SITL прямо з bash:
sh
make tests
Окремі тестові двійкові файли GTest знаходяться в каталозі build/px4_sitl_test/
і можуть бути запущені безпосередньо в налагоджувачі більшості IDE.
Фільтр, щоб запустити лише підмножину тестів, використовуючи регулярний вираз для імені ctest за допомогою цієї команди:
sh
make tests TESTFILTER=<regex filter expression>
Наприклад:
make tests TESTFILTER=unit
only run GTest unit testsmake tests TESTFILTER=sitl
only run simulation testsmake tests TESTFILTER=Attitude
only run theAttitudeControl
test