2023-09-11 13:33:13 +00:00
|
|
|
/**
|
|
|
|
* test_hsm.h
|
|
|
|
* =============================================================================
|
2025-01-21 01:18:59 +00:00
|
|
|
* Copyright (c) 2023-present Serhii Snitsaruk and the LimboAI contributors.
|
2023-09-11 13:33:13 +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_HSM_H
|
|
|
|
#define TEST_HSM_H
|
|
|
|
|
|
|
|
#include "limbo_test.h"
|
|
|
|
|
|
|
|
#include "modules/limboai/hsm/limbo_hsm.h"
|
|
|
|
#include "modules/limboai/hsm/limbo_state.h"
|
|
|
|
|
|
|
|
#include "core/object/object.h"
|
|
|
|
#include "core/object/ref_counted.h"
|
|
|
|
#include "core/os/memory.h"
|
|
|
|
#include "core/variant/variant.h"
|
|
|
|
|
|
|
|
namespace TestHSM {
|
|
|
|
|
2024-05-01 11:59:57 +00:00
|
|
|
inline void wire_callbacks(LimboState *p_state, Ref<CallbackCounter> p_entries_counter, Ref<CallbackCounter> p_updates_counter, Ref<CallbackCounter> p_exits_counter) {
|
|
|
|
p_state->call_on_enter(callable_mp(p_entries_counter.ptr(), &CallbackCounter::callback));
|
|
|
|
p_state->call_on_update(callable_mp(p_updates_counter.ptr(), &CallbackCounter::callback_delta));
|
|
|
|
p_state->call_on_exit(callable_mp(p_exits_counter.ptr(), &CallbackCounter::callback));
|
|
|
|
}
|
|
|
|
|
2023-09-11 13:33:13 +00:00
|
|
|
class TestGuard : public RefCounted {
|
|
|
|
GDCLASS(TestGuard, RefCounted);
|
|
|
|
|
|
|
|
public:
|
|
|
|
bool permitted_to_enter = false;
|
|
|
|
bool can_enter() { return permitted_to_enter; }
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_CASE("[Modules][LimboAI] HSM") {
|
|
|
|
Node *agent = memnew(Node);
|
|
|
|
LimboHSM *hsm = memnew(LimboHSM);
|
|
|
|
|
|
|
|
Ref<CallbackCounter> alpha_entries = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> alpha_exits = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> alpha_updates = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> beta_entries = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> beta_exits = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> beta_updates = memnew(CallbackCounter);
|
2024-05-01 11:59:57 +00:00
|
|
|
Ref<CallbackCounter> nested_entries = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> nested_exits = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> nested_updates = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> gamma_entries = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> gamma_exits = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> gamma_updates = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> delta_entries = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> delta_exits = memnew(CallbackCounter);
|
|
|
|
Ref<CallbackCounter> delta_updates = memnew(CallbackCounter);
|
2023-09-11 13:33:13 +00:00
|
|
|
|
|
|
|
LimboState *state_alpha = memnew(LimboState);
|
2024-05-01 11:59:57 +00:00
|
|
|
wire_callbacks(state_alpha, alpha_entries, alpha_updates, alpha_exits);
|
2023-09-11 13:33:13 +00:00
|
|
|
LimboState *state_beta = memnew(LimboState);
|
2024-05-01 11:59:57 +00:00
|
|
|
wire_callbacks(state_beta, beta_entries, beta_updates, beta_exits);
|
|
|
|
LimboHSM *nested_hsm = memnew(LimboHSM);
|
|
|
|
wire_callbacks(nested_hsm, nested_entries, nested_updates, nested_exits);
|
|
|
|
LimboState *state_gamma = memnew(LimboState);
|
|
|
|
wire_callbacks(state_gamma, gamma_entries, gamma_updates, gamma_exits);
|
|
|
|
LimboState *state_delta = memnew(LimboState);
|
|
|
|
wire_callbacks(state_delta, delta_entries, delta_updates, delta_exits);
|
2023-09-11 13:33:13 +00:00
|
|
|
|
|
|
|
hsm->add_child(state_alpha);
|
|
|
|
hsm->add_child(state_beta);
|
2024-05-01 11:59:57 +00:00
|
|
|
hsm->add_child(nested_hsm);
|
|
|
|
nested_hsm->add_child(state_gamma);
|
|
|
|
nested_hsm->add_child(state_delta);
|
2023-09-11 13:33:13 +00:00
|
|
|
|
|
|
|
hsm->add_transition(state_alpha, state_beta, "event_one");
|
|
|
|
hsm->add_transition(state_beta, state_alpha, "event_two");
|
2024-05-01 11:59:57 +00:00
|
|
|
hsm->add_transition(hsm->anystate(), nested_hsm, "goto_nested");
|
|
|
|
nested_hsm->add_transition(state_gamma, state_delta, "goto_delta");
|
|
|
|
nested_hsm->add_transition(state_delta, state_gamma, "goto_gamma");
|
2023-09-11 13:33:13 +00:00
|
|
|
|
|
|
|
hsm->set_initial_state(state_alpha);
|
2024-03-08 19:04:27 +00:00
|
|
|
Ref<Blackboard> parent_scope = memnew(Blackboard);
|
|
|
|
hsm->initialize(agent, parent_scope);
|
2023-09-11 13:33:13 +00:00
|
|
|
hsm->set_active(true);
|
|
|
|
|
2024-07-21 12:02:27 +00:00
|
|
|
SUBCASE("Test has_transition() and remove_transition()") {
|
|
|
|
CHECK(hsm->has_transition(state_alpha, "event_one"));
|
|
|
|
CHECK(hsm->has_transition(state_beta, "event_two"));
|
|
|
|
CHECK(hsm->has_transition(hsm->anystate(), "goto_nested"));
|
|
|
|
CHECK_FALSE(hsm->has_transition(state_alpha, "event_two"));
|
|
|
|
CHECK_FALSE(hsm->has_transition(state_beta, "event_one"));
|
|
|
|
CHECK_FALSE(hsm->has_transition(hsm->anystate(), "event_one"));
|
|
|
|
|
|
|
|
hsm->remove_transition(state_alpha, "event_one");
|
|
|
|
CHECK_FALSE(hsm->has_transition(state_alpha, "event_one"));
|
|
|
|
hsm->remove_transition(state_beta, "event_two");
|
|
|
|
CHECK_FALSE(hsm->has_transition(state_beta, "event_two"));
|
|
|
|
hsm->remove_transition(hsm->anystate(), "goto_nested");
|
|
|
|
CHECK_FALSE(hsm->has_transition(hsm->anystate(), "goto_nested"));
|
|
|
|
}
|
2023-09-11 13:33:13 +00:00
|
|
|
SUBCASE("Test get_root()") {
|
|
|
|
CHECK(state_alpha->get_root() == hsm);
|
|
|
|
CHECK(state_beta->get_root() == hsm);
|
|
|
|
CHECK(hsm->get_root() == hsm);
|
|
|
|
}
|
|
|
|
SUBCASE("Test with basic workflow and transitions") {
|
|
|
|
REQUIRE(hsm->is_active());
|
|
|
|
REQUIRE(hsm->get_active_state() == state_alpha);
|
|
|
|
CHECK(alpha_entries->num_callbacks == 1); // * entered
|
|
|
|
CHECK(alpha_updates->num_callbacks == 0);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 0);
|
|
|
|
CHECK(beta_entries->num_callbacks == 0);
|
|
|
|
CHECK(beta_updates->num_callbacks == 0);
|
|
|
|
CHECK(beta_exits->num_callbacks == 0);
|
|
|
|
|
|
|
|
hsm->update(0.01666);
|
|
|
|
CHECK(alpha_entries->num_callbacks == 1);
|
|
|
|
CHECK(alpha_updates->num_callbacks == 1); // * updated
|
|
|
|
CHECK(alpha_exits->num_callbacks == 0);
|
|
|
|
CHECK(beta_entries->num_callbacks == 0);
|
|
|
|
CHECK(beta_updates->num_callbacks == 0);
|
|
|
|
CHECK(beta_exits->num_callbacks == 0);
|
|
|
|
|
|
|
|
hsm->update(0.01666);
|
|
|
|
CHECK(alpha_entries->num_callbacks == 1);
|
|
|
|
CHECK(alpha_updates->num_callbacks == 2); // * updated x2
|
|
|
|
CHECK(alpha_exits->num_callbacks == 0);
|
|
|
|
CHECK(beta_entries->num_callbacks == 0);
|
|
|
|
CHECK(beta_updates->num_callbacks == 0);
|
|
|
|
CHECK(beta_exits->num_callbacks == 0);
|
|
|
|
|
|
|
|
hsm->dispatch("event_one");
|
|
|
|
REQUIRE(hsm->get_active_state() == state_beta);
|
|
|
|
CHECK(alpha_entries->num_callbacks == 1);
|
|
|
|
CHECK(alpha_updates->num_callbacks == 2);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 1); // * (1) exited
|
|
|
|
CHECK(beta_entries->num_callbacks == 1); // * (2) entered
|
|
|
|
CHECK(beta_updates->num_callbacks == 0);
|
|
|
|
CHECK(beta_exits->num_callbacks == 0);
|
|
|
|
|
|
|
|
hsm->update(0.01666);
|
|
|
|
CHECK(alpha_entries->num_callbacks == 1);
|
|
|
|
CHECK(alpha_updates->num_callbacks == 2);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 1);
|
|
|
|
CHECK(beta_entries->num_callbacks == 1);
|
|
|
|
CHECK(beta_updates->num_callbacks == 1); // * updated
|
|
|
|
CHECK(beta_exits->num_callbacks == 0);
|
|
|
|
|
|
|
|
hsm->update(0.01666);
|
|
|
|
CHECK(alpha_entries->num_callbacks == 1);
|
|
|
|
CHECK(alpha_updates->num_callbacks == 2);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 1);
|
|
|
|
CHECK(beta_entries->num_callbacks == 1);
|
|
|
|
CHECK(beta_updates->num_callbacks == 2); // * updated x2
|
|
|
|
CHECK(beta_exits->num_callbacks == 0);
|
|
|
|
|
|
|
|
hsm->dispatch("event_two");
|
|
|
|
REQUIRE(hsm->get_active_state() == state_alpha);
|
|
|
|
CHECK(alpha_entries->num_callbacks == 2); // * (2) entered
|
|
|
|
CHECK(alpha_updates->num_callbacks == 2);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 1);
|
|
|
|
CHECK(beta_entries->num_callbacks == 1);
|
|
|
|
CHECK(beta_updates->num_callbacks == 2);
|
|
|
|
CHECK(beta_exits->num_callbacks == 1); // * (1) exited
|
|
|
|
|
|
|
|
hsm->update(0.01666);
|
|
|
|
CHECK(alpha_entries->num_callbacks == 2);
|
|
|
|
CHECK(alpha_updates->num_callbacks == 3); // * updated
|
|
|
|
CHECK(alpha_exits->num_callbacks == 1);
|
|
|
|
CHECK(beta_entries->num_callbacks == 1);
|
|
|
|
CHECK(beta_updates->num_callbacks == 2);
|
|
|
|
CHECK(beta_exits->num_callbacks == 1);
|
|
|
|
|
2024-01-18 10:32:32 +00:00
|
|
|
hsm->dispatch(hsm->event_finished());
|
2023-09-11 13:33:13 +00:00
|
|
|
CHECK(alpha_entries->num_callbacks == 2);
|
|
|
|
CHECK(alpha_updates->num_callbacks == 3);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 2); // * exited
|
|
|
|
CHECK(beta_entries->num_callbacks == 1);
|
|
|
|
CHECK(beta_updates->num_callbacks == 2);
|
|
|
|
CHECK(beta_exits->num_callbacks == 1);
|
|
|
|
CHECK_FALSE(hsm->is_active()); // * not active
|
|
|
|
CHECK(hsm->get_active_state() == nullptr);
|
|
|
|
}
|
2024-07-21 12:07:56 +00:00
|
|
|
SUBCASE("Test change_active_state()") {
|
|
|
|
REQUIRE(hsm->is_active());
|
|
|
|
REQUIRE(hsm->get_active_state() == state_alpha);
|
|
|
|
|
|
|
|
CHECK(alpha_entries->num_callbacks == 1); // * entered
|
|
|
|
CHECK(alpha_updates->num_callbacks == 0);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 0);
|
|
|
|
CHECK(beta_entries->num_callbacks == 0);
|
|
|
|
CHECK(beta_updates->num_callbacks == 0);
|
|
|
|
CHECK(beta_exits->num_callbacks == 0);
|
|
|
|
|
|
|
|
hsm->change_active_state(state_beta);
|
|
|
|
CHECK(hsm->get_active_state() == state_beta);
|
|
|
|
|
|
|
|
CHECK(alpha_entries->num_callbacks == 1);
|
|
|
|
CHECK(alpha_updates->num_callbacks == 0);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 1); // * exited
|
|
|
|
CHECK(beta_entries->num_callbacks == 1); // * entered
|
|
|
|
CHECK(beta_updates->num_callbacks == 0);
|
|
|
|
CHECK(beta_exits->num_callbacks == 0);
|
|
|
|
|
|
|
|
hsm->change_active_state(state_beta); // * should exit and re-enter
|
|
|
|
CHECK(hsm->get_active_state() == state_beta);
|
|
|
|
|
|
|
|
CHECK(alpha_entries->num_callbacks == 1);
|
|
|
|
CHECK(alpha_updates->num_callbacks == 0);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 1);
|
|
|
|
CHECK(beta_entries->num_callbacks == 2); // * re-entered
|
|
|
|
CHECK(beta_updates->num_callbacks == 0);
|
|
|
|
CHECK(beta_exits->num_callbacks == 1); // * exited
|
|
|
|
}
|
2024-11-01 13:59:10 +00:00
|
|
|
SUBCASE("Test transition with state-wide guard") {
|
2023-09-11 13:33:13 +00:00
|
|
|
Ref<TestGuard> guard = memnew(TestGuard);
|
|
|
|
state_beta->set_guard(callable_mp(guard.ptr(), &TestGuard::can_enter));
|
|
|
|
|
|
|
|
SUBCASE("When entry is permitted") {
|
|
|
|
guard->permitted_to_enter = true;
|
|
|
|
hsm->dispatch("event_one");
|
|
|
|
CHECK(hsm->get_active_state() == state_beta);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 1);
|
|
|
|
CHECK(beta_entries->num_callbacks == 1);
|
|
|
|
}
|
|
|
|
SUBCASE("When entry is not permitted") {
|
|
|
|
guard->permitted_to_enter = false;
|
|
|
|
hsm->dispatch("event_one");
|
|
|
|
CHECK(hsm->get_active_state() == state_alpha);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 0);
|
|
|
|
CHECK(beta_entries->num_callbacks == 0);
|
|
|
|
}
|
|
|
|
}
|
2024-11-01 13:59:10 +00:00
|
|
|
SUBCASE("Test transition with transition-scoped guard") {
|
|
|
|
Ref<TestGuard> guard = memnew(TestGuard);
|
|
|
|
hsm->add_transition(state_alpha, state_beta, "guarded_transition", callable_mp(guard.ptr(), &TestGuard::can_enter));
|
|
|
|
|
|
|
|
SUBCASE("When entry is permitted") {
|
|
|
|
guard->permitted_to_enter = true;
|
|
|
|
hsm->dispatch("guarded_transition");
|
|
|
|
CHECK(hsm->get_active_state() == state_beta);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 1);
|
|
|
|
CHECK(beta_entries->num_callbacks == 1);
|
|
|
|
}
|
|
|
|
SUBCASE("When entry is not permitted") {
|
|
|
|
guard->permitted_to_enter = false;
|
|
|
|
hsm->dispatch("guarded_transition");
|
|
|
|
CHECK(hsm->get_active_state() == state_alpha);
|
|
|
|
CHECK(alpha_exits->num_callbacks == 0);
|
|
|
|
CHECK(beta_entries->num_callbacks == 0);
|
|
|
|
}
|
|
|
|
}
|
2023-09-11 13:33:13 +00:00
|
|
|
SUBCASE("When there is no transition for given event") {
|
|
|
|
hsm->dispatch("not_found");
|
|
|
|
CHECK(alpha_exits->num_callbacks == 0);
|
|
|
|
CHECK(beta_entries->num_callbacks == 0);
|
|
|
|
CHECK(hsm->is_active());
|
|
|
|
CHECK(hsm->get_active_state() == state_alpha);
|
|
|
|
}
|
2024-03-08 19:04:27 +00:00
|
|
|
SUBCASE("Check if parent scope is accessible") {
|
|
|
|
parent_scope->set_var("parent_var", 100);
|
|
|
|
CHECK(state_alpha->get_blackboard()->get_parent() == parent_scope);
|
|
|
|
CHECK(state_beta->get_blackboard()->get_parent() == parent_scope);
|
|
|
|
CHECK(state_alpha->get_blackboard()->get_var("parent_var", Variant()) == Variant(100));
|
|
|
|
}
|
2024-05-01 11:59:57 +00:00
|
|
|
SUBCASE("Test flow with a nested HSM, and test dispatch() from nested states") {
|
|
|
|
state_gamma->dispatch("goto_nested");
|
|
|
|
CHECK(hsm->get_leaf_state() == state_gamma);
|
|
|
|
CHECK(nested_entries->num_callbacks == 1);
|
|
|
|
CHECK(nested_updates->num_callbacks == 0);
|
|
|
|
CHECK(nested_exits->num_callbacks == 0);
|
|
|
|
CHECK(gamma_entries->num_callbacks == 1);
|
|
|
|
CHECK(gamma_updates->num_callbacks == 0);
|
|
|
|
CHECK(gamma_exits->num_callbacks == 0);
|
|
|
|
|
|
|
|
hsm->update(0.01666);
|
|
|
|
CHECK(nested_entries->num_callbacks == 1);
|
|
|
|
CHECK(nested_updates->num_callbacks == 1);
|
|
|
|
CHECK(nested_exits->num_callbacks == 0);
|
|
|
|
CHECK(gamma_entries->num_callbacks == 1);
|
|
|
|
CHECK(gamma_updates->num_callbacks == 1);
|
|
|
|
CHECK(gamma_exits->num_callbacks == 0);
|
|
|
|
|
|
|
|
state_gamma->dispatch("goto_delta");
|
|
|
|
CHECK(hsm->get_leaf_state() == state_delta);
|
|
|
|
CHECK(nested_entries->num_callbacks == 1);
|
|
|
|
CHECK(nested_updates->num_callbacks == 1);
|
|
|
|
CHECK(nested_exits->num_callbacks == 0);
|
|
|
|
CHECK(gamma_entries->num_callbacks == 1);
|
|
|
|
CHECK(gamma_updates->num_callbacks == 1);
|
|
|
|
CHECK(gamma_exits->num_callbacks == 1);
|
|
|
|
CHECK(delta_entries->num_callbacks == 1);
|
|
|
|
CHECK(delta_updates->num_callbacks == 0);
|
|
|
|
CHECK(delta_exits->num_callbacks == 0);
|
|
|
|
|
|
|
|
state_delta->dispatch(hsm->event_finished());
|
|
|
|
CHECK(nested_entries->num_callbacks == 1);
|
|
|
|
CHECK(nested_updates->num_callbacks == 1);
|
|
|
|
CHECK(nested_exits->num_callbacks == 1);
|
|
|
|
CHECK(gamma_entries->num_callbacks == 1);
|
|
|
|
CHECK(gamma_updates->num_callbacks == 1);
|
|
|
|
CHECK(gamma_exits->num_callbacks == 1);
|
|
|
|
CHECK(delta_entries->num_callbacks == 1);
|
|
|
|
CHECK(delta_updates->num_callbacks == 0);
|
|
|
|
CHECK(delta_exits->num_callbacks == 1);
|
|
|
|
CHECK(hsm->is_active() == false);
|
|
|
|
CHECK(hsm->get_leaf_state() == hsm);
|
|
|
|
}
|
|
|
|
SUBCASE("Test get_root()") {
|
|
|
|
CHECK(hsm->get_root() == hsm);
|
|
|
|
CHECK(state_alpha->get_root() == hsm);
|
|
|
|
CHECK(state_beta->get_root() == hsm);
|
|
|
|
CHECK(nested_hsm->get_root() == hsm);
|
|
|
|
CHECK(state_delta->get_root() == hsm);
|
|
|
|
CHECK(state_gamma->get_root() == hsm);
|
|
|
|
}
|
2023-09-11 13:33:13 +00:00
|
|
|
|
|
|
|
memdelete(agent);
|
|
|
|
memdelete(hsm);
|
|
|
|
}
|
|
|
|
|
|
|
|
} //namespace TestHSM
|
|
|
|
|
|
|
|
#endif // TEST_HSM_H
|