mapping::Tree: introduce attributes. Add PAIRS type

This commit is contained in:
Leonid Stryzhevskyi 2024-04-29 02:48:51 +03:00
parent ad3136480a
commit 249fc4f137
7 changed files with 182 additions and 24 deletions

View File

@ -27,7 +27,7 @@
#include "oatpp/data/stream/BufferStream.hpp" #include "oatpp/data/stream/BufferStream.hpp"
#include "oatpp/utils/Conversion.hpp" #include "oatpp/utils/Conversion.hpp"
namespace oatpp { namespace data { namespace oatpp { namespace data { namespace mapping {
oatpp::String ObjectToTreeMapper::MappingState::errorStacktrace() const { oatpp::String ObjectToTreeMapper::MappingState::errorStacktrace() const {
stream::BufferOutputStream ss; stream::BufferOutputStream ss;
@ -298,4 +298,4 @@ void ObjectToTreeMapper::mapObject(ObjectToTreeMapper* mapper, MappingState& sta
} }
}} }}}

View File

@ -27,7 +27,7 @@
#include "./Tree.hpp" #include "./Tree.hpp"
namespace oatpp { namespace data { namespace oatpp { namespace data { namespace mapping {
class ObjectToTreeMapper : public base::Countable { class ObjectToTreeMapper : public base::Countable {
public: public:
@ -86,6 +86,6 @@ public:
}; };
}} }}}
#endif //oatpp_data_mapping_ObjectToTreeMapper_hpp #endif //oatpp_data_mapping_ObjectToTreeMapper_hpp

View File

@ -26,7 +26,7 @@
#include "oatpp/data/stream/BufferStream.hpp" #include "oatpp/data/stream/BufferStream.hpp"
namespace oatpp { namespace data { namespace oatpp { namespace data { namespace mapping {
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Tree::Map // Tree::Map
@ -63,6 +63,59 @@ v_uint64 Tree::Map::size() const {
return m_map.size(); return m_map.size();
} }
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Tree::Attributes
Tree::Attributes::Attributes()
: m_attributes(nullptr)
{}
Tree::Attributes::Attributes(const Attributes& other)
: m_attributes(other.m_attributes != nullptr ? new std::unordered_map<oatpp::String, oatpp::String>(*other.m_attributes) : nullptr)
{}
Tree::Attributes::Attributes(Attributes&& other) noexcept
: m_attributes(other.m_attributes)
{
other.m_attributes = nullptr;
}
Tree::Attributes& Tree::Attributes::operator = (const Attributes& other) {
if(other.m_attributes) {
if(m_attributes) {
*m_attributes = *other.m_attributes;
} else {
m_attributes = new std::unordered_map<oatpp::String, oatpp::String>(*other.m_attributes);
}
} else {
delete m_attributes;
m_attributes = nullptr;
}
return *this;
}
Tree::Attributes& Tree::Attributes::operator = (Attributes&& other) noexcept {
delete m_attributes;
m_attributes = other.m_attributes;
other.m_attributes = nullptr;
return *this;
}
Tree::Attributes::~Attributes() {
delete m_attributes;
}
bool Tree::Attributes::empty() const {
return m_attributes == nullptr || m_attributes->empty();
}
v_uint64 Tree::Attributes::size() const {
if(m_attributes) {
return m_attributes->size();
}
return 0;
}
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Tree // Tree
@ -80,11 +133,18 @@ Tree::Tree(const Tree& other)
Tree::Tree(Tree&& other) noexcept Tree::Tree(Tree&& other) noexcept
: m_type(other.m_type) : m_type(other.m_type)
, m_data(other.m_data) , m_data(other.m_data)
, m_attributes(std::move(other.m_attributes))
{ {
other.m_type = Type::UNDEFINED; other.m_type = Type::UNDEFINED;
other.m_data = 0; other.m_data = 0;
} }
Tree::Tree(const oatpp::String& value)
: Tree()
{
setString(value);
}
Tree::~Tree() { Tree::~Tree() {
deleteValueObject(); deleteValueObject();
} }
@ -99,6 +159,11 @@ Tree& Tree::operator = (Tree&& other) noexcept {
return *this; return *this;
} }
Tree& Tree::operator = (const oatpp::String& value) {
setString(value);
return *this;
}
void Tree::deleteValueObject() { void Tree::deleteValueObject() {
switch (m_type) { switch (m_type) {
@ -140,6 +205,11 @@ void Tree::deleteValueObject() {
delete data; delete data;
break; break;
} }
case Type::PAIRS: {
auto data = reinterpret_cast<std::vector<std::pair<oatpp::String, Tree>> *>(m_data);
delete data;
break;
}
default: default:
// DO-NOTHING // DO-NOTHING
@ -176,6 +246,7 @@ void Tree::setCopy(const Tree& other) {
deleteValueObject(); deleteValueObject();
m_type = other.m_type; m_type = other.m_type;
m_attributes = other.m_attributes;
switch (other.m_type) { switch (other.m_type) {
@ -229,6 +300,15 @@ void Tree::setCopy(const Tree& other) {
m_data = reinterpret_cast<LARGEST_TYPE>(ptr); m_data = reinterpret_cast<LARGEST_TYPE>(ptr);
break; break;
} }
case Type::PAIRS: {
auto otherData = reinterpret_cast<std::vector<std::pair<oatpp::String, Tree>> *>(other.m_data);
if(otherData == nullptr) {
throw std::runtime_error("[oatpp::data::Tree::setCopy()]: other.data is null, other.type is 'PAIRS'");
}
auto ptr = new std::vector<std::pair<oatpp::String, Tree>>(*otherData);
m_data = reinterpret_cast<LARGEST_TYPE>(ptr);
break;
}
default: default:
m_data = other.m_data; m_data = other.m_data;
@ -239,11 +319,16 @@ void Tree::setCopy(const Tree& other) {
} }
void Tree::setMove(Tree&& other) { void Tree::setMove(Tree&& other) {
deleteValueObject(); deleteValueObject();
m_type = other.m_type; m_type = other.m_type;
m_data = other.m_data; m_data = other.m_data;
m_attributes = std::move(other.m_attributes);
other.m_type = Type::NULL_VALUE; other.m_type = Type::NULL_VALUE;
other.m_data = 0; other.m_data = 0;
} }
void Tree::setNull() { void Tree::setNull() {
@ -298,6 +383,13 @@ void Tree::setMap(const Map& value) {
m_data = reinterpret_cast<LARGEST_TYPE>(data); m_data = reinterpret_cast<LARGEST_TYPE>(data);
} }
void Tree::setPairs(const std::vector<std::pair<oatpp::String, Tree>>& value) {
deleteValueObject();
m_type = Type::PAIRS;
auto data = new std::vector<std::pair<oatpp::String, Tree>>(value);
m_data = reinterpret_cast<LARGEST_TYPE>(data);
}
bool Tree::isNull() const { bool Tree::isNull() const {
return m_type == Type::NULL_VALUE; return m_type == Type::NULL_VALUE;
} }
@ -332,6 +424,7 @@ bool Tree::isPrimitive() const {
case Type::STRING: case Type::STRING:
case Type::VECTOR: case Type::VECTOR:
case Type::MAP: case Type::MAP:
case Type::PAIRS:
default: default:
return false; return false;
} }
@ -367,6 +460,7 @@ v_int32 Tree::primitiveDataSize() const {
case Type::STRING: case Type::STRING:
case Type::VECTOR: case Type::VECTOR:
case Type::MAP: case Type::MAP:
case Type::PAIRS:
default: default:
return -1; return -1;
} }
@ -397,6 +491,7 @@ bool Tree::isFloatPrimitive() const {
case Type::STRING: case Type::STRING:
case Type::VECTOR: case Type::VECTOR:
case Type::MAP: case Type::MAP:
case Type::PAIRS:
default: default:
return false; return false;
} }
@ -428,6 +523,7 @@ bool Tree::isIntPrimitive() const {
case Type::STRING: case Type::STRING:
case Type::VECTOR: case Type::VECTOR:
case Type::MAP: case Type::MAP:
case Type::PAIRS:
default: default:
return false; return false;
} }
@ -475,6 +571,14 @@ const Tree::Map& Tree::getMap() const {
return *data; return *data;
} }
const std::vector<std::pair<oatpp::String, Tree>>& Tree::getPairs() const {
if(m_type != Type::PAIRS) {
throw std::runtime_error("[oatpp::data::Tree::getPairs()]: NOT a PAIRS.");
}
auto data = reinterpret_cast<const std::vector<std::pair<oatpp::String, Tree>>*>(m_data);
return *data;
}
std::vector<Tree>& Tree::getVector() { std::vector<Tree>& Tree::getVector() {
if(m_type == Type::UNDEFINED) { if(m_type == Type::UNDEFINED) {
setVector({}); setVector({});
@ -497,6 +601,25 @@ Tree::Map& Tree::getMap() {
return *data; return *data;
} }
std::vector<std::pair<oatpp::String, Tree>>& Tree::getPairs() {
if(m_type == Type::UNDEFINED) {
setPairs({});
}
if(m_type != Type::MAP) {
throw std::runtime_error("[oatpp::data::Tree::getMap()]: NOT a MAP.");
}
auto data = reinterpret_cast<std::vector<std::pair<oatpp::String, Tree>>*>(m_data);
return *data;
}
Tree::Attributes& Tree::attributes() {
return m_attributes;
}
const Tree::Attributes& Tree::attributes() const {
return m_attributes;
}
oatpp::String Tree::debugPrint(v_uint32 indent0, v_uint32 indentDelta, bool firstLineIndent) const { oatpp::String Tree::debugPrint(v_uint32 indent0, v_uint32 indentDelta, bool firstLineIndent) const {
stream::BufferOutputStream ss; stream::BufferOutputStream ss;
@ -588,7 +711,7 @@ oatpp::String Tree::debugPrint(v_uint32 indent0, v_uint32 indentDelta, bool firs
} }
case Type::VECTOR: { case Type::VECTOR: {
ss << "[\n"; ss << "VECTOR [\n";
auto& vector = getVector(); auto& vector = getVector();
for(auto& v : vector) { for(auto& v : vector) {
ss << v.debugPrint(indent0 + indentDelta, indentDelta) << "\n"; ss << v.debugPrint(indent0 + indentDelta, indentDelta) << "\n";
@ -597,7 +720,7 @@ oatpp::String Tree::debugPrint(v_uint32 indent0, v_uint32 indentDelta, bool firs
break; break;
} }
case Type::MAP: { case Type::MAP: {
ss << "{\n"; ss << "MAP {\n";
auto& map = getMap(); auto& map = getMap();
for(v_uint32 i = 0; i < map.size(); i ++) { for(v_uint32 i = 0; i < map.size(); i ++) {
const auto& node = map[i]; const auto& node = map[i];
@ -606,6 +729,15 @@ oatpp::String Tree::debugPrint(v_uint32 indent0, v_uint32 indentDelta, bool firs
ss << indentStr0 << "}"; ss << indentStr0 << "}";
break; break;
} }
case Type::PAIRS: {
ss << "PAIRS {\n";
auto& pairs = getPairs();
for(auto& node : pairs) {
ss << indentStr0 << indentDeltaStr << node.first << ": " << node.second.debugPrint(indent0 + indentDelta, indentDelta, false) << "\n";
}
ss << indentStr0 << "}";
break;
}
default: default:
break; break;
@ -615,4 +747,4 @@ oatpp::String Tree::debugPrint(v_uint32 indent0, v_uint32 indentDelta, bool firs
} }
}} }}}

View File

@ -27,7 +27,7 @@
#include "oatpp/Types.hpp" #include "oatpp/Types.hpp"
namespace oatpp { namespace data { namespace oatpp { namespace data { namespace mapping {
class Tree { class Tree {
public: public:
@ -58,7 +58,8 @@ public:
STRING = 14, STRING = 14,
VECTOR = 15, VECTOR = 15,
MAP = 16 MAP = 16,
PAIRS = 17
}; };
@ -84,6 +85,28 @@ public:
}; };
public:
class Attributes {
private:
std::unordered_map<oatpp::String, oatpp::String>* m_attributes;
public:
Attributes();
Attributes(const Attributes& other);
Attributes(Attributes&& other) noexcept;
Attributes& operator = (const Attributes& other);
Attributes& operator = (Attributes&& other) noexcept;
~Attributes();
bool empty() const;
v_uint64 size() const;
};
private: private:
typedef v_uint64 LARGEST_TYPE; typedef v_uint64 LARGEST_TYPE;
private: private:
@ -91,6 +114,7 @@ private:
private: private:
Type m_type; Type m_type;
LARGEST_TYPE m_data; LARGEST_TYPE m_data;
Attributes m_attributes;
public: public:
Tree(); Tree();
@ -104,11 +128,7 @@ public:
setValue<T>(value); setValue<T>(value);
} }
explicit Tree(const oatpp::String& value) explicit Tree(const oatpp::String& value);
: Tree()
{
setString(value);
}
~Tree(); ~Tree();
@ -121,10 +141,7 @@ public:
return *this; return *this;
} }
Tree& operator = (const oatpp::String& value) { Tree& operator = (const oatpp::String& value);
setString(value);
return *this;
}
template <typename T, typename enabled = typename NodePrimitiveType<T>::value_type> template <typename T, typename enabled = typename NodePrimitiveType<T>::value_type>
operator T () const { operator T () const {
@ -154,6 +171,7 @@ public:
case Type::STRING: case Type::STRING:
case Type::VECTOR: case Type::VECTOR:
case Type::MAP: case Type::MAP:
case Type::PAIRS:
default: default:
break; break;
@ -205,6 +223,7 @@ public:
void setVector(const std::vector<Tree>& value); void setVector(const std::vector<Tree>& value);
void setVector(v_uint64 size); void setVector(v_uint64 size);
void setMap(const Map& value); void setMap(const Map& value);
void setPairs(const std::vector<std::pair<oatpp::String, Tree>>& value);
bool isNull() const; bool isNull() const;
bool isUndefined() const; bool isUndefined() const;
@ -221,9 +240,14 @@ public:
const std::vector<Tree>& getVector() const; const std::vector<Tree>& getVector() const;
const Map& getMap() const; const Map& getMap() const;
const std::vector<std::pair<oatpp::String, Tree>>& getPairs() const;
std::vector<Tree>& getVector(); std::vector<Tree>& getVector();
Map& getMap(); Map& getMap();
std::vector<std::pair<oatpp::String, Tree>>& getPairs();
Attributes& attributes();
const Attributes& attributes() const;
oatpp::String debugPrint(v_uint32 indent0 = 0, v_uint32 indentDelta = 2, bool firstLineIndent = true) const; oatpp::String debugPrint(v_uint32 indent0 = 0, v_uint32 indentDelta = 2, bool firstLineIndent = true) const;
@ -309,6 +333,6 @@ struct Tree::NodePrimitiveType<v_float64> {
typedef v_float64 value_type; typedef v_float64 value_type;
}; };
}} }}}
#endif //oatpp_data_mapping_Tree_hpp #endif //oatpp_data_mapping_Tree_hpp

View File

@ -27,7 +27,7 @@
#include "oatpp/data/stream/BufferStream.hpp" #include "oatpp/data/stream/BufferStream.hpp"
#include "oatpp/utils/Conversion.hpp" #include "oatpp/utils/Conversion.hpp"
namespace oatpp { namespace data { namespace oatpp { namespace data { namespace mapping {
oatpp::String TreeToObjectMapper::MappingState::errorStacktrace() const { oatpp::String TreeToObjectMapper::MappingState::errorStacktrace() const {
stream::BufferOutputStream ss; stream::BufferOutputStream ss;
@ -382,4 +382,4 @@ oatpp::Void TreeToObjectMapper::mapObject(TreeToObjectMapper* mapper, MappingSta
} }
}} }}}

View File

@ -27,7 +27,7 @@
#include "./Tree.hpp" #include "./Tree.hpp"
namespace oatpp { namespace data { namespace oatpp { namespace data { namespace mapping {
class TreeToObjectMapper : public base::Countable { class TreeToObjectMapper : public base::Countable {
public: public:
@ -90,6 +90,6 @@ public:
}; };
}} }}}
#endif //oatpp_data_mapping_TreeToObjectMapper_hpp #endif //oatpp_data_mapping_TreeToObjectMapper_hpp

View File

@ -68,6 +68,8 @@
#include "oatpp/async/Coroutine.hpp" #include "oatpp/async/Coroutine.hpp"
#include "oatpp/Types.hpp" #include "oatpp/Types.hpp"
#include "oatpp/data/mapping/Tree.hpp"
#include "oatpp/Environment.hpp" #include "oatpp/Environment.hpp"
#include <iostream> #include <iostream>