diff --git a/.github/ISSUE_TEMPLATE/bug_report.md b/.github/ISSUE_TEMPLATE/bug_report.md index 0c9f313c4..2b2dd289b 100644 --- a/.github/ISSUE_TEMPLATE/bug_report.md +++ b/.github/ISSUE_TEMPLATE/bug_report.md @@ -24,7 +24,7 @@ If applicable, add screenshots to help explain your problem. **Desktop (please complete the following information):** - OS: [e.g. mac/linux/windows] - - Version [e.g. 1.2.16] + - Version [e.g. 1.2.17] **Additional context** Add any other context about the problem here. diff --git a/CMakeLists.txt b/CMakeLists.txt index b18b37d4e..fdd5fd5f4 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -1,5 +1,5 @@ cmake_minimum_required(VERSION 3.10.0) -project(Griddly VERSION 1.2.16) +project(Griddly VERSION 1.2.17) set(BINARY ${CMAKE_PROJECT_NAME}) diff --git a/bindings/python.cpp b/bindings/python.cpp index a08b2ce72..837c68bd3 100644 --- a/bindings/python.cpp +++ b/bindings/python.cpp @@ -12,7 +12,7 @@ namespace griddly { PYBIND11_MODULE(python_griddly, m) { m.doc() = "Griddly python bindings"; - m.attr("version") = "1.2.16"; + m.attr("version") = "1.2.17"; #ifndef NDEBUG spdlog::set_level(spdlog::level::debug); @@ -88,12 +88,24 @@ PYBIND11_MODULE(python_griddly, m) { // Get list of possible variable names, ordered by ID game_process.def("get_object_variable_names", &Py_GameWrapper::getObjectVariableNames); + // Get a mapping of objects to their variable names + game_process.def("get_object_variable_map", &Py_GameWrapper::getObjectVariableMap); + + // Get a list of the global variable names + game_process.def("get_global_variable_names", &Py_GameWrapper::getGlobalVariableNames); + // Get a list of the events that have happened in the game up to this point game_process.def("get_history", &Py_GameWrapper::getHistory, py::arg("purge")=true); // Release resources for vulkan stuff game_process.def("release", &Py_GameWrapper::release); + // Create an entity observer given a configuration of the entities and the custom variables that we want to view in the features + game_process.def("get_entity_observer", &Py_GameWrapper::createEntityObserver, py::arg("config")=py::dict()); + + py::class_> entityObserver(m, "EntityObserver"); + entityObserver.def("observe", &Py_EntityObserverWrapper::observe); + py::class_> player(m, "Player"); player.def("step", &Py_StepPlayerWrapper::stepSingle); player.def("step_multi", &Py_StepPlayerWrapper::stepMulti); diff --git a/bindings/wrapper/EntityObserverWrapper.cpp b/bindings/wrapper/EntityObserverWrapper.cpp new file mode 100644 index 000000000..e44cae56b --- /dev/null +++ b/bindings/wrapper/EntityObserverWrapper.cpp @@ -0,0 +1,140 @@ +#pragma once + +#include + +#include "../../src/Griddly/Core/TurnBasedGameProcess.hpp" +#include "NumpyWrapper.cpp" +#include "StepPlayerWrapper.cpp" + +namespace griddly { + +class Py_EntityObserverWrapper { + public: + Py_EntityObserverWrapper(py::dict entityObserverConfig, std::shared_ptr gdyFactory, std::shared_ptr gameProcess) : gameProcess_(gameProcess), gdyFactory_(gdyFactory) { + spdlog::debug("Created entity observer."); + + if (entityObserverConfig.contains("VariableMapping")) { + entityVariableMapping_ = entityObserverConfig["variableMapping"].cast>>(); + } else { + entityVariableMapping_ = gameProcess_->getGrid()->getObjectVariableMap(); + } + + for (auto entityVariables : entityVariableMapping_) { + for (auto variableName : entityVariables.second) { + spdlog::debug("Entity {0}, will include variable {1} in entity observations.", entityVariables.first, variableName); + } + } + } + + py::dict observe(int playerId) { + py::dict observation; + + auto entityObservationsAndIds = buildEntityObservations(playerId); + auto actionsAndMasks = buildEntityMasks(playerId); + + observation["Entities"] = entityObservationsAndIds["Entities"]; + observation["EntityIds"] = entityObservationsAndIds["EntityIds"]; + observation["EntityMasks"] = actionsAndMasks; + + return observation; + } + + private: + // Build entity masks (for transformer architectures) + py::dict buildEntityMasks(int playerId) const { + std::map>> entityMasks; + std::map> entityIds; + + std::unordered_set allAvailableActionNames; + + py::dict entitiesAndMasks; + + auto grid = gameProcess_->getGrid(); + + for (auto actionNamesAtLocation : gameProcess_->getAvailableActionNames(playerId)) { + auto location = actionNamesAtLocation.first; + auto actionNames = actionNamesAtLocation.second; + + auto locationVec = glm::ivec2{location[0], location[1]}; + + for (auto actionName : actionNames) { + spdlog::debug("[{0}] available at location [{1}, {2}]", actionName, location.x, location.y); + + auto actionInputsDefinitions = gdyFactory_->getActionInputsDefinitions(); + std::vector mask(actionInputsDefinitions[actionName].inputMappings.size() + 1); + mask[0] = 1; // NOP is always available + + auto objectAtLocation = grid->getObject(location); + auto entityId = std::hash>()(objectAtLocation); + auto actionIdsForName = gameProcess_->getAvailableActionIdsAtLocation(locationVec, actionName); + + for (auto id : actionIdsForName) { + mask[id] = 1; + } + + entityMasks[actionName].push_back(mask); + entityIds[actionName].push_back(entityId); + + allAvailableActionNames.insert(actionName); + } + } + + for (auto actionName : allAvailableActionNames) { + py::dict entitiesAndMasksForAction; + entitiesAndMasksForAction["EntityIds"] = entityIds[actionName]; + entitiesAndMasksForAction["Masks"] = entityMasks[actionName]; + + entitiesAndMasks[actionName.c_str()] = entitiesAndMasksForAction; + } + + return entitiesAndMasks; + } + + // Build entity observations (for transformer architectures) + py::dict buildEntityObservations(int playerId) const { + py::dict entityObservationsAndIds; + + std::map>> entityObservations; + std::vector entityIds; + + auto grid = gameProcess_->getGrid(); + + for (auto object : grid->getObjects()) { + auto name = object->getObjectName(); + auto location = object->getLocation(); + auto orientation = object->getObjectOrientation().getUnitVector(); + auto objectPlayerId = object->getPlayerId(); + auto zIdx = object->getZIdx(); + + auto featureVariables = entityVariableMapping_.at(name); + + auto numVariables = featureVariables.size(); + auto numFeatures = 5 + numVariables; + + std::vector featureVector(numFeatures); + featureVector[0] = static_cast(location[0]); + featureVector[1] = static_cast(location[1]); + featureVector[2] = static_cast(zIdx); + featureVector[3] = static_cast(orientation[0] + 2 * orientation[1]); + featureVector[4] = static_cast(objectPlayerId); + for (int32_t i = 0; i < numVariables; i++) { + auto variableValue = *object->getVariableValue(featureVariables[i]); + featureVector[5 + i] = static_cast(variableValue); + } + + entityObservations[name].push_back(featureVector); + + entityIds.push_back(std::hash>()(object)); + } + + entityObservationsAndIds["Entities"] = entityObservations; + entityObservationsAndIds["EntityIds"] = entityIds; + + return entityObservationsAndIds; + } + + std::unordered_map> entityVariableMapping_; + const std::shared_ptr gdyFactory_; + const std::shared_ptr gameProcess_; +}; +} // namespace griddly \ No newline at end of file diff --git a/bindings/wrapper/GameWrapper.cpp b/bindings/wrapper/GameWrapper.cpp index 94e5912e8..e6991b47a 100644 --- a/bindings/wrapper/GameWrapper.cpp +++ b/bindings/wrapper/GameWrapper.cpp @@ -5,6 +5,7 @@ #include "../../src/Griddly/Core/TurnBasedGameProcess.hpp" #include "NumpyWrapper.cpp" #include "StepPlayerWrapper.cpp" +#include "EntityObserverWrapper.cpp" namespace griddly { @@ -357,6 +358,20 @@ class Py_GameWrapper { return py_state; } + std::vector getGlobalVariableNames() const { + std::vector globalVariableNames; + auto globalVariables = gameProcess_->getGrid()->getGlobalVariables(); + + for (auto globalVariableIt : globalVariables) { + globalVariableNames.push_back(globalVariableIt.first); + } + return globalVariableNames; + } + + py::dict getObjectVariableMap() const { + return py::cast(gameProcess_->getGrid()->getObjectVariableMap()); + } + py::dict getGlobalVariables(std::vector variables) const { py::dict py_globalVariables; auto globalVariables = gameProcess_->getGrid()->getGlobalVariables(); @@ -419,7 +434,11 @@ class Py_GameWrapper { } std::vector getObjectVariableNames() { - return gameProcess_->getGrid()->getObjectVariableNames(); + return gameProcess_->getGrid()->getAllObjectVariableNames(); + } + + std::shared_ptr createEntityObserver(py::dict entityObserverConfig) { + return std::make_shared(Py_EntityObserverWrapper(entityObserverConfig, gdyFactory_, gameProcess_)); } private: diff --git a/docs/conf.py b/docs/conf.py index 08a185266..9ac97daa5 100644 --- a/docs/conf.py +++ b/docs/conf.py @@ -22,7 +22,7 @@ author = 'Chris Bamford' # The full version, including alpha/beta/rc tags -release = '1.2.16' +release = '1.2.17' # -- General configuration --------------------------------------------------- diff --git a/docs/tutorials/Proximity Triggers/index.rst b/docs/tutorials/Proximity Triggers/index.rst index b78ee256b..086b428f1 100644 --- a/docs/tutorials/Proximity Triggers/index.rst +++ b/docs/tutorials/Proximity Triggers/index.rst @@ -66,7 +66,7 @@ Step 1 - Create the lava, water and spider objects ********************************************** -Step 3 - Set up the proximity trigger for lava +Step 2 - Set up the proximity trigger for lava ********************************************** For the lava, we want the spider to be catch fire instantly if it is next to the lava, but have a small chance of catching fire if it is close, but not right next to it. @@ -113,7 +113,7 @@ Additionally you can set a ``Probability`` for an action to set how likely the a *********************************************** -Step 4 - Set up the proximity trigger for water +Step 3 - Set up the proximity trigger for water *********************************************** diff --git a/python/setup.py b/python/setup.py index f0411ef02..fe46e0d52 100644 --- a/python/setup.py +++ b/python/setup.py @@ -71,7 +71,7 @@ def griddly_package_data(config='Debug'): setup( name='griddly', - version="1.2.16", + version="1.2.17", author_email="chrisbam4d@gmail.com", description="Griddly Python Libraries", long_description=long_description, diff --git a/src/Griddly/Core/GDY/Objects/ObjectGenerator.cpp b/src/Griddly/Core/GDY/Objects/ObjectGenerator.cpp index f43934a30..280b08660 100644 --- a/src/Griddly/Core/GDY/Objects/ObjectGenerator.cpp +++ b/src/Griddly/Core/GDY/Objects/ObjectGenerator.cpp @@ -250,7 +250,8 @@ std::unordered_map ObjectGenerator::getActionProbabilities() return actionProbabilities_; } -std::unordered_map> ObjectGenerator::getObjectDefinitions() const { +// The order of object definitions needs to be consistent across levels and maps, so we have to make sure this is ordered here. +std::map> ObjectGenerator::getObjectDefinitions() const { return objectDefinitions_; } diff --git a/src/Griddly/Core/GDY/Objects/ObjectGenerator.hpp b/src/Griddly/Core/GDY/Objects/ObjectGenerator.hpp index 2dd8e6be8..591c8a2d2 100644 --- a/src/Griddly/Core/GDY/Objects/ObjectGenerator.hpp +++ b/src/Griddly/Core/GDY/Objects/ObjectGenerator.hpp @@ -60,11 +60,13 @@ class ObjectGenerator : public std::enable_shared_from_this { virtual std::unordered_map getActionTriggerDefinitions() const; virtual std::unordered_map getActionProbabilities() const; - virtual std::unordered_map> getObjectDefinitions() const; + virtual std::map> getObjectDefinitions() const; private: std::unordered_map objectChars_; - std::unordered_map> objectDefinitions_; + + // This needs to be ordered, so object types are always in a consistent order across multiple instantiations of games. + std::map> objectDefinitions_; std::string avatarObject_; diff --git a/src/Griddly/Core/Grid.cpp b/src/Griddly/Core/Grid.cpp index 1c9f508b1..1ef36e24b 100644 --- a/src/Griddly/Core/Grid.cpp +++ b/src/Griddly/Core/Grid.cpp @@ -463,7 +463,7 @@ const std::vector Grid::getObjectNames() const { return orderedNames; } -const std::vector Grid::getObjectVariableNames() const { +const std::vector Grid::getAllObjectVariableNames() const { auto namesCount = objectVariableIds_.size(); std::vector orderedNames(namesCount); @@ -476,12 +476,18 @@ const std::vector Grid::getObjectVariableNames() const { return orderedNames; } +const std::unordered_map> Grid::getObjectVariableMap() const { + return objectVariableMap_; +} + void Grid::initObject(std::string objectName, std::vector variableNames) { objectIds_.insert({objectName, objectIds_.size()}); for (auto& variableName : variableNames) { objectVariableIds_.insert({variableName, objectVariableIds_.size()}); } + + objectVariableMap_[objectName] = variableNames; } std::unordered_map> Grid::getObjectCounter(std::string objectName) { diff --git a/src/Griddly/Core/Grid.hpp b/src/Griddly/Core/Grid.hpp index 0ad6fc3cb..eaaefb8aa 100644 --- a/src/Griddly/Core/Grid.hpp +++ b/src/Griddly/Core/Grid.hpp @@ -133,7 +133,12 @@ class Grid : public std::enable_shared_from_this { /** * Gets an ordered list of objectVariableNames */ - virtual const std::vector getObjectVariableNames() const; + virtual const std::vector getAllObjectVariableNames() const; + + /** + * Get a mapping of objects to their defined variables + */ + virtual const std::unordered_map> getObjectVariableMap() const; /** * Gets an ordered list of objectNames @@ -180,6 +185,7 @@ class Grid : public std::enable_shared_from_this { std::unordered_map objectIds_; std::unordered_map objectVariableIds_; + std::unordered_map> objectVariableMap_; std::unordered_set> objects_; std::unordered_map occupiedLocations_; std::unordered_map>> objectCounters_; diff --git a/tests/src/Griddly/Core/GDY/GDYFactoryTest.cpp b/tests/src/Griddly/Core/GDY/GDYFactoryTest.cpp index f634ce2fe..c6229dbfc 100644 --- a/tests/src/Griddly/Core/GDY/GDYFactoryTest.cpp +++ b/tests/src/Griddly/Core/GDY/GDYFactoryTest.cpp @@ -817,8 +817,8 @@ TEST(GDYFactoryTest, loadAction_src_missing) { testBehaviourDefinition(yamlString, expectedBehaviourDefinition, true); } -std::unordered_map> mockObjectDefs(std::vector objectNames) { - std::unordered_map> mockObjectDefinitions; +std::map> mockObjectDefs(std::vector objectNames) { + std::map> mockObjectDefinitions; for (auto name : objectNames) { ObjectDefinition objectDefinition = { name}; diff --git a/tests/src/Griddly/Core/GameProcessTest.cpp b/tests/src/Griddly/Core/GameProcessTest.cpp index 916f277f7..0da22d1cd 100644 --- a/tests/src/Griddly/Core/GameProcessTest.cpp +++ b/tests/src/Griddly/Core/GameProcessTest.cpp @@ -1006,7 +1006,7 @@ TEST(GameProcessTest, clone) { EXPECT_CALL(*mockGridPtr, getGlobalVariables()) .WillRepeatedly(ReturnRef(globalVariables)); - std::unordered_map> mockObjectDefinitions = { + std::map> mockObjectDefinitions = { {"object1", std::make_shared(ObjectDefinition{"object1", 'a'})}, {"object2", std::make_shared(ObjectDefinition{"object2", 'b'})}, {"object3", std::make_shared(ObjectDefinition{"object3", 'c'})}, diff --git a/tests/src/Griddly/Core/LevelGenerator/MapReaderTest.cpp b/tests/src/Griddly/Core/LevelGenerator/MapReaderTest.cpp index fe0345390..e9b51b795 100644 --- a/tests/src/Griddly/Core/LevelGenerator/MapReaderTest.cpp +++ b/tests/src/Griddly/Core/LevelGenerator/MapReaderTest.cpp @@ -17,8 +17,8 @@ using ::testing::ReturnRef; namespace griddly { -std::unordered_map> mockObjectDefinitions(std::vector objectNames) { - std::unordered_map> mockObjectDefinitions; +std::map> mockObjectDefinitions(std::vector objectNames) { + std::map> mockObjectDefinitions; for (auto name : objectNames) { ObjectDefinition objectDefinition = { name}; diff --git a/tests/src/Mocks/Griddly/Core/GDY/Objects/MockObjectGenerator.hpp b/tests/src/Mocks/Griddly/Core/GDY/Objects/MockObjectGenerator.hpp index c153ef770..305b21955 100644 --- a/tests/src/Mocks/Griddly/Core/GDY/Objects/MockObjectGenerator.hpp +++ b/tests/src/Mocks/Griddly/Core/GDY/Objects/MockObjectGenerator.hpp @@ -19,7 +19,7 @@ class MockObjectGenerator : public ObjectGenerator { MOCK_METHOD(std::shared_ptr, cloneInstance, (std::shared_ptr, std::shared_ptr grid), ()); MOCK_METHOD(std::string&, getObjectNameFromMapChar, (char character), ()); - MOCK_METHOD((std::unordered_map>), getObjectDefinitions, (), (const)); + MOCK_METHOD((std::map>), getObjectDefinitions, (), (const)); MOCK_METHOD(void, setAvatarObject, (std::string objectName), ()); };