Quick renaming of stuff to be more generic as 'AI'. Now maybe I can get some sweet sweet investor money.
	
		
	
				
					
				
			
							parent
							
								
									9d6dc2f5dd
								
							
						
					
					
						commit
						a079f882df
					
				| @ -0,0 +1,76 @@ | ||||
| #pragma once | ||||
| #include <vector> | ||||
| #include "matrix.hpp" | ||||
| #include <bitset> | ||||
| #include <limits> | ||||
| #include <optional> | ||||
| #include <nlohmann/json.hpp> | ||||
| 
 | ||||
| namespace ai { | ||||
|   constexpr const int SCORE_MAX = std::numeric_limits<int>::max(); | ||||
|   constexpr const size_t STATE_MAX = 32; | ||||
| 
 | ||||
|   using State = std::bitset<STATE_MAX>; | ||||
| 
 | ||||
|   const State ALL_ZERO; | ||||
|   const State ALL_ONES = ~ALL_ZERO; | ||||
| 
 | ||||
|   struct Action { | ||||
|     std::string $name; | ||||
|     int $cost = 0; | ||||
| 
 | ||||
|     State $positive_preconds; | ||||
|     State $negative_preconds; | ||||
| 
 | ||||
|     State $positive_effects; | ||||
|     State $negative_effects; | ||||
| 
 | ||||
|     Action(std::string name, int cost) : | ||||
|       $name(name), $cost(cost) { } | ||||
| 
 | ||||
|     void needs(int name, bool val); | ||||
|     void effect(int name, bool val); | ||||
|     void load(nlohmann::json &profile, nlohmann::json& config); | ||||
| 
 | ||||
|     bool can_effect(State& state); | ||||
|     State apply_effect(State& state); | ||||
| 
 | ||||
|     bool operator==(const Action& other) const { | ||||
|       return other.$name == $name; | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   using Script = std::deque<Action>; | ||||
| 
 | ||||
|   const Action FINAL_ACTION("END", SCORE_MAX); | ||||
| 
 | ||||
|   struct ActionState { | ||||
|     Action action; | ||||
|     State state; | ||||
| 
 | ||||
|     ActionState(Action action, State state) : | ||||
|       action(action), state(state) {} | ||||
| 
 | ||||
|     bool operator==(const ActionState& other) const { | ||||
|       return other.action == action && other.state == state; | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   bool is_subset(State& source, State& target); | ||||
| 
 | ||||
|   int distance_to_goal(State& from, State& to); | ||||
| 
 | ||||
|   std::optional<Script> plan_actions(std::vector<Action>& actions, State& start, State& goal); | ||||
| } | ||||
| 
 | ||||
| template<> struct std::hash<ai::Action> { | ||||
|   size_t operator()(const ai::Action& p) const { | ||||
|     return std::hash<std::string>{}(p.$name); | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| template<> struct std::hash<ai::ActionState> { | ||||
|   size_t operator()(const ai::ActionState& p) const { | ||||
|     return std::hash<ai::Action>{}(p.action) ^ std::hash<ai::State>{}(p.state); | ||||
|   } | ||||
| }; | ||||
| @ -1,76 +0,0 @@ | ||||
| #pragma once | ||||
| #include <vector> | ||||
| #include "matrix.hpp" | ||||
| #include <bitset> | ||||
| #include <limits> | ||||
| #include <optional> | ||||
| #include <nlohmann/json.hpp> | ||||
| 
 | ||||
| namespace ailol { | ||||
|   constexpr const int SCORE_MAX = std::numeric_limits<int>::max(); | ||||
|   constexpr const size_t STATE_MAX = 32; | ||||
| 
 | ||||
|   using GOAPState = std::bitset<STATE_MAX>; | ||||
| 
 | ||||
|   const GOAPState ALL_ZERO; | ||||
|   const GOAPState ALL_ONES = ~ALL_ZERO; | ||||
| 
 | ||||
|   struct Action { | ||||
|     std::string $name; | ||||
|     int $cost = 0; | ||||
| 
 | ||||
|     GOAPState $positive_preconds; | ||||
|     GOAPState $negative_preconds; | ||||
| 
 | ||||
|     GOAPState $positive_effects; | ||||
|     GOAPState $negative_effects; | ||||
| 
 | ||||
|     Action(std::string name, int cost) : | ||||
|       $name(name), $cost(cost) { } | ||||
| 
 | ||||
|     void needs(int name, bool val); | ||||
|     void effect(int name, bool val); | ||||
|     void load(nlohmann::json &profile, nlohmann::json& config); | ||||
| 
 | ||||
|     bool can_effect(GOAPState& state); | ||||
|     GOAPState apply_effect(GOAPState& state); | ||||
| 
 | ||||
|     bool operator==(const Action& other) const { | ||||
|       return other.$name == $name; | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   using AStarPath = std::deque<Action>; | ||||
| 
 | ||||
|   const Action FINAL_ACTION("END", SCORE_MAX); | ||||
| 
 | ||||
|   struct ActionState { | ||||
|     Action action; | ||||
|     GOAPState state; | ||||
| 
 | ||||
|     ActionState(Action action, GOAPState state) : | ||||
|       action(action), state(state) {} | ||||
| 
 | ||||
|     bool operator==(const ActionState& other) const { | ||||
|       return other.action == action && other.state == state; | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   bool is_subset(GOAPState& source, GOAPState& target); | ||||
| 
 | ||||
|   int distance_to_goal(GOAPState& from, GOAPState& to); | ||||
| 
 | ||||
|   std::optional<AStarPath> plan_actions(std::vector<Action>& actions, GOAPState& start, GOAPState& goal); | ||||
| } | ||||
| 
 | ||||
| template<> struct std::hash<ailol::Action> { | ||||
|   size_t operator()(const ailol::Action& p) const { | ||||
|     return std::hash<std::string>{}(p.$name); | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| template<> struct std::hash<ailol::ActionState> { | ||||
|   size_t operator()(const ailol::ActionState& p) const { | ||||
|     return std::hash<ailol::Action>{}(p.action) ^ std::hash<ailol::GOAPState>{}(p.state); | ||||
|   } | ||||
| }; | ||||
| @ -1,194 +0,0 @@ | ||||
| #include <catch2/catch_test_macros.hpp> | ||||
| #include "dbc.hpp" | ||||
| #include "goap.hpp" | ||||
| #include <iostream> | ||||
| 
 | ||||
| using namespace dbc; | ||||
| using namespace ailol; | ||||
| using namespace nlohmann; | ||||
| 
 | ||||
| TEST_CASE("worldstate works", "[goap]") { | ||||
|   enum StateNames { | ||||
|     ENEMY_IN_RANGE, | ||||
|     ENEMY_DEAD | ||||
|   }; | ||||
| 
 | ||||
|   GOAPState goal; | ||||
|   GOAPState start; | ||||
|   std::vector<Action> actions; | ||||
| 
 | ||||
|   // start off enemy not dead and not in range
 | ||||
|   start[ENEMY_DEAD] = false; | ||||
|   start[ENEMY_IN_RANGE] = false; | ||||
| 
 | ||||
|   // end goal is enemy is dead
 | ||||
|   goal[ENEMY_DEAD] = true; | ||||
| 
 | ||||
|   Action move_closer("move_closer", 10); | ||||
|   move_closer.needs(ENEMY_IN_RANGE, false); | ||||
|   move_closer.effect(ENEMY_IN_RANGE, true); | ||||
| 
 | ||||
|   REQUIRE(move_closer.can_effect(start)); | ||||
|   auto after_move_state = move_closer.apply_effect(start); | ||||
|   REQUIRE(start[ENEMY_IN_RANGE] == false); | ||||
|   REQUIRE(after_move_state[ENEMY_IN_RANGE] == true); | ||||
|   REQUIRE(after_move_state[ENEMY_DEAD] == false); | ||||
|   // start is clean but after move is dirty
 | ||||
|   REQUIRE(move_closer.can_effect(start)); | ||||
|   REQUIRE(!move_closer.can_effect(after_move_state)); | ||||
|   REQUIRE(distance_to_goal(start, after_move_state) == 1); | ||||
| 
 | ||||
|   Action kill_it("kill_it", 10); | ||||
|   kill_it.needs(ENEMY_IN_RANGE, true); | ||||
|   kill_it.needs(ENEMY_DEAD, false); | ||||
|   kill_it.effect(ENEMY_DEAD, true); | ||||
| 
 | ||||
|   REQUIRE(!kill_it.can_effect(start)); | ||||
|   REQUIRE(kill_it.can_effect(after_move_state)); | ||||
| 
 | ||||
|   auto after_kill_state = kill_it.apply_effect(after_move_state); | ||||
|   REQUIRE(!kill_it.can_effect(after_kill_state)); | ||||
|   REQUIRE(distance_to_goal(after_move_state, after_kill_state) == 1); | ||||
| 
 | ||||
|   actions.push_back(kill_it); | ||||
|   actions.push_back(move_closer); | ||||
| 
 | ||||
|   REQUIRE(start != goal); | ||||
| } | ||||
| 
 | ||||
| TEST_CASE("basic feature tests", "[goap]") { | ||||
|   enum StateNames { | ||||
|     ENEMY_IN_RANGE, | ||||
|     ENEMY_DEAD | ||||
|   }; | ||||
| 
 | ||||
|   GOAPState goal; | ||||
|   GOAPState start; | ||||
|   std::vector<Action> actions; | ||||
| 
 | ||||
|   // start off enemy not dead and not in range
 | ||||
|   start[ENEMY_DEAD] = false; | ||||
|   start[ENEMY_IN_RANGE] = false; | ||||
| 
 | ||||
|   // end goal is enemy is dead
 | ||||
|   goal[ENEMY_DEAD] = true; | ||||
| 
 | ||||
|   Action move_closer("move_closer", 10); | ||||
|   move_closer.needs(ENEMY_IN_RANGE, false); | ||||
|   move_closer.effect(ENEMY_IN_RANGE, true); | ||||
| 
 | ||||
|   Action kill_it("kill_it", 10); | ||||
|   kill_it.needs(ENEMY_IN_RANGE, true); | ||||
|   // this is duplicated on purpose to confirm that setting
 | ||||
|   // a positive then a negative properly cancels out
 | ||||
|   kill_it.needs(ENEMY_DEAD, true); | ||||
|   kill_it.needs(ENEMY_DEAD, false); | ||||
| 
 | ||||
|   // same thing with effects
 | ||||
|   kill_it.effect(ENEMY_DEAD, false); | ||||
|   kill_it.effect(ENEMY_DEAD, true); | ||||
| 
 | ||||
|   // order seems to matter which is wrong
 | ||||
|   actions.push_back(kill_it); | ||||
|   actions.push_back(move_closer); | ||||
| 
 | ||||
|   auto result = plan_actions(actions, start, goal); | ||||
|   REQUIRE(result != std::nullopt); | ||||
| 
 | ||||
|   auto state = start; | ||||
| 
 | ||||
|   for(auto& action : *result) { | ||||
|     state = action.apply_effect(state); | ||||
|   } | ||||
| 
 | ||||
|   REQUIRE(state[ENEMY_DEAD]); | ||||
| } | ||||
| 
 | ||||
| TEST_CASE("wargame test from cppGOAP", "[goap]") { | ||||
|   std::vector<Action> actions; | ||||
|   auto profile = R"({ | ||||
|     "target_acquired": 0, | ||||
|     "target_lost": 1, | ||||
|     "target_in_warhead_range": 2, | ||||
|     "target_dead": 3 | ||||
|   })"_json; | ||||
| 
 | ||||
|   // Now establish all the possible actions for the action pool
 | ||||
|   // In this example we're providing the AI some different FPS actions
 | ||||
|   Action spiral("searchSpiral", 5); | ||||
|   auto config = R"({ | ||||
|     "needs": { | ||||
|       "target_acquired": false, | ||||
|       "target_lost": true | ||||
|     }, | ||||
|     "effects": { | ||||
|       "target_acquired": true | ||||
|     } | ||||
|   })"_json; | ||||
|   spiral.load(profile, config); | ||||
|   actions.push_back(spiral); | ||||
| 
 | ||||
|   Action serpentine("searchSerpentine", 5); | ||||
|   config = R"({ | ||||
|     "needs": { | ||||
|       "target_acquired": false, | ||||
|       "target_lost": false | ||||
|     }, | ||||
|     "effects": { | ||||
|       "target_acquired": true | ||||
|     } | ||||
|   })"_json; | ||||
|   serpentine.load(profile, config); | ||||
|   actions.push_back(serpentine); | ||||
| 
 | ||||
|   Action intercept("interceptTarget", 5); | ||||
|   config = R"({ | ||||
|     "needs": { | ||||
|       "target_acquired": true, | ||||
|       "target_dead": false | ||||
|     }, | ||||
|     "effects": { | ||||
|       "target_in_warhead_range": true | ||||
|     } | ||||
|   })"_json; | ||||
|   intercept.load(profile, config); | ||||
|   actions.push_back(intercept); | ||||
| 
 | ||||
| 
 | ||||
|   Action detonateNearTarget("detonateNearTarget", 5); | ||||
|   config = R"({ | ||||
|     "needs": { | ||||
|       "target_in_warhead_range": true, | ||||
|       "target_acquired": true, | ||||
|       "target_dead": false | ||||
|     }, | ||||
|     "effects": { | ||||
|       "target_dead": true | ||||
|     } | ||||
|   })"_json; | ||||
|   detonateNearTarget.load(profile, config); | ||||
|   actions.push_back(detonateNearTarget); | ||||
| 
 | ||||
|   // Here's the initial state...
 | ||||
|   GOAPState initial_state; | ||||
|   initial_state[profile["target_acquired"]] = false; | ||||
|   initial_state[profile["target_lost"]] = true; | ||||
|   initial_state[profile["target_in_warhead_range"]] = false; | ||||
|   initial_state[profile["target_dead"]] = false; | ||||
| 
 | ||||
|   // ...and the goal state
 | ||||
|   GOAPState goal_target_dead; | ||||
|   goal_target_dead[profile["target_dead"]] = true; | ||||
| 
 | ||||
|   auto result = plan_actions(actions, initial_state, goal_target_dead); | ||||
|   REQUIRE(result != std::nullopt); | ||||
| 
 | ||||
|   auto state = initial_state; | ||||
| 
 | ||||
|   for(auto& action : *result) { | ||||
|     fmt::println("ACTION: {}", action.$name); | ||||
|     state = action.apply_effect(state); | ||||
|   } | ||||
| 
 | ||||
|   REQUIRE(state[profile["target_dead"]]); | ||||
| } | ||||
		Reference in new issue