/** * test_parallel.h * ============================================================================= * Copyright (c) 2023-present Serhii Snitsaruk and the LimboAI contributors. * * 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_PARALLEL_H #define TEST_PARALLEL_H #include "limbo_test.h" #include "modules/limboai/bt/tasks/bt_task.h" #include "modules/limboai/bt/tasks/composites/bt_parallel.h" namespace TestParallel { TEST_CASE("[Modules][LimboAI] BTParallel with num_required_successes: 1 and num_required_failures: 1") { Ref par = memnew(BTParallel); Ref task1 = memnew(BTTestAction()); Ref task2 = memnew(BTTestAction()); Ref task3 = memnew(BTTestAction()); par->add_child(task1); par->add_child(task2); par->add_child(task3); REQUIRE(par->get_child_count() == 3); SUBCASE("BTParallel composition {RUNNING, SUCCESS, FAILURE} and successes/failures required 1/1") { // * Case #1: When reached both success and failure required, we expect one that triggered sooner (SUCCESS in this case). task1->ret_status = BTTask::RUNNING; task2->ret_status = BTTask::SUCCESS; task3->ret_status = BTTask::FAILURE; par->set_num_successes_required(1); par->set_num_failures_required(1); par->set_repeat(false); CHECK(par->execute(0.01666) == BTTask::SUCCESS); // When reached both conditions. CHECK(task1->get_status() == BTTask::RUNNING); CHECK(task2->get_status() == BTTask::SUCCESS); CHECK(task3->get_status() == BTTask::FAILURE); CHECK_ENTRIES_TICKS_EXITS(task1, 1, 1, 0); // * running CHECK_ENTRIES_TICKS_EXITS(task2, 1, 1, 1); // * finished CHECK_ENTRIES_TICKS_EXITS(task3, 1, 1, 1); // * finished } SUBCASE("BTParallel composition {RUNNING, SUCCESS, RUNNING} and successes/failures required 1/1") { // * Case #1b: When reached required number of successes, we expect SUCCESS. task1->ret_status = BTTask::RUNNING; task2->ret_status = BTTask::SUCCESS; task3->ret_status = BTTask::RUNNING; par->set_num_successes_required(1); par->set_num_failures_required(1); par->set_repeat(false); CHECK(par->execute(0.01666) == BTTask::SUCCESS); CHECK(task1->get_status() == BTTask::RUNNING); CHECK(task2->get_status() == BTTask::SUCCESS); CHECK(task3->get_status() == BTTask::RUNNING); CHECK_ENTRIES_TICKS_EXITS(task1, 1, 1, 0); // * running CHECK_ENTRIES_TICKS_EXITS(task2, 1, 1, 1); // * finished CHECK_ENTRIES_TICKS_EXITS(task3, 1, 1, 0); // * running } SUBCASE("BTParallel composition {RUNNING, FAILURE, RUNNING} and successes/failures required 1/1") { // * Case #1c: When reached required number of failures, we expect FAILURE. task1->ret_status = BTTask::RUNNING; task2->ret_status = BTTask::FAILURE; task3->ret_status = BTTask::RUNNING; par->set_num_successes_required(1); par->set_num_failures_required(1); par->set_repeat(false); CHECK(par->execute(0.01666) == BTTask::FAILURE); CHECK(task1->get_status() == BTTask::RUNNING); CHECK(task2->get_status() == BTTask::FAILURE); CHECK(task3->get_status() == BTTask::RUNNING); CHECK_ENTRIES_TICKS_EXITS(task1, 1, 1, 0); // * running CHECK_ENTRIES_TICKS_EXITS(task2, 1, 1, 1); // * finished CHECK_ENTRIES_TICKS_EXITS(task3, 1, 1, 0); // * running } SUBCASE("BTParallel composition {SUCCESS, RUNNING, FAILURE} with successes/failures required 3/3 (not repeating)") { // * Case #2: When failed to reach required number of successes or failures, // * and not all children finished executing while not repeating, we expect RUNNING. task1->ret_status = BTTask::SUCCESS; task2->ret_status = BTTask::RUNNING; task3->ret_status = BTTask::FAILURE; par->set_num_successes_required(3); par->set_num_failures_required(3); par->set_repeat(false); CHECK(par->execute(0.01666) == BTTask::RUNNING); CHECK(task1->get_status() == BTTask::SUCCESS); CHECK(task2->get_status() == BTTask::RUNNING); CHECK(task3->get_status() == BTTask::FAILURE); CHECK_ENTRIES_TICKS_EXITS(task1, 1, 1, 1); // * finished CHECK_ENTRIES_TICKS_EXITS(task2, 1, 1, 0); // * running CHECK_ENTRIES_TICKS_EXITS(task3, 1, 1, 1); // * finished } SUBCASE("BTParallel composition {SUCCESS, FAILURE, SUCCESS} with successes/failures required 3/3 (not repeating)") { // * Case #3: When failed to reach required number of successes or failures, // * and all children finished executing while not repeating, we expect FAILURE. task1->ret_status = BTTask::SUCCESS; task2->ret_status = BTTask::FAILURE; task3->ret_status = BTTask::SUCCESS; par->set_num_successes_required(3); par->set_num_failures_required(3); par->set_repeat(false); CHECK(par->execute(0.01666) == BTTask::FAILURE); CHECK(task1->get_status() == BTTask::SUCCESS); CHECK(task2->get_status() == BTTask::FAILURE); CHECK(task3->get_status() == BTTask::SUCCESS); CHECK_ENTRIES_TICKS_EXITS(task1, 1, 1, 1); CHECK_ENTRIES_TICKS_EXITS(task2, 1, 1, 1); CHECK_ENTRIES_TICKS_EXITS(task3, 1, 1, 1); } SUBCASE("BTParallel composition {SUCCESS, FAILURE, SUCCESS} with successes/failures required 3/3 (repeating)") { // * Case #4: When failed to reach required number of successes or failures, // * and all children finished executing while repeating, we expect RUNNING. task1->ret_status = BTTask::SUCCESS; task2->ret_status = BTTask::FAILURE; task3->ret_status = BTTask::SUCCESS; par->set_num_successes_required(3); par->set_num_failures_required(3); par->set_repeat(true); CHECK(par->execute(0.01666) == BTTask::RUNNING); CHECK(task1->get_status() == BTTask::SUCCESS); CHECK(task2->get_status() == BTTask::FAILURE); CHECK(task3->get_status() == BTTask::SUCCESS); CHECK_ENTRIES_TICKS_EXITS(task1, 1, 1, 1); CHECK_ENTRIES_TICKS_EXITS(task2, 1, 1, 1); CHECK_ENTRIES_TICKS_EXITS(task3, 1, 1, 1); // * Execution #2: Check if tasks are repeated, when set so (there is no RUNNING task). CHECK(par->execute(0.01666) == BTTask::RUNNING); CHECK(task1->get_status() == BTTask::SUCCESS); CHECK(task2->get_status() == BTTask::FAILURE); CHECK(task3->get_status() == BTTask::SUCCESS); CHECK_ENTRIES_TICKS_EXITS(task1, 2, 2, 2); // * repeated CHECK_ENTRIES_TICKS_EXITS(task2, 2, 2, 2); // * repeated CHECK_ENTRIES_TICKS_EXITS(task3, 2, 2, 2); // * repeated } SUBCASE("BTParallel composition {SUCCESS, RUNNING, FAILURE} with successes/failures required 2/2 (not repeating)") { // * Case #5: When failed to reach required number of successes or failures, // * but not all children finished executing (not repeating), we expect RUNNING. task1->ret_status = BTTask::SUCCESS; task2->ret_status = BTTask::RUNNING; task3->ret_status = BTTask::FAILURE; par->set_num_successes_required(2); par->set_num_failures_required(2); par->set_repeat(false); CHECK(par->execute(0.01666) == BTTask::RUNNING); CHECK(task1->get_status() == BTTask::SUCCESS); CHECK(task2->get_status() == BTTask::RUNNING); CHECK(task3->get_status() == BTTask::FAILURE); CHECK_ENTRIES_TICKS_EXITS(task1, 1, 1, 1); CHECK_ENTRIES_TICKS_EXITS(task2, 1, 1, 0); CHECK_ENTRIES_TICKS_EXITS(task3, 1, 1, 1); // * Execution #2: Check if tasks are not repeated, when set so. CHECK(par->execute(0.01666) == BTTask::RUNNING); CHECK(task1->get_status() == BTTask::SUCCESS); CHECK(task2->get_status() == BTTask::RUNNING); CHECK(task3->get_status() == BTTask::FAILURE); CHECK_ENTRIES_TICKS_EXITS(task1, 1, 1, 1); // * not repeated CHECK_ENTRIES_TICKS_EXITS(task2, 1, 2, 0); // * continued CHECK_ENTRIES_TICKS_EXITS(task3, 1, 1, 1); // * not repeated // * Execution #3: Check if tasks are repeated, when set so. par->set_repeat(true); CHECK(par->execute(0.01666) == BTTask::RUNNING); CHECK(task1->get_status() == BTTask::SUCCESS); CHECK(task2->get_status() == BTTask::RUNNING); CHECK(task3->get_status() == BTTask::FAILURE); CHECK_ENTRIES_TICKS_EXITS(task1, 2, 2, 2); // * repeated CHECK_ENTRIES_TICKS_EXITS(task2, 1, 3, 0); // * continued CHECK_ENTRIES_TICKS_EXITS(task3, 2, 2, 2); // * repeated } } } //namespace TestParallel #endif // TEST_PARALLEL_H