dictionary.hpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. /*************************************************************************
  2. *
  3. * Copyright 2019 Realm Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. *
  17. **************************************************************************/
  18. #ifndef REALM_DICTIONARY_HPP
  19. #define REALM_DICTIONARY_HPP
  20. #include <realm/collection.hpp>
  21. #include <realm/obj.hpp>
  22. #include <realm/mixed.hpp>
  23. #include <realm/array_mixed.hpp>
  24. #include <realm/dictionary_cluster_tree.hpp>
  25. namespace realm {
  26. class DictionaryClusterTree;
  27. class Dictionary final : public CollectionBaseImpl<CollectionBase> {
  28. public:
  29. using Base = CollectionBaseImpl<CollectionBase>;
  30. class Iterator;
  31. Dictionary() {}
  32. ~Dictionary();
  33. Dictionary(const Obj& obj, ColKey col_key);
  34. Dictionary(const Dictionary& other)
  35. : Base(static_cast<const Base&>(other))
  36. , m_key_type(other.m_key_type)
  37. {
  38. *this = other;
  39. }
  40. Dictionary& operator=(const Dictionary& other);
  41. bool operator==(const Dictionary& other) const noexcept
  42. {
  43. return CollectionBaseImpl<CollectionBase>::operator==(other);
  44. }
  45. DataType get_key_data_type() const;
  46. DataType get_value_data_type() const;
  47. std::pair<Mixed, Mixed> get_pair(size_t ndx) const;
  48. Mixed get_key(size_t ndx) const;
  49. // Overriding members of CollectionBase:
  50. std::unique_ptr<CollectionBase> clone_collection() const final;
  51. size_t size() const final;
  52. bool is_null(size_t ndx) const final;
  53. Mixed get_any(size_t ndx) const final;
  54. size_t find_any(Mixed value) const final;
  55. size_t find_any_key(Mixed value) const noexcept;
  56. util::Optional<Mixed> min(size_t* return_ndx = nullptr) const final;
  57. util::Optional<Mixed> max(size_t* return_ndx = nullptr) const final;
  58. util::Optional<Mixed> sum(size_t* return_cnt = nullptr) const final;
  59. util::Optional<Mixed> avg(size_t* return_cnt = nullptr) const final;
  60. void sort(std::vector<size_t>& indices, bool ascending = true) const final;
  61. void distinct(std::vector<size_t>& indices, util::Optional<bool> sort_order = util::none) const final;
  62. void sort_keys(std::vector<size_t>& indices, bool ascending = true) const;
  63. void distinct_keys(std::vector<size_t>& indices, util::Optional<bool> sort_order = util::none) const;
  64. void create();
  65. // first points to inserted/updated element.
  66. // second is true if the element was inserted
  67. std::pair<Iterator, bool> insert(Mixed key, Mixed value);
  68. std::pair<Iterator, bool> insert(Mixed key, const Obj& obj);
  69. Obj create_and_insert_linked_object(Mixed key);
  70. // throws std::out_of_range if key is not found
  71. Mixed get(Mixed key) const;
  72. // Noexcept version
  73. util::Optional<Mixed> try_get(Mixed key) const noexcept;
  74. // adds entry if key is not found
  75. const Mixed operator[](Mixed key);
  76. Obj get_object(StringData key)
  77. {
  78. auto val = try_get(key);
  79. Obj obj;
  80. if (val && (*val).is_type(type_Link)) {
  81. return get_target_table()->get_object((*val).get<ObjKey>());
  82. }
  83. return obj;
  84. }
  85. bool contains(Mixed key) const noexcept;
  86. Iterator find(Mixed key) const noexcept;
  87. void erase(Mixed key);
  88. void erase(Iterator it);
  89. bool try_erase(Mixed key);
  90. void nullify(Mixed);
  91. void remove_backlinks(CascadeState& state) const;
  92. void clear() final;
  93. template <class T>
  94. void for_all_values(T&& f)
  95. {
  96. if (m_clusters) {
  97. ArrayMixed leaf(m_obj.get_alloc());
  98. // Iterate through cluster and call f on each value
  99. auto trv_func = [&leaf, &f](const Cluster* cluster) {
  100. size_t e = cluster->node_size();
  101. cluster->init_leaf(DictionaryClusterTree::s_values_col, &leaf);
  102. for (size_t i = 0; i < e; i++) {
  103. f(leaf.get(i));
  104. }
  105. // Continue
  106. return false;
  107. };
  108. m_clusters->traverse(trv_func);
  109. }
  110. }
  111. template <class T, class Func>
  112. void for_all_keys(Func&& f)
  113. {
  114. if (m_clusters) {
  115. typename ColumnTypeTraits<T>::cluster_leaf_type leaf(m_obj.get_alloc());
  116. ColKey col = m_clusters->get_keys_column_key();
  117. // Iterate through cluster and call f on each value
  118. auto trv_func = [&leaf, &f, col](const Cluster* cluster) {
  119. size_t e = cluster->node_size();
  120. cluster->init_leaf(col, &leaf);
  121. for (size_t i = 0; i < e; i++) {
  122. f(leaf.get(i));
  123. }
  124. // Continue
  125. return false;
  126. };
  127. m_clusters->traverse(trv_func);
  128. }
  129. }
  130. Iterator begin() const;
  131. Iterator end() const;
  132. static ObjKey get_internal_obj_key(Mixed key)
  133. {
  134. return ObjKey{int64_t(key.hash() & s_hash_mask)};
  135. }
  136. #ifdef REALM_DEBUG
  137. static uint64_t set_hash_mask(uint64_t mask)
  138. {
  139. auto tmp = s_hash_mask;
  140. s_hash_mask = mask;
  141. return tmp;
  142. }
  143. #else
  144. static uint64_t set_hash_mask(uint64_t)
  145. {
  146. return 0;
  147. }
  148. #endif
  149. private:
  150. template <typename T, typename Op>
  151. friend class CollectionColumnAggregate;
  152. friend class DictionaryLinkValues;
  153. mutable std::unique_ptr<DictionaryClusterTree> m_clusters;
  154. DataType m_key_type = type_String;
  155. #ifdef REALM_DEBUG
  156. static uint64_t s_hash_mask;
  157. #else
  158. static constexpr uint64_t s_hash_mask = 0x7FFFFFFFFFFFFFFFULL;
  159. #endif
  160. bool init_from_parent() const final;
  161. Mixed do_get(const ClusterNode::State&) const;
  162. Mixed do_get_key(const ClusterNode::State&) const;
  163. std::pair<Mixed, Mixed> do_get_pair(const ClusterNode::State&) const;
  164. bool clear_backlink(Mixed value, CascadeState& state) const;
  165. void align_indices(std::vector<size_t>& indices) const;
  166. void swap_content(Array& fields1, Array& fields2, size_t index1, size_t index2);
  167. ObjKey handle_collision_in_erase(const Mixed& key, ObjKey k, ClusterNode::State& state);
  168. friend struct CollectionIterator<Dictionary>;
  169. };
  170. class Dictionary::Iterator : public ClusterTree::Iterator {
  171. public:
  172. typedef std::forward_iterator_tag iterator_category;
  173. typedef std::pair<const Mixed, Mixed> value_type;
  174. typedef ptrdiff_t difference_type;
  175. typedef const value_type* pointer;
  176. typedef const value_type& reference;
  177. value_type operator*() const;
  178. Iterator& operator++()
  179. {
  180. return static_cast<Iterator&>(ClusterTree::Iterator::operator++());
  181. }
  182. Iterator& operator+=(ptrdiff_t adj)
  183. {
  184. return static_cast<Iterator&>(ClusterTree::Iterator::operator+=(adj));
  185. }
  186. Iterator operator+(ptrdiff_t n) const
  187. {
  188. Iterator ret(*this);
  189. ret += n;
  190. return ret;
  191. }
  192. private:
  193. friend class Dictionary;
  194. using ClusterTree::Iterator::get_position;
  195. DataType m_key_type;
  196. Iterator(const Dictionary* dict, size_t pos);
  197. };
  198. // An interface used when the value type of the dictionary consists of
  199. // links to a single table. Implementation of the ObjList interface on
  200. // top of a Dictionary of objects. This is the dictionary equivilent of
  201. // LnkLst and LnkSet.
  202. class DictionaryLinkValues final : public ObjCollectionBase<CollectionBase> {
  203. public:
  204. DictionaryLinkValues() = default;
  205. DictionaryLinkValues(const Obj& obj, ColKey col_key);
  206. DictionaryLinkValues(const Dictionary& source);
  207. // Overrides of ObjList:
  208. ObjKey get_key(size_t ndx) const final;
  209. bool is_obj_valid(size_t ndx) const noexcept final;
  210. Obj get_object(size_t row_ndx) const final;
  211. // Overrides of CollectionBase, these simply forward to the underlying dictionary.
  212. size_t size() const final
  213. {
  214. return m_source.size();
  215. }
  216. bool is_null(size_t ndx) const final
  217. {
  218. return m_source.is_null(ndx);
  219. }
  220. Mixed get_any(size_t ndx) const final
  221. {
  222. return m_source.get_any(ndx);
  223. }
  224. void clear() final
  225. {
  226. m_source.clear();
  227. }
  228. util::Optional<Mixed> min(size_t* return_ndx = nullptr) const final
  229. {
  230. return m_source.min(return_ndx);
  231. }
  232. util::Optional<Mixed> max(size_t* return_ndx = nullptr) const final
  233. {
  234. return m_source.max(return_ndx);
  235. }
  236. util::Optional<Mixed> sum(size_t* return_cnt = nullptr) const final
  237. {
  238. return m_source.sum(return_cnt);
  239. }
  240. util::Optional<Mixed> avg(size_t* return_cnt = nullptr) const final
  241. {
  242. return m_source.avg(return_cnt);
  243. }
  244. std::unique_ptr<CollectionBase> clone_collection() const final
  245. {
  246. return std::make_unique<DictionaryLinkValues>(m_source);
  247. }
  248. LinkCollectionPtr clone_obj_list() const final
  249. {
  250. return std::make_unique<DictionaryLinkValues>(m_source);
  251. }
  252. void sort(std::vector<size_t>& indices, bool ascending = true) const final
  253. {
  254. m_source.sort(indices, ascending);
  255. }
  256. void distinct(std::vector<size_t>& indices, util::Optional<bool> sort_order = util::none) const final
  257. {
  258. m_source.distinct(indices, sort_order);
  259. }
  260. size_t find_any(Mixed value) const final
  261. {
  262. return m_source.find_any(value);
  263. }
  264. const Obj& get_obj() const noexcept final
  265. {
  266. return m_source.get_obj();
  267. }
  268. ColKey get_col_key() const noexcept final
  269. {
  270. return m_source.get_col_key();
  271. }
  272. bool has_changed() const final
  273. {
  274. return m_source.has_changed();
  275. }
  276. // Overrides of ObjCollectionBase:
  277. bool do_update_if_needed() const final
  278. {
  279. return m_source.update_if_needed();
  280. }
  281. bool do_init_from_parent() const final
  282. {
  283. return m_source.init_from_parent();
  284. }
  285. BPlusTree<ObjKey>* get_mutable_tree() const
  286. {
  287. // We are faking being an ObjList because the underlying storage is not
  288. // actually a BPlusTree<ObjKey> for dictionaries it is all mixed values.
  289. // But this is ok, because we don't need to deal with unresolved link
  290. // maintenance because they are not hidden from view in dictionaries in
  291. // the same way as for LnkSet and LnkLst. This means that the functions
  292. // that call get_mutable_tree do not need to do anything for dictionaries.
  293. return nullptr;
  294. }
  295. private:
  296. Dictionary m_source;
  297. };
  298. inline std::pair<Dictionary::Iterator, bool> Dictionary::insert(Mixed key, const Obj& obj)
  299. {
  300. return insert(key, Mixed(obj.get_link()));
  301. }
  302. inline std::unique_ptr<CollectionBase> Dictionary::clone_collection() const
  303. {
  304. return m_obj.get_dictionary_ptr(m_col_key);
  305. }
  306. } // namespace realm
  307. #endif /* SRC_REALM_DICTIONARY_HPP_ */