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