This exercise will walk through development of basic unit tests for the Led
component created in the component implementation section.
To start off, use fprime-util
to generate a unit test outline for the Led
component.
First, register unit tests for the Led
component with the build system by adding these lines to the component CMakeLists.txt
file in your led-blinker/Components/Led
directory.
set(UT_SOURCE_FILES
"${CMAKE_CURRENT_LIST_DIR}/Led.fpp"
)
set(UT_AUTO_HELPERS ON) # Additional Unit-Test autocoding
register_fprime_ut()
Next, generate a unit test build cache by running the following terminal commands:
#In led-blinker/Components/Led
fprime-util generate --ut
Unit tests run with special build settings and as such need their own build cache generated.
Next we will generate unit test template files. This is similar to the component implementations we have run, but will set up the complete unit test harness.
To do so, run the implementation command in the terminal within your led-blinker/Components/Led
directory:
#In led-blinker/Components/Led
fprime-util impl --ut
This step should create the files Tester.cpp
, Tester.hpp
, and TestMain.cpp
in your current directory. Move them to a new subdirectory called test/ut
.
This is done with:
#In led-blinker/Components/Led
mkdir -p test/ut
mv Test* test/ut/
Next, update the CMakeLists.txt
file in your led-blinker/Components/Led
directory to add those files to the list of unit-test source files. Include a module dependency of Os
for the mutex we used. That section should look like this:
set(UT_SOURCE_FILES "${CMAKE_CURRENT_LIST_DIR}/Led.fpp" "${CMAKE_CURRENT_LIST_DIR}/test/ut/TestMain.cpp" "${CMAKE_CURRENT_LIST_DIR}/test/ut/Tester.cpp" ) set(UT_MOD_DEPS Os ) set(UT_AUTO_HELPERS ON) # Additional Unit-Test autocoding register_fprime_ut()
Finally, test the skeleton unit tests with the following command:
#In led-blinker/Components/Led
fprime-util check
check
will build and run unit tests. To simply build them, runfprime-util build --ut
.
Now that unit tests have been written, we can add our first unit test case. First, remove the default ToDo
test and add a new test case called testBlinking
.
In led-blinker/Components/Led/test-ut/Tester.hpp
remove the definition for testToDo
and add:
public:
...
void testBlinking();
In led-blinker/Components/Led/test-ut/Tester.cpp
remove the definition for testToDo
and add:
void Tester ::
testBlinking()
{
}
In led-blinker/Components/Led/test-ut/TestMain.cpp
:
TEST(Nominal, TestBlinking) {
Components::Tester tester;
tester.testBlinking();
}
Use fprime-util check
to make sure the new check builds and passes.
Ensure all errors are resolved before continuing.
The first test we will write is to test that the LED doesn't blink when blinking is disabled.
Add the following code to the testBlinking
method in led-blinker/Components/Led/test-ut/Tester.cpp
:
// Ensure LED stays off when blinking is disabled
// The Led component defaults to blinking off
this->invoke_to_run(0,0); // invoke the 'run' port to simulate running one cycle
ASSERT_EVENTS_LedState_SIZE(0); // ensure no LedState change events we emitted
ASSERT_from_gpioSet_SIZE(0); // ensure gpio LED wasn't set
ASSERT_TLM_LedTransitions_SIZE(0); // ensure no LedTransitions were recorded
The this->invoke_to_<port-name>()
methods are used to call input ports on the component under test acting like a port invocation in the system topology but driven by out test harness.
The F´ unit testing framework generates a series of history buffers to store a fixed amount of events, telemetry, and output ports emitted from the component.
The ASSERT_<>_SIZE(size)
(e.g. ASSERT_EVENTS_LedState_SIZE(0)
) macros are used to assert the size of the history buffer matches your expectations. The ASSERT_<>(index, <arg 1>, <arg 1>, <arg N>)
macros are used to check that items in the history buffer match expectations.
Use fprime-util check
to make sure the test case builds and passes.
Next, enable blinking, then step through 3 cycles to verify the LED component blinks the LED on, off, and the on again.
// Send command to enable blinking
this->sendCmd_BLINKING_ON_OFF(0, 0, Fw::On::ON);
this->component.doDispatch(); // Trigger execution of async command
The F´ unit test framework provides this->sendCmd_COMMAND_NAME(args)
function that allows calling a command on the component under test. BLINKING_ON_OFF
is an async
command, it's not dispatched immediately, but instead added to an execution queue that would normally be driven off the component's thread.
To dispatch a queued command, unit tests must explicitly call the doDispatch()
function to dispatch the first message on the queue.
Now, check that the state of the component matches expectations after each of three cycles. Write assertions to fill in the todo comments.
// Step through 3 run cycles to observe LED turning on and off 3 times
// Cycle 1: LED initalization->On
this->invoke_to_run(0,0);
ASSERT_EVENTS_LedState_SIZE(1);
ASSERT_EVENTS_LedState(0, Fw::On::ON);
ASSERT_from_gpioSet_SIZE(1);
ASSERT_from_gpioSet(0, Fw::Logic::HIGH);
ASSERT_TLM_LedTransitions_SIZE(1);
ASSERT_TLM_LedTransitions(0, 1);
// Cycle 2: LED On->Off
this->invoke_to_run(0,0);
ASSERT_EVENTS_LedState_SIZE(2);
ASSERT_EVENTS_LedState(1, Fw::On::OFF);
ASSERT_from_gpioSet_SIZE(2);
ASSERT_from_gpioSet(1, Fw::Logic::LOW);
//TODO: Add assertions for LedTransitions telemetry
// Cycle 3: LED Off->On
this->invoke_to_run(0,0);
//TODO: Write assertions for third cycle
Run fprime-util check
and make sure the new assertions pass.
This second test will test that adjusting BLINK_INTERVAL
will impact the blink interval.
Add a new test case called testBlinkInterval
and use the following code as a starting point:
void Tester ::
testBlinkInterval()
{
// Enable LED Blinking
this->sendCmd_BLINKING_ON_OFF(0, 0, Fw::On::ON);
this->component.doDispatch(); // Trigger execution of async command
// Adjust blink interval to 4 cycles
U32 blinkInterval = 4;
this->paramSet_BLINK_INTERVAL(blinkInterval, Fw::ParamValid::VALID);
this->paramSend_BLINK_INTERVAL(0, 0);
ASSERT_EVENTS_BlinkIntervalSet_SIZE(1);
// TODO: Add logic to test adjusted blink interval
}
Don't forget to add
testBlinkInterval()
toTester.hpp
as well. Run `fprime-util check and resolve any issues before continuing.
Coverage of the code can be easily checked by adding the --coverage
flag and opening the report with your web browser.
# In led-blinker/Components/Led
fprime-util check --coverage
Now open the file led-blinker/Components/Led/coverage/coverage.html
with your web browser and explore the coverage report.
Congratulations! You've tested the Led
component with some unit-tests!
The final section of this tutorial is to test the component via some system tests!