#pragma once #include #include #include #include "query_engine/exceptions/exceptions.hpp" #include "query_engine/code_generator/clause_action.hpp" #include "query_engine/code_generator/entity_search.hpp" #include "storage/model/properties/all.hpp" #include "utils/assert.hpp" #include "utils/underlying_cast.hpp" // used for storing data related to an entity (node or relationship) // data can be: // * tags: labels or type // * props: property name, property value struct EntityData { std::vector tags; std::vector properties; void add_tag(const std::string &tag) { tags.push_back(tag); } void add_property(const std::string &property) { properties.push_back(property); } }; // used for storing indices of parameters (parameters are stripped before // compiling process into the array), so somehow the compiler has to know // how to find appropriate parameter during the compile process // // parameter index key can be related to: // * internal_id and entity_name, e.g.: // ID(n)=35445 -> ID(n)=2 -> index[PropertyIndexKey(Type::InternalId, n)] = // 2 // * entity_name and entity_property, e.g.: // n.name = "test" -> n.name = 3 -> index[PropertyIndexKey(entity_name, // entity_property)] = 3 struct ParameterIndexKey { enum class Type : uint8_t { InternalId, Projection }; ParameterIndexKey(Type type, const std::string &entity_name) : type(type), entity_name(entity_name) { } ParameterIndexKey(const std::string &entity_name, const std::string &entity_property) : type(Type::Projection), entity_name(entity_name), entity_property(entity_property) { } const Type type; const std::string entity_name; const std::string entity_property; bool operator<(const ParameterIndexKey &rhs) const { runtime_assert(type == rhs.type, "ParameterIndexKey types should be the same"); if (type == Type::InternalId) return entity_name < rhs.entity_name; if (entity_name == rhs.entity_name) return entity_property < rhs.entity_property; return entity_name < rhs.entity_name; } }; struct RelationshipData : public EntityData { enum class Direction { Left, Right }; using nodes_t = std::pair; RelationshipData(nodes_t nodes, Direction direction) : nodes(nodes), direction(direction) { } std::pair nodes; Direction direction; }; struct QueryActionData { std::map parameter_index; std::map actions; std::map entity_data; std::map relationship_data; CypherStateMachine csm; QueryActionData() = default; QueryActionData(QueryActionData &&other) = default; void create_entity(const std::string &entity) { if (entity_data.find(entity) == entity_data.end()) entity_data.emplace(entity, EntityData()); } void add_entity_tag(const std::string &entity, const std::string &tag) { create_entity(entity); entity_data.at(entity).add_tag(tag); } void add_entitiy_property(const std::string &entity, const std::string &property) { create_entity(entity); entity_data.at(entity).add_property(property); } // TODO: refactor name auto get_entity_property(const std::string& entity) const { if (entity_data.find(entity) == entity_data.end()) throw CppGeneratorException("Entity doesn't exist"); return entity_data.at(entity); } void print() const { for (auto const &action : actions) { std::cout << action.first << " " << underlying_cast(action.second) << std::endl; } } };