limboai/tests/test_parallel.h

213 lines
8.1 KiB
C++

/**
* test_parallel.h
* =============================================================================
* Copyright 2021-2023 Serhii Snitsaruk
*
* 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);
Ref<BTTestAction> task1 = memnew(BTTestAction(BTTask::SUCCESS));
Ref<BTTestAction> task2 = memnew(BTTestAction(BTTask::SUCCESS));
Ref<BTTestAction> task3 = memnew(BTTestAction(BTTask::SUCCESS));
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).
par->set_num_successes_required(1);
par->set_num_failures_required(1);
par->set_repeat(false);
task1->ret_status = BTTask::RUNNING;
task2->ret_status = BTTask::SUCCESS;
task3->ret_status = BTTask::FAILURE;
CHECK(par->execute(0.1666) == 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.
par->set_num_successes_required(1);
par->set_num_failures_required(1);
par->set_repeat(false);
task1->ret_status = BTTask::RUNNING;
task2->ret_status = BTTask::SUCCESS;
task3->ret_status = BTTask::RUNNING;
CHECK(par->execute(0.1666) == 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.
par->set_num_successes_required(1);
par->set_num_failures_required(1);
par->set_repeat(false);
task1->ret_status = BTTask::RUNNING;
task2->ret_status = BTTask::FAILURE;
task3->ret_status = BTTask::RUNNING;
CHECK(par->execute(0.1666) == 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.1666) == 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.1666) == 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.1666) == 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.1666) == 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.1666) == 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.1666) == 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.1666) == 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