From 862d8b4d81fbae7f542bd68ea54dea11503db921 Mon Sep 17 00:00:00 2001 From: "Zed A. Shaw" Date: Mon, 31 Mar 2025 11:27:48 -0400 Subject: [PATCH] Now have a cycle and repeated action mitigation technique in the AI algorithm called 'delete shit you've seen'. --- Makefile | 4 ++-- goap.cpp | 52 ++++++++++++++++++++++++++++++++++------------- tests/rituals.cpp | 7 +++---- 3 files changed, 43 insertions(+), 20 deletions(-) diff --git a/Makefile b/Makefile index 562ac6d..5e01bce 100644 --- a/Makefile +++ b/Makefile @@ -22,7 +22,7 @@ tracy_build: meson compile -j 10 -C builddir test: build - ./builddir/runtests + ./builddir/runtests "[rituals]" run: build test powershell "cp ./builddir/zedcaster.exe ." @@ -41,7 +41,7 @@ clean: meson compile --clean -C builddir debug_test: build - gdb --nx -x .gdbinit --ex run --args builddir/runtests.exe -e + gdb --nx -x .gdbinit --ex run --args builddir/runtests.exe -e "[rituals]" win_installer: powershell 'start "C:\Program Files (x86)\solicus\InstallForge\bin\ifbuilderenvx86.exe" win_installer.ifp' diff --git a/goap.cpp b/goap.cpp index 0ab6d64..5204c17 100644 --- a/goap.cpp +++ b/goap.cpp @@ -54,26 +54,52 @@ namespace ai { return count; } + inline void dump_came_from(std::string msg, std::unordered_map& came_from, Action& current) { + fmt::println("{}: {}", msg, current.name); + + for(auto& [from, to] : came_from) { + fmt::println("from={}; to={}", from.name, to.name); + } + } + + inline void path_invariant(std::unordered_map& came_from, Action& current) { +#if defined(NDEBUG) + (void)came_from; // disable errors about unused + (void)current; +#else + bool final_found = current == FINAL_ACTION; + + for(size_t i = 0; i <= came_from.size() && came_from.contains(current); i++) { + current = came_from.at(current); + final_found = current == FINAL_ACTION; + } + + if(!final_found) { + dump_came_from("CYCLE DETECTED!", came_from, current); + dbc::sentinel("AI CYCLE FOUND!"); + } +#endif + } + Script reconstruct_path(std::unordered_map& came_from, Action& current) { Script total_path{current}; - bool final_found = false; + + path_invariant(came_from, current); for(size_t i = 0; i <= came_from.size() && came_from.contains(current); i++) { - current = came_from.at(current); - if(current != FINAL_ACTION) { - total_path.push_front(current); + auto next = came_from.at(current); + + if(next != FINAL_ACTION) { + // remove the previous node to avoid cycles and repeated actions + total_path.push_front(next); + came_from.erase(current); + current = next; } else { - final_found = true; + // found the terminator, done + break; } } - // this here temporarily while I figure out cycle detects/prevention - if(!final_found && total_path[0] != FINAL_ACTION) { - auto error = fmt::format("!!!!! You may have a cycle in your json. No FINAL found. Here's the path: "); - for(auto& action : total_path) error += fmt::format("{} ", action.name); - dbc::sentinel(error); - } - return total_path; } @@ -147,11 +173,9 @@ namespace ai { came_from.insert_or_assign(neighbor_action, current.action); g_score.insert_or_assign(neighbor, tentative_g_score); - // open_set gets the fScore ActionState neighbor_as{neighbor_action, neighbor}; int score = tentative_g_score + h(neighbor, goal); - // could maintain lowest here and avoid searching all things f_score.emplace_back(score, neighbor_as); std::push_heap(f_score.begin(), f_score.end(), FScorePair_cmp); diff --git a/tests/rituals.cpp b/tests/rituals.cpp index c46e426..f904f92 100644 --- a/tests/rituals.cpp +++ b/tests/rituals.cpp @@ -25,13 +25,12 @@ TEST_CASE("RitualEngine basic tests", "[rituals]") { re.set_state(ritual, "has_spikes", true); re.plan(ritual); - /* fmt::println("\n\n------------ TEST WILL DO MAGICK TOO"); ritual.dump(); - REQUIRE(ritual.will_do("magick_type")); + REQUIRE(ritual.will_do("pierce_type")); ritual.pop(); - REQUIRE(ritual.will_do("pierce_type")); + REQUIRE(ritual.will_do("magick_type")); re.reset(ritual); re.set_state(ritual, "has_magick", true); @@ -48,7 +47,6 @@ TEST_CASE("RitualEngine basic tests", "[rituals]") { re.plan(ritual); fmt::println("\n\n------------ TEST WILL DO LARGE DAMAGE BOOST"); ritual.dump(); - */ } TEST_CASE("confirm that cycles are avoided/detected", "[rituals]") { @@ -58,6 +56,7 @@ TEST_CASE("confirm that cycles are avoided/detected", "[rituals]") { re.set_state(ritual, "has_magick", true); re.set_state(ritual, "cursed_item", true); re.set_state(ritual, "shiny_bauble", true); + re.plan(ritual); fmt::println("\n\n------------ CYCLES AVOIDED"); ritual.dump();