// // Struct.h // // Library: Foundation // Package: Dynamic // Module: Struct // // Definition of the Struct class. // // Copyright (c) 2007, Applied Informatics Software Engineering GmbH. // and Contributors. // // SPDX-License-Identifier: BSL-1.0 // #ifndef Foundation_Struct_INCLUDED #define Foundation_Struct_INCLUDED #include "Poco/Foundation.h" #include "Poco/Dynamic/Var.h" #include "Poco/Dynamic/VarHolder.h" #include "Poco/SharedPtr.h" #include "Poco/OrderedMap.h" #include "Poco/OrderedSet.h" #include #include namespace Poco { namespace Dynamic { template std::string structToString(const S& data, bool wrap = true) /// Utility function for converting DynamicStruct to std::string. /// Set wrap to false in order to prevent string values wrapping /// (useful to prevent JSON fragments from being treated as strings). { std::string val; val.append("{ "); I it = data.begin(); I itEnd = data.end(); if (!data.empty()) { Var key(it->first); Impl::appendJSONKey(val, key); val.append(": "); Impl::appendJSONValue(val, it->second, wrap); ++it; } for (; it != itEnd; ++it) { val.append(", "); Var key(it->first); Impl::appendJSONKey(val, key); val.append(": "); Impl::appendJSONValue(val, it->second, wrap); } val.append(" }"); return val; } template , typename S = std::set> class Struct /// Struct allows to define a named collection of Var objects. { public: typedef M Data; typedef S NameSet; typedef typename Data::iterator Iterator; typedef typename Data::const_iterator ConstIterator; typedef typename Struct::Data::value_type ValueType; typedef typename Struct::Data::size_type SizeType; typedef typename std::pair::Iterator, bool> InsRetVal; typedef typename Poco::SharedPtr> Ptr; Struct(): _data() /// Creates an empty Struct { } Struct(const Data& val): _data(val) /// Creates the Struct from the given value. { } template Struct(const std::map& val) { assignMap(val); } template Struct(const OrderedMap& val) { assignMap(val); } virtual ~Struct() /// Destroys the Struct. { } inline Var& operator [] (const K& name) /// Returns the Var with the given name, creates an entry if not found. { return _data[name]; } const Var& operator [] (const K& name) const /// Returns the Var with the given name, throws a /// NotFoundException if the data member is not found. { ConstIterator it = find(name); if (it == end()) throw NotFoundException(name); return it->second; } inline bool contains(const K& name) const /// Returns true if the Struct contains a member with the given name { return find(name) != end(); } inline Iterator find(const K& name) /// Returns an iterator, pointing to the pair containing /// the element, or it returns end() if the member was not found { return _data.find(name); } inline ConstIterator find(const K& name) const /// Returns a const iterator, pointing to the pair containing /// the element, or it returns end() if the member was not found { return _data.find(name); } inline Iterator end() /// Returns the end iterator for the Struct { return _data.end(); } inline ConstIterator end() const /// Returns the end const iterator for the Struct { return _data.end(); } inline Iterator begin() /// Returns the begin iterator for the Struct { return _data.begin(); } inline ConstIterator begin() const /// Returns the begin const iterator for the Struct { return _data.begin(); } template inline InsRetVal insert(const K& key, const T& value) /// Inserts a pair into the Struct, /// returns a pair containing the iterator and a boolean which /// indicates success or not (is true, when insert succeeded, false, /// when already another element was present, in this case Iterator /// points to that other element) { // fix: SunPro C++ is silly ... ValueType valueType(key, value); return insert(valueType); } inline InsRetVal insert(const ValueType& aPair) /// Inserts a pair into the Struct, /// returns a pair containing the iterator and a boolean which /// indicates success or not (is true, when insert succeeded, false, /// when already another element was present, in this case Iterator /// points to that other element) { return _data.insert(aPair); } inline SizeType erase(const K& key) /// Erases the element if found, returns number of elements deleted { return _data.erase(key); } inline void erase(Iterator& it) /// Erases the element at the given position { _data.erase(it); } inline void clear() /// Remove all elements from the struct { _data.clear(); } inline void swap(Struct& other) noexcept /// Swap content of Struct with another Struct { _data.swap(other._data); } inline bool empty() const /// Returns true if the Struct doesn't contain any members { return _data.empty(); } SizeType size() const /// Returns the number of members the Struct contains { return _data.size(); } inline NameSet members() const /// Returns a sorted collection containing all member names { NameSet keys; ConstIterator it = begin(); ConstIterator itEnd = end(); for (; it != itEnd; ++it) keys.insert(it->first); return keys; } inline Var getVar(const K& key) const /// Returns the var value of the element with the given name. /// Throws a NotFoundException if the key does not exist. { ConstIterator it = find(key); if(it == end()) { throw NotFoundException("Key not found in Struct"); } return it->second; } template inline Var getVar(const K& key, const DefT& defaultValue) const /// Returns the var value of the element with the given name. /// or defaultValue if none is found. { ConstIterator it = find(key); if(it == end()) { return defaultValue; } return it->second; } std::string toString(bool wrap = true) const /// Returns the DynamicStruct as string. /// /// To prevent unwanted string wrapping /// (eg. when a value is JSON string), /// `wrap` should be false. Note, however, /// that wrap argument is of a limited utility /// because it applies to the entire Struct, /// so it should not be relied on when mixed content /// (ie. plain string, which should be wrapped, /// and JSON-as-string entries, which shouldn't) /// is held. { return structToString(_data, wrap); } private: template void assignMap(const T& map) { typedef typename T::const_iterator MapConstIterator; MapConstIterator it = map.begin(); MapConstIterator end = map.end(); for (; it != end; ++it) _data.insert(ValueType(it->first, Var(it->second))); } Data _data; }; #if defined(POCO_OS_FAMILY_WINDOWS) extern template class Struct; extern template class Struct; extern template class Struct, Poco::OrderedSet>; extern template class Struct, OrderedSet>; #else extern template class Foundation_API Struct; extern template class Foundation_API Struct; extern template class Foundation_API Struct, Poco::OrderedSet>; extern template class Foundation_API Struct, OrderedSet>; #endif template <> class VarHolderImpl, std::set>>: public VarHolder { public: typedef std::string KeyType; typedef std::map MapType; typedef std::set SetType; typedef Struct ValueType; VarHolderImpl(const ValueType& val): _val(val) { } ~VarHolderImpl() { } const std::type_info& type() const { return typeid(ValueType); } void convert(Int8&) const { throw BadCastException("Cannot cast Struct type to Int8"); } void convert(Int16&) const { throw BadCastException("Cannot cast Struct type to Int16"); } void convert(Int32&) const { throw BadCastException("Cannot cast Struct type to Int32"); } void convert(Int64&) const { throw BadCastException("Cannot cast Struct type to Int64"); } void convert(UInt8&) const { throw BadCastException("Cannot cast Struct type to UInt8"); } void convert(UInt16&) const { throw BadCastException("Cannot cast Struct type to UInt16"); } void convert(UInt32&) const { throw BadCastException("Cannot cast Struct type to UInt32"); } void convert(UInt64&) const { throw BadCastException("Cannot cast Struct type to UInt64"); } void convert(bool&) const { throw BadCastException("Cannot cast Struct type to bool"); } void convert(float&) const { throw BadCastException("Cannot cast Struct type to float"); } void convert(double&) const { throw BadCastException("Cannot cast Struct type to double"); } void convert(char&) const { throw BadCastException("Cannot cast Struct type to char"); } void convert(std::string& val) const { val = structToString(_val); } void convert(Poco::DateTime&) const { throw BadCastException("Struct -> Poco::DateTime"); } void convert(Poco::LocalDateTime&) const { throw BadCastException("Struct -> Poco::LocalDateTime"); } void convert(Poco::Timestamp&) const { throw BadCastException("Struct -> Poco::Timestamp"); } VarHolder* clone(Placeholder* pVarHolder = 0) const { return cloneHolder(pVarHolder, _val); } const ValueType& value() const { return _val; } bool isArray() const { return false; } bool isStruct() const { return true; } bool isOrdered() const { return false; } bool isInteger() const { return false; } bool isSigned() const { return false; } bool isNumeric() const { return false; } bool isString() const { return false; } std::size_t size() const { return _val.size(); } Var& operator [] (const KeyType& name) { return _val[name]; } const Var& operator [] (const KeyType& name) const { return _val[name]; } private: ValueType _val; }; template <> class VarHolderImpl, std::set>> : public VarHolder { public: typedef int KeyType; typedef std::map MapType; typedef std::set SetType; typedef Struct ValueType; VarHolderImpl(const ValueType& val) : _val(val) { } ~VarHolderImpl() { } const std::type_info& type() const { return typeid(ValueType); } void convert(Int8&) const { throw BadCastException("Cannot cast Struct type to Int8"); } void convert(Int16&) const { throw BadCastException("Cannot cast Struct type to Int16"); } void convert(Int32&) const { throw BadCastException("Cannot cast Struct type to Int32"); } void convert(Int64&) const { throw BadCastException("Cannot cast Struct type to Int64"); } void convert(UInt8&) const { throw BadCastException("Cannot cast Struct type to UInt8"); } void convert(UInt16&) const { throw BadCastException("Cannot cast Struct type to UInt16"); } void convert(UInt32&) const { throw BadCastException("Cannot cast Struct type to UInt32"); } void convert(UInt64&) const { throw BadCastException("Cannot cast Struct type to UInt64"); } void convert(bool&) const { throw BadCastException("Cannot cast Struct type to bool"); } void convert(float&) const { throw BadCastException("Cannot cast Struct type to float"); } void convert(double&) const { throw BadCastException("Cannot cast Struct type to double"); } void convert(char&) const { throw BadCastException("Cannot cast Struct type to char"); } void convert(std::string& val) const { val = structToString(_val); } void convert(Poco::DateTime&) const { throw BadCastException("Struct -> Poco::DateTime"); } void convert(Poco::LocalDateTime&) const { throw BadCastException("Struct -> Poco::LocalDateTime"); } void convert(Poco::Timestamp&) const { throw BadCastException("Struct -> Poco::Timestamp"); } VarHolder* clone(Placeholder* pVarHolder = 0) const { return cloneHolder(pVarHolder, _val); } const ValueType& value() const { return _val; } bool isArray() const { return false; } bool isStruct() const { return true; } bool isOrdered() const { return false; } bool isInteger() const { return false; } bool isSigned() const { return false; } bool isNumeric() const { return false; } bool isString() const { return false; } std::size_t size() const { return _val.size(); } Var& operator [] (const KeyType& name) { return _val[name]; } const Var& operator [] (const KeyType& name) const { return _val[name]; } private: ValueType _val; }; template <> class VarHolderImpl, Poco::OrderedSet>> : public VarHolder { public: typedef std::string KeyType; typedef Poco::OrderedMap MapType; typedef Poco::OrderedSet SetType; typedef Struct ValueType; VarHolderImpl(const ValueType& val) : _val(val) { } ~VarHolderImpl() { } const std::type_info& type() const { return typeid(ValueType); } void convert(Int8&) const { throw BadCastException("Cannot cast Struct type to Int8"); } void convert(Int16&) const { throw BadCastException("Cannot cast Struct type to Int16"); } void convert(Int32&) const { throw BadCastException("Cannot cast Struct type to Int32"); } void convert(Int64&) const { throw BadCastException("Cannot cast Struct type to Int64"); } void convert(UInt8&) const { throw BadCastException("Cannot cast Struct type to UInt8"); } void convert(UInt16&) const { throw BadCastException("Cannot cast Struct type to UInt16"); } void convert(UInt32&) const { throw BadCastException("Cannot cast Struct type to UInt32"); } void convert(UInt64&) const { throw BadCastException("Cannot cast Struct type to UInt64"); } void convert(bool&) const { throw BadCastException("Cannot cast Struct type to bool"); } void convert(float&) const { throw BadCastException("Cannot cast Struct type to float"); } void convert(double&) const { throw BadCastException("Cannot cast Struct type to double"); } void convert(char&) const { throw BadCastException("Cannot cast Struct type to char"); } void convert(std::string& val) const { val = structToString(_val); } void convert(Poco::DateTime&) const { throw BadCastException("Struct -> Poco::DateTime"); } void convert(Poco::LocalDateTime&) const { throw BadCastException("Struct -> Poco::LocalDateTime"); } void convert(Poco::Timestamp&) const { throw BadCastException("Struct -> Poco::Timestamp"); } VarHolder* clone(Placeholder* pVarHolder = 0) const { return cloneHolder(pVarHolder, _val); } const ValueType& value() const { return _val; } bool isArray() const { return false; } bool isStruct() const { return true; } bool isOrdered() const { return true; } bool isInteger() const { return false; } bool isSigned() const { return false; } bool isNumeric() const { return false; } bool isString() const { return false; } std::size_t size() const { return _val.size(); } Var& operator [] (const KeyType& name) { return _val[name]; } const Var& operator [] (const KeyType& name) const { return _val[name]; } private: ValueType _val; }; template <> class VarHolderImpl, Poco::OrderedSet>> : public VarHolder { public: typedef int KeyType; typedef Poco::OrderedMap MapType; typedef Poco::OrderedSet SetType; typedef Struct ValueType; VarHolderImpl(const ValueType& val) : _val(val) { } ~VarHolderImpl() { } const std::type_info& type() const { return typeid(ValueType); } void convert(Int8&) const { throw BadCastException("Cannot cast Struct type to Int8"); } void convert(Int16&) const { throw BadCastException("Cannot cast Struct type to Int16"); } void convert(Int32&) const { throw BadCastException("Cannot cast Struct type to Int32"); } void convert(Int64&) const { throw BadCastException("Cannot cast Struct type to Int64"); } void convert(UInt8&) const { throw BadCastException("Cannot cast Struct type to UInt8"); } void convert(UInt16&) const { throw BadCastException("Cannot cast Struct type to UInt16"); } void convert(UInt32&) const { throw BadCastException("Cannot cast Struct type to UInt32"); } void convert(UInt64&) const { throw BadCastException("Cannot cast Struct type to UInt64"); } void convert(bool&) const { throw BadCastException("Cannot cast Struct type to bool"); } void convert(float&) const { throw BadCastException("Cannot cast Struct type to float"); } void convert(double&) const { throw BadCastException("Cannot cast Struct type to double"); } void convert(char&) const { throw BadCastException("Cannot cast Struct type to char"); } void convert(std::string& val) const { val = structToString(_val); } void convert(Poco::DateTime&) const { throw BadCastException("Struct -> Poco::DateTime"); } void convert(Poco::LocalDateTime&) const { throw BadCastException("Struct -> Poco::LocalDateTime"); } void convert(Poco::Timestamp&) const { throw BadCastException("Struct -> Poco::Timestamp"); } VarHolder* clone(Placeholder* pVarHolder = 0) const { return cloneHolder(pVarHolder, _val); } const ValueType& value() const { return _val; } bool isArray() const { return false; } bool isStruct() const { return true; } bool isOrdered() const { return true; } bool isInteger() const { return false; } bool isSigned() const { return false; } bool isNumeric() const { return false; } bool isString() const { return false; } std::size_t size() const { return _val.size(); } Var& operator [] (const KeyType& name) { return _val[name]; } const Var& operator [] (const KeyType& name) const { return _val[name]; } private: ValueType _val; }; } // namespace Dynamic using DynamicStruct = Dynamic::Struct; using OrderedDynamicStruct = Dynamic::Struct, Poco::OrderedSet>; } // namespace Poco #endif // Foundation_Struct_INCLUDED