Skip to content

Commit 9703e49

Browse files
RC-Repositoriesaggarg
authored andcommitted
docs: Document unit testing with configASSERT
This commit adds an explanation in `unit_testing.md` on how configASSERT is handled in `freertos_command_pool.c` tests and other test files. The intended behaviour is that an assertion failure causes the rest of the code to stop running, but can be handled by a test and does not stop the rest of the test's checks running. Signed-off-by: Reuben Cartwright <[email protected]>
1 parent 78c58e1 commit 9703e49

File tree

1 file changed

+95
-0
lines changed

1 file changed

+95
-0
lines changed

docs/unit_testing.md

+95
Original file line numberDiff line numberDiff line change
@@ -413,6 +413,101 @@ This is testing the function `initialize_network`, however this function does no
413413

414414
> :bulb: This can be avoided by using a Test Driven Development (TDD) approach which will ensure that all functions developed can be tested. Writing unit test cases for existing modules does not guarantee that the behavior can be tested without observing implementation details.
415415
416+
### Testing with configASSERT
417+
418+
We define the macro configASSERT to be:
419+
420+
```c
421+
#define configASSERT( x ) if( ( x ) == 0 ) vAssertCalled( __FILE__, __LINE__ );
422+
```
423+
Where `vAssertCalled` is a void-returning fake function.
424+
425+
We want to test the below function:
426+
```c
427+
MQTTAgentCommand_t * Agent_GetCommand( uint32_t blockTimeMs )
428+
{
429+
MQTTAgentCommand_t * structToUse = NULL;
430+
bool structRetrieved = false;
431+
432+
/* Check queue has been created. */
433+
configASSERT( initStatus == QUEUE_INITIALIZED );
434+
435+
/* Retrieve a struct from the queue. */
436+
structRetrieved = Agent_MessageReceive( &commandStructMessageCtx, &( structToUse ), blockTimeMs );
437+
438+
if( !structRetrieved )
439+
{
440+
LogDebug( ( "No command structure available.\n" ) );
441+
}
442+
443+
return structToUse;
444+
}
445+
```
446+
447+
We want to test that if the function is called with the command pool uninitialized, then it will error and not call `Agent_MessageReceive`.
448+
449+
A naive test would be:
450+
451+
```cpp
452+
TEST_F(TestFreertosCommandPool, does_not_try_to_get_command_if_pool_not_initialized) {
453+
Agent_MessageReceive_fake.return_val = true;
454+
Agent_MessageSend_fake.return_val = true;
455+
456+
// We expect MessageReceive to never be called on unsafe memory.
457+
Agent_GetCommand(20);
458+
expect_errors();
459+
EXPECT_EQ(Agent_MessageReceive_fake.call_count, 0);
460+
}
461+
```
462+
The above will not work. If we do not define a custom fake for `vAssertCalled`, the program will not stop at the line `configASSERT` fails, but instead continue to the next line and call `Agent_MessageReceive`, which will cause the test to fail.
463+
464+
The desired behaviour is:
465+
466+
1. Failing an assertion causes the function under test to stop running.
467+
2. GoogleTest still checks other assertions (such as `Agent_MessageReceive_fake.call_count equals zero`).
468+
469+
The method in use at the moment is to:
470+
471+
1. Define an `ASSERTION_FAILURE` error code and a custom fake for `vAssertCalled`.
472+
2. Define a custom fake for `vAssertCalled`
473+
```cpp
474+
#define ASSERTION_FAILURE 1
475+
476+
/* Mocks for vAssertCalled */
477+
void throw_assertion_failure ( const char * pcFile,
478+
unsigned long ulLine ) {
479+
/*
480+
Behaviour wanted:
481+
- Encounters assertion fail, stops running any more code. E.g. does not go to next line.
482+
- But checks all assertions in the google test program hold.
483+
*/
484+
throw (ASSERTION_FAILURE);
485+
}
486+
```
487+
3. Assign the `vAssertCalled` custom fake for every test (by default). Within the initialisation for `TestFreertosCommandPool()`:
488+
```cpp
489+
vAssertCalled_fake.custom_fake = throw_assertion_failure;
490+
```
491+
4. Within tests expected to fail an assertion, catch the assertion.
492+
```cpp
493+
TEST_F(TestFreertosCommandPool, does_not_try_to_get_command_if_pool_not_initialized) {
494+
Agent_MessageReceive_fake.return_val = true;
495+
Agent_MessageSend_fake.return_val = true;
496+
497+
// We expect MessageReceive to never be called on unsafe memory.
498+
try{
499+
Agent_GetCommand(20);
500+
} catch (int num) {
501+
if (num != ASSERTION_FAILURE) {
502+
throw (num);
503+
}
504+
}
505+
expect_errors();
506+
EXPECT_EQ(Agent_MessageReceive_fake.call_count, 0);
507+
}
508+
```
509+
510+
The above example is from the file [`test_freertos_command_pool.cpp`](../components/aws_iot/coremqtt_agent/integration/tests/test_freertos_command_pool.cpp).
416511
417512
### Integrating unit test file with CMake
418513

0 commit comments

Comments
 (0)