/** * test_for_each.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_FOR_EACH_H #define TEST_FOR_EACH_H #include "limbo_test.h" #include "modules/limboai/bt/tasks/bt_task.h" #include "modules/limboai/bt/tasks/decorators/bt_for_each.h" namespace TestForEach { TEST_CASE("[Modules][LimboAI] BTForEach") { Ref fe = memnew(BTForEach); Node *dummy = memnew(Node); Ref blackboard = memnew(Blackboard); fe->initialize(dummy, blackboard, dummy); Array arr; arr.append("apple"); arr.append("raspberry"); arr.append("mushroom"); blackboard->set_var("array", arr); SUBCASE("When empty") { ERR_PRINT_OFF; CHECK(fe->execute(0.01666) == BTTask::FAILURE); ERR_PRINT_ON; } Ref task = memnew(BTTestAction(BTTask::SUCCESS)); fe->add_child(task); fe->set_array_var("array"); fe->set_save_var("element"); SUBCASE("When child returns SUCCESS") { CHECK(fe->execute(0.01666) == BTTask::RUNNING); CHECK(task->get_status() == BTTask::SUCCESS); CHECK_ENTRIES_TICKS_EXITS(task, 1, 1, 1); CHECK(blackboard->get_var("element", "wetgoop") == "apple"); CHECK(fe->execute(0.01666) == BTTask::RUNNING); CHECK(task->get_status() == BTTask::SUCCESS); CHECK_ENTRIES_TICKS_EXITS(task, 2, 2, 2); CHECK(blackboard->get_var("element", "wetgoop") == "raspberry"); CHECK(fe->execute(0.01666) == BTTask::SUCCESS); // * finished iterating - returning SUCCESS CHECK(task->get_status() == BTTask::SUCCESS); CHECK_ENTRIES_TICKS_EXITS(task, 3, 3, 3); CHECK(blackboard->get_var("element", "wetgoop") == "mushroom"); } SUBCASE("When child task takes more than one tick to finish") { task->ret_status = BTTask::RUNNING; CHECK(fe->execute(0.01666) == BTTask::RUNNING); CHECK(task->get_status() == BTTask::RUNNING); CHECK_ENTRIES_TICKS_EXITS(task, 1, 1, 0); CHECK(blackboard->get_var("element", "wetgoop") == "apple"); task->ret_status = BTTask::SUCCESS; CHECK(fe->execute(0.01666) == BTTask::RUNNING); CHECK(task->get_status() == BTTask::SUCCESS); CHECK_ENTRIES_TICKS_EXITS(task, 1, 2, 1); CHECK(blackboard->get_var("element", "wetgoop") == "apple"); task->ret_status = BTTask::RUNNING; CHECK(fe->execute(0.01666) == BTTask::RUNNING); CHECK(task->get_status() == BTTask::RUNNING); CHECK_ENTRIES_TICKS_EXITS(task, 2, 3, 1); CHECK(blackboard->get_var("element", "wetgoop") == "raspberry"); task->ret_status = BTTask::SUCCESS; CHECK(fe->execute(0.01666) == BTTask::RUNNING); CHECK(task->get_status() == BTTask::SUCCESS); CHECK_ENTRIES_TICKS_EXITS(task, 2, 4, 2); CHECK(blackboard->get_var("element", "wetgoop") == "raspberry"); task->ret_status = BTTask::RUNNING; CHECK(fe->execute(0.01666) == BTTask::RUNNING); CHECK(task->get_status() == BTTask::RUNNING); CHECK_ENTRIES_TICKS_EXITS(task, 3, 5, 2); CHECK(blackboard->get_var("element", "wetgoop") == "mushroom"); task->ret_status = BTTask::SUCCESS; CHECK(fe->execute(0.01666) == BTTask::SUCCESS); CHECK(task->get_status() == BTTask::SUCCESS); CHECK_ENTRIES_TICKS_EXITS(task, 3, 6, 3); CHECK(blackboard->get_var("element", "wetgoop") == "mushroom"); } SUBCASE("Shouldn't crash if elements are removed during iteration") { CHECK(fe->execute(0.01666) == BTTask::RUNNING); CHECK(task->get_status() == BTTask::SUCCESS); CHECK_ENTRIES_TICKS_EXITS(task, 1, 1, 1); CHECK(blackboard->get_var("element", "wetgoop") == "apple"); arr.clear(); ERR_PRINT_OFF; CHECK(fe->execute(0.01666) == BTTask::SUCCESS); // Returns SUCCESS and prints a warning without executing child task. ERR_PRINT_ON; CHECK(task->get_status() == BTTask::SUCCESS); // Same status. CHECK_ENTRIES_TICKS_EXITS(task, 1, 1, 1); // Task is not re-executed as there is not enough elements to continue iteration. CHECK(blackboard->get_var("element", "wetgoop") == "apple"); // Not changed. } } } //namespace TestForEach #endif // TEST_FOR_EACH_H