123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218 |
- #ifndef NODE_DETAIL_IMPL_H_62B23520_7C8E_11DE_8A39_0800200C9A66
- #define NODE_DETAIL_IMPL_H_62B23520_7C8E_11DE_8A39_0800200C9A66
- #if defined(_MSC_VER) || \
- (defined(__GNUC__) && (__GNUC__ == 3 && __GNUC_MINOR__ >= 4) || \
- (__GNUC__ >= 4)) // GCC supports "pragma once" correctly since 3.4
- #pragma once
- #endif
- #include "yaml-cpp/node/detail/node.h"
- #include "yaml-cpp/node/detail/node_data.h"
- #include <type_traits>
- namespace YAML {
- namespace detail {
- template <typename Key, typename Enable = void>
- struct get_idx {
- static node* get(const std::vector<node*>& /* sequence */,
- const Key& /* key */, shared_memory_holder /* pMemory */) {
- return nullptr;
- }
- };
- template <typename Key>
- struct get_idx<Key,
- typename std::enable_if<std::is_unsigned<Key>::value &&
- !std::is_same<Key, bool>::value>::type> {
- static node* get(const std::vector<node*>& sequence, const Key& key,
- shared_memory_holder /* pMemory */) {
- return key < sequence.size() ? sequence[key] : nullptr;
- }
- static node* get(std::vector<node*>& sequence, const Key& key,
- shared_memory_holder pMemory) {
- if (key > sequence.size() || (key > 0 && !sequence[key - 1]->is_defined()))
- return 0;
- if (key == sequence.size())
- sequence.push_back(&pMemory->create_node());
- return sequence[key];
- }
- };
- template <typename Key>
- struct get_idx<Key, typename std::enable_if<std::is_signed<Key>::value>::type> {
- static node* get(const std::vector<node*>& sequence, const Key& key,
- shared_memory_holder pMemory) {
- return key >= 0 ? get_idx<std::size_t>::get(
- sequence, static_cast<std::size_t>(key), pMemory)
- : nullptr;
- }
- static node* get(std::vector<node*>& sequence, const Key& key,
- shared_memory_holder pMemory) {
- return key >= 0 ? get_idx<std::size_t>::get(
- sequence, static_cast<std::size_t>(key), pMemory)
- : nullptr;
- }
- };
- template <typename Key, typename Enable = void>
- struct remove_idx {
- static bool remove(std::vector<node*>&, const Key&) { return false; }
- };
- template <typename Key>
- struct remove_idx<
- Key, typename std::enable_if<std::is_unsigned<Key>::value &&
- !std::is_same<Key, bool>::value>::type> {
- static bool remove(std::vector<node*>& sequence, const Key& key) {
- if (key >= sequence.size()) {
- return false;
- } else {
- sequence.erase(sequence.begin() + key);
- return true;
- }
- }
- };
- template <typename Key>
- struct remove_idx<Key,
- typename std::enable_if<std::is_signed<Key>::value>::type> {
- static bool remove(std::vector<node*>& sequence, const Key& key) {
- return key >= 0 ? remove_idx<std::size_t>::remove(
- sequence, static_cast<std::size_t>(key))
- : false;
- }
- };
- template <typename T>
- inline bool node::equals(const T& rhs, shared_memory_holder pMemory) {
- T lhs;
- if (convert<T>::decode(Node(*this, pMemory), lhs)) {
- return lhs == rhs;
- }
- return false;
- }
- inline bool node::equals(const char* rhs, shared_memory_holder pMemory) {
- return equals<std::string>(rhs, pMemory);
- }
- // indexing
- template <typename Key>
- inline node* node_data::get(const Key& key,
- shared_memory_holder pMemory) const {
- switch (m_type) {
- case NodeType::Map:
- break;
- case NodeType::Undefined:
- case NodeType::Null:
- return nullptr;
- case NodeType::Sequence:
- if (node* pNode = get_idx<Key>::get(m_sequence, key, pMemory))
- return pNode;
- return nullptr;
- case NodeType::Scalar:
- throw BadSubscript(key);
- }
- for (node_map::const_iterator it = m_map.begin(); it != m_map.end(); ++it) {
- if (it->first->equals(key, pMemory)) {
- return it->second;
- }
- }
- return nullptr;
- }
- template <typename Key>
- inline node& node_data::get(const Key& key, shared_memory_holder pMemory) {
- switch (m_type) {
- case NodeType::Map:
- break;
- case NodeType::Undefined:
- case NodeType::Null:
- case NodeType::Sequence:
- if (node* pNode = get_idx<Key>::get(m_sequence, key, pMemory)) {
- m_type = NodeType::Sequence;
- return *pNode;
- }
- convert_to_map(pMemory);
- break;
- case NodeType::Scalar:
- throw BadSubscript(key);
- }
- for (node_map::const_iterator it = m_map.begin(); it != m_map.end(); ++it) {
- if (it->first->equals(key, pMemory)) {
- return *it->second;
- }
- }
- node& k = convert_to_node(key, pMemory);
- node& v = pMemory->create_node();
- insert_map_pair(k, v);
- return v;
- }
- template <typename Key>
- inline bool node_data::remove(const Key& key, shared_memory_holder pMemory) {
- if (m_type == NodeType::Sequence) {
- return remove_idx<Key>::remove(m_sequence, key);
- } else if (m_type == NodeType::Map) {
- kv_pairs::iterator it = m_undefinedPairs.begin();
- while (it != m_undefinedPairs.end()) {
- kv_pairs::iterator jt = std::next(it);
- if (it->first->equals(key, pMemory)) {
- m_undefinedPairs.erase(it);
- }
- it = jt;
- }
- for (node_map::iterator iter = m_map.begin(); iter != m_map.end(); ++iter) {
- if (iter->first->equals(key, pMemory)) {
- m_map.erase(iter);
- return true;
- }
- }
- }
- return false;
- }
- // map
- template <typename Key, typename Value>
- inline void node_data::force_insert(const Key& key, const Value& value,
- shared_memory_holder pMemory) {
- switch (m_type) {
- case NodeType::Map:
- break;
- case NodeType::Undefined:
- case NodeType::Null:
- case NodeType::Sequence:
- convert_to_map(pMemory);
- break;
- case NodeType::Scalar:
- throw BadInsert();
- }
- node& k = convert_to_node(key, pMemory);
- node& v = convert_to_node(value, pMemory);
- insert_map_pair(k, v);
- }
- template <typename T>
- inline node& node_data::convert_to_node(const T& rhs,
- shared_memory_holder pMemory) {
- Node value = convert<T>::encode(rhs);
- value.EnsureNodeExists();
- pMemory->merge(*value.m_pMemory);
- return *value.m_pNode;
- }
- }
- }
- #endif // NODE_DETAIL_IMPL_H_62B23520_7C8E_11DE_8A39_0800200C9A66
|