ellensp ee1d1faa52 🔨 Sim pins / INI cleanup | 6 months ago | |
---|---|---|
.. | ||
001-default.ini | 7 months ago | |
002-extruders_1_runout.ini | 7 months ago | |
003-extruders_3_runout.ini | 6 months ago | |
README.md | 7 months ago | |
unit_tests.cpp | 7 months ago | |
unit_tests.h | 7 months ago |
Marlin included two types of automated tests:
This document focuses on Unit tests.
Unit testing allows for functional testing of Marlin logic on a local machine. This strategy is available to all developers, and will be able to be used on generic GitHub workers to automate testing. While PlatformIO does support the execution of unit tests on target controllers, that is not yet implemented and not really practical. This would require dedicated testing labs, and would be less broadly usable than testing directly on the development or build machine.
Unit tests verify the behavior of small discrete sections of Marlin code. By thoroughly unit testing important parts of Marlin code, we effectively provide "guard rails" which will prevent major regressions in behavior. As long as all submissions go through the Pull Request process and execute automated checks, it is possible to catch most major issues prior to completion of a PR.
Unit tests can be used to validate the logic of single functions or whole features, as long as that function or feature doesn't depend on real hardware. So, for example, we can test whether a G-code command is parsed correctly and produces all the expected state changes, but we can't test whether a G-code triggered an endstop or the filament runout sensor without adding a new layer to simulate pins.
Generally speaking, the types of errors caught by unit tests are most often caught in the initial process of writing the tests, and thereafter they shore up the codebase against regressions, especially in core classes and types, which can be very useful for refactoring.
A: Yes, and it can be especially difficult with existing code that wasn't designed for unit testing. Some common sense should be used to decide where to employ unit testing, and at what level to perform it. While unit testing takes effort, it pays dividends in preventing regressions, and helping to pinpoint the source of failures when they do occur.
A: Yes and No. Of course if you refactor code that unit tests use directly, it will have to be reworked as well. It actually can make refactoring more efficient, by providing assurance that the mechanism still works as intended.
A: That's a great question, without a known immediate answer. It is likely possible to debug them interactively through PlatformIO, but that can at times take some creativity to configure. Unit tests are generally extremely small, so even without interactive debugging it can get you fairly close to the cause of the problem.
We are currently using PlatformIO unit tests.
Since Marlin only compiles code required by the configuration, a separate test binary must be generated for any configuration change. The following process is used to unit test a variety of configurations:
config.ini
), each containing a distinct set of configuration options for unit testing. All applicable unit tests will be run for each of these configurations.Marlin/tests
folder contains the CPP code for all Unit Tests. Marlin macros (ENABLED(feature)
, TERN(FEATURE, A, B)
, etc.) are used to determine which tests should be registered and to alter test behavior.linux_native_test
PlatformIO environment specifies a script to collect all the tests from this folder and add them to PlatformIO's list of test targets.Makefile
commands unit-test-all-local
or unit-test-all-local-docker
.