2023-08-29 22:07:47 +00:00
/**
* test_dynamic_selector . h
* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
2025-01-21 01:18:59 +00:00
* Copyright ( c ) 2023 - present Serhii Snitsaruk and the LimboAI contributors .
2023-08-29 22:07:47 +00:00
*
* Use of this source code is governed by an MIT - style
* license that can be found in the LICENSE file or at
* https : //opensource.org/licenses/MIT.
* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
*/
# ifndef TEST_DYNAMIC_SELECTOR_H
# define TEST_DYNAMIC_SELECTOR_H
# include "limbo_test.h"
# include "modules/limboai/bt/tasks/bt_task.h"
# include "modules/limboai/bt/tasks/composites/bt_dynamic_selector.h"
namespace TestDynamicSelector {
TEST_CASE ( " [Modules][LimboAI] BTDynamicSelector " ) {
Ref < BTDynamicSelector > sel = memnew ( BTDynamicSelector ) ;
2023-08-30 08:34:12 +00:00
Ref < BTTestAction > task1 = memnew ( BTTestAction ( ) ) ;
Ref < BTTestAction > task2 = memnew ( BTTestAction ( ) ) ;
Ref < BTTestAction > task3 = memnew ( BTTestAction ( ) ) ;
2023-08-29 22:07:47 +00:00
sel - > add_child ( task1 ) ;
sel - > add_child ( task2 ) ;
sel - > add_child ( task3 ) ;
REQUIRE ( sel - > get_child_count ( ) = = 3 ) ;
SUBCASE ( " Subcase #1: Dynamic selector processes tasks sequentially until first SUCCESS, while re-evaluating its child tasks in every execution tick. " ) {
task1 - > ret_status = BTTask : : FAILURE ;
task2 - > ret_status = BTTask : : RUNNING ;
task3 - > ret_status = BTTask : : SUCCESS ;
2023-08-30 08:34:12 +00:00
CHECK ( sel - > execute ( 0.01666 ) = = BTTask : : RUNNING ) ;
2023-08-29 22:07:47 +00:00
CHECK ( task1 - > get_status ( ) = = BTTask : : FAILURE ) ;
CHECK ( task2 - > get_status ( ) = = BTTask : : RUNNING ) ;
CHECK ( task3 - > get_status ( ) = = BTTask : : FRESH ) ;
CHECK_ENTRIES_TICKS_EXITS ( task1 , 1 , 1 , 1 ) ; // * finished with FAILURE
CHECK_ENTRIES_TICKS_EXITS ( task2 , 1 , 1 , 0 ) ; // * enetered and running
CHECK_ENTRIES_TICKS_EXITS ( task3 , 0 , 0 , 0 ) ; // * still fresh
SUBCASE ( " Subcase 1A: With no changes, first task is re-evaluated. " ) {
2023-08-30 08:34:12 +00:00
CHECK ( sel - > execute ( 0.01666 ) = = BTTask : : RUNNING ) ;
2023-08-29 22:07:47 +00:00
CHECK ( task1 - > get_status ( ) = = BTTask : : FAILURE ) ;
CHECK ( task2 - > get_status ( ) = = BTTask : : RUNNING ) ;
CHECK ( task3 - > get_status ( ) = = BTTask : : FRESH ) ;
CHECK_ENTRIES_TICKS_EXITS ( task1 , 2 , 2 , 2 ) ; // * re-evaluated with FAILURE
CHECK_ENTRIES_TICKS_EXITS ( task2 , 1 , 2 , 0 ) ; // * continued
CHECK_ENTRIES_TICKS_EXITS ( task3 , 0 , 0 , 0 ) ; // * still fresh
}
SUBCASE ( " Subcase 1B: When second task succeeds, we finish with SUCCESS. " ) {
task2 - > ret_status = BTTask : : SUCCESS ;
2023-08-30 08:34:12 +00:00
CHECK ( sel - > execute ( 0.01666 ) = = BTTask : : SUCCESS ) ;
2023-08-29 22:07:47 +00:00
CHECK ( task1 - > get_status ( ) = = BTTask : : FAILURE ) ;
CHECK ( task2 - > get_status ( ) = = BTTask : : SUCCESS ) ;
CHECK ( task3 - > get_status ( ) = = BTTask : : FRESH ) ;
CHECK_ENTRIES_TICKS_EXITS ( task1 , 2 , 2 , 2 ) ; // * re-evaluated with FAILURE
CHECK_ENTRIES_TICKS_EXITS ( task2 , 1 , 2 , 1 ) ; // * ticked and exited with SUCCESS
CHECK_ENTRIES_TICKS_EXITS ( task3 , 0 , 0 , 0 ) ; // * still fresh
}
SUBCASE ( " Subcase 1C: When first task re-evaluates to SUCCESS, second task should be cancelled and exited. " ) {
task1 - > ret_status = BTTask : : SUCCESS ;
2023-08-30 08:34:12 +00:00
CHECK ( sel - > execute ( 0.01666 ) = = BTTask : : SUCCESS ) ;
2023-08-29 22:07:47 +00:00
CHECK ( task1 - > get_status ( ) = = BTTask : : SUCCESS ) ;
CHECK ( task2 - > get_status ( ) = = BTTask : : FRESH ) ; // * cancelled - status changed to FRESH
CHECK ( task3 - > get_status ( ) = = BTTask : : FRESH ) ;
CHECK_ENTRIES_TICKS_EXITS ( task1 , 2 , 2 , 2 ) ; // * re-evaluated with SUCCESS
CHECK_ENTRIES_TICKS_EXITS ( task2 , 1 , 1 , 1 ) ; // * cancelled - not ticked and exited
CHECK_ENTRIES_TICKS_EXITS ( task3 , 0 , 0 , 0 ) ; // * still fresh
}
SUBCASE ( " Subcase 1D: When first two fail, third one is executed. " ) {
task1 - > ret_status = BTTask : : FAILURE ;
task2 - > ret_status = BTTask : : FAILURE ;
task3 - > ret_status = BTTask : : RUNNING ;
2023-08-30 08:34:12 +00:00
CHECK ( sel - > execute ( 0.01666 ) = = BTTask : : RUNNING ) ;
2023-08-29 22:07:47 +00:00
CHECK ( task1 - > get_status ( ) = = BTTask : : FAILURE ) ;
CHECK ( task2 - > get_status ( ) = = BTTask : : FAILURE ) ;
CHECK ( task3 - > get_status ( ) = = BTTask : : RUNNING ) ;
CHECK_ENTRIES_TICKS_EXITS ( task1 , 2 , 2 , 2 ) ; // * re-evaluated with FAILURE
CHECK_ENTRIES_TICKS_EXITS ( task2 , 1 , 2 , 1 ) ; // * ticked and exited with FAILURE
CHECK_ENTRIES_TICKS_EXITS ( task3 , 1 , 1 , 0 ) ; // * entered and running
SUBCASE ( " Subcase 1D1: First two are re-evaluated, and when all finish with FAILURE, we expect FAILURE. " ) {
task1 - > ret_status = BTTask : : FAILURE ;
task2 - > ret_status = BTTask : : FAILURE ;
task3 - > ret_status = BTTask : : FAILURE ;
2023-08-30 08:34:12 +00:00
CHECK ( sel - > execute ( 0.01666 ) = = BTTask : : FAILURE ) ;
2023-08-29 22:07:47 +00:00
CHECK ( task1 - > get_status ( ) = = BTTask : : FAILURE ) ;
CHECK ( task2 - > get_status ( ) = = BTTask : : FAILURE ) ;
CHECK ( task3 - > get_status ( ) = = BTTask : : FAILURE ) ;
CHECK_ENTRIES_TICKS_EXITS ( task1 , 3 , 3 , 3 ) ; // * re-evaluated with SUCCESS
CHECK_ENTRIES_TICKS_EXITS ( task2 , 2 , 3 , 2 ) ; // * re-evaluated with SUCCESS
CHECK_ENTRIES_TICKS_EXITS ( task3 , 1 , 2 , 1 ) ; // * ticked and exited with SUCCESS
}
}
}
}
} //namespace TestDynamicSelector
# endif // TEST_DYNAMIC_SELECTOR_H