2023-08-29 19:46:02 +00:00
|
|
|
/**
|
|
|
|
* test_parallel.h
|
|
|
|
* =============================================================================
|
2025-01-21 01:18:59 +00:00
|
|
|
* Copyright (c) 2023-present Serhii Snitsaruk and the LimboAI contributors.
|
2023-08-29 19:46:02 +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_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<BTParallel> par = memnew(BTParallel);
|
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 19:46:02 +00:00
|
|
|
|
|
|
|
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;
|
2023-08-30 08:34:12 +00:00
|
|
|
par->set_num_successes_required(1);
|
|
|
|
par->set_num_failures_required(1);
|
|
|
|
par->set_repeat(false);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
2023-08-30 08:34:12 +00:00
|
|
|
CHECK(par->execute(0.01666) == BTTask::SUCCESS); // When reached both conditions.
|
2023-08-29 19:46:02 +00:00
|
|
|
|
|
|
|
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;
|
2023-08-30 08:34:12 +00:00
|
|
|
par->set_num_successes_required(1);
|
|
|
|
par->set_num_failures_required(1);
|
|
|
|
par->set_repeat(false);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
2023-08-30 08:34:12 +00:00
|
|
|
CHECK(par->execute(0.01666) == BTTask::SUCCESS);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
|
|
|
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;
|
2023-08-30 08:34:12 +00:00
|
|
|
par->set_num_successes_required(1);
|
|
|
|
par->set_num_failures_required(1);
|
|
|
|
par->set_repeat(false);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
2023-08-30 08:34:12 +00:00
|
|
|
CHECK(par->execute(0.01666) == BTTask::FAILURE);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
|
|
|
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);
|
|
|
|
|
2023-08-30 08:34:12 +00:00
|
|
|
CHECK(par->execute(0.01666) == BTTask::RUNNING);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
|
|
|
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);
|
2023-08-30 08:34:12 +00:00
|
|
|
CHECK(par->execute(0.01666) == BTTask::FAILURE);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
|
|
|
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);
|
2023-08-30 08:34:12 +00:00
|
|
|
CHECK(par->execute(0.01666) == BTTask::RUNNING);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
|
|
|
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).
|
2023-08-30 08:34:12 +00:00
|
|
|
CHECK(par->execute(0.01666) == BTTask::RUNNING);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
|
|
|
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);
|
2023-08-30 08:34:12 +00:00
|
|
|
CHECK(par->execute(0.01666) == BTTask::RUNNING);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
|
|
|
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.
|
2023-08-30 08:34:12 +00:00
|
|
|
CHECK(par->execute(0.01666) == BTTask::RUNNING);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
|
|
|
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);
|
2023-08-30 08:34:12 +00:00
|
|
|
CHECK(par->execute(0.01666) == BTTask::RUNNING);
|
2023-08-29 19:46:02 +00:00
|
|
|
|
|
|
|
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
|