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"]]); |
||||
} |
Loading…
Reference in new issue