obj.hpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637
  1. /*************************************************************************
  2. *
  3. * Copyright 2016 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_OBJ_HPP
  19. #define REALM_OBJ_HPP
  20. #include <realm/node.hpp>
  21. #include <realm/table_ref.hpp>
  22. #include <realm/keys.hpp>
  23. #include <realm/mixed.hpp>
  24. #include <map>
  25. #define REALM_CLUSTER_IF
  26. namespace realm {
  27. class ClusterTree;
  28. class Replication;
  29. class TableView;
  30. class CollectionBase;
  31. class CascadeState;
  32. class LstBase;
  33. class SetBase;
  34. class ObjList;
  35. struct GlobalKey;
  36. template <class>
  37. class Lst;
  38. template <class>
  39. class Set;
  40. template <class T>
  41. using LstPtr = std::unique_ptr<Lst<T>>;
  42. using LstBasePtr = std::unique_ptr<LstBase>;
  43. template <class T>
  44. using SetPtr = std::unique_ptr<Set<T>>;
  45. using SetBasePtr = std::unique_ptr<SetBase>;
  46. using CollectionBasePtr = std::unique_ptr<CollectionBase>;
  47. using LinkCollectionPtr = std::unique_ptr<ObjList>;
  48. class LnkLst;
  49. using LnkLstPtr = std::unique_ptr<LnkLst>;
  50. class LnkSet;
  51. using LnkSetPtr = std::unique_ptr<LnkSet>;
  52. template <class>
  53. class Set;
  54. class Dictionary;
  55. class DictionaryLinkValues;
  56. using DictionaryPtr = std::unique_ptr<Dictionary>;
  57. namespace _impl {
  58. class DeepChangeChecker;
  59. }
  60. enum JSONOutputMode {
  61. output_mode_json, // default / existing implementation for outputting realm to json
  62. output_mode_xjson, // extended json as described in the spec
  63. output_mode_xjson_plus, // extended json as described in the spec with additional modifier used for sync
  64. };
  65. /// The status of an accessor after a call to `update_if_needed()`.
  66. enum class UpdateStatus {
  67. /// The owning object or column no longer exist, and the accessor could
  68. /// not be updated. The accessor should be left in a detached state
  69. /// after this, and further calls to `update_if_needed()` are not
  70. /// guaranteed to reattach the accessor.
  71. Detached,
  72. /// The underlying data of the accessor was changed since the last call
  73. /// to `update_if_needed()`. The accessor is still valid.
  74. Updated,
  75. /// The underlying data of the accessor did not change since the last
  76. /// call to `update_if_needed()`, and the accessor is still valid in its
  77. /// current state.
  78. NoChange,
  79. };
  80. // 'Object' would have been a better name, but it clashes with a class in ObjectStore
  81. class Obj {
  82. public:
  83. constexpr Obj()
  84. : m_table(nullptr)
  85. , m_row_ndx(size_t(-1))
  86. , m_storage_version(-1)
  87. , m_valid(false)
  88. {
  89. }
  90. Obj(TableRef table, MemRef mem, ObjKey key, size_t row_ndx);
  91. TableRef get_table() const noexcept
  92. {
  93. return m_table.cast_away_const();
  94. }
  95. Allocator& get_alloc() const;
  96. bool operator==(const Obj& other) const;
  97. ObjKey get_key() const noexcept
  98. {
  99. return m_key;
  100. }
  101. GlobalKey get_object_id() const;
  102. ObjLink get_link() const;
  103. Replication* get_replication() const;
  104. // Check if this object is default constructed
  105. explicit operator bool() const noexcept
  106. {
  107. return m_table != nullptr;
  108. }
  109. /// Check if the object is still alive
  110. bool is_valid() const noexcept;
  111. /// Delete object from table. Object is invalid afterwards.
  112. void remove();
  113. /// Invalidate
  114. /// - this turns the object into a tombstone if links to the object exist.
  115. /// - deletes the object is no links to the object exist.
  116. /// - To be used by the Sync client.
  117. void invalidate();
  118. template <typename U>
  119. U get(ColKey col_key) const;
  120. Mixed get_any(ColKey col_key) const;
  121. Mixed get_any(StringData col_name) const
  122. {
  123. return get_any(get_column_key(col_name));
  124. }
  125. Mixed get_any(std::vector<std::string>::iterator path_start, std::vector<std::string>::iterator path_end) const;
  126. Mixed get_primary_key() const;
  127. template <typename U>
  128. U get(StringData col_name) const
  129. {
  130. return get<U>(get_column_key(col_name));
  131. }
  132. bool is_unresolved(ColKey col_key) const;
  133. size_t get_link_count(ColKey col_key) const;
  134. TableRef get_target_table(ColKey col_key) const;
  135. bool is_null(ColKey col_key) const;
  136. bool is_null(StringData col_name) const
  137. {
  138. return is_null(get_column_key(col_name));
  139. }
  140. bool has_backlinks(bool only_strong_links) const;
  141. size_t get_backlink_count() const;
  142. size_t get_backlink_count(const Table& origin, ColKey origin_col_key) const;
  143. ObjKey get_backlink(const Table& origin, ColKey origin_col_key, size_t backlink_ndx) const;
  144. TableView get_backlink_view(TableRef src_table, ColKey src_col_key);
  145. // To be used by the query system when a single object should
  146. // be tested. Will allow a function to be called in the context
  147. // of the owning cluster.
  148. template <class T>
  149. bool evaluate(T func) const;
  150. void to_json(std::ostream& out, size_t link_depth, const std::map<std::string, std::string>& renames,
  151. std::vector<ObjLink>& followed, JSONOutputMode output_mode) const;
  152. void to_json(std::ostream& out, size_t link_depth, const std::map<std::string, std::string>& renames,
  153. JSONOutputMode output_mode = output_mode_json) const
  154. {
  155. std::vector<ObjLink> followed;
  156. to_json(out, link_depth, renames, followed, output_mode);
  157. }
  158. std::string to_string() const;
  159. // Get the path in a minimal format without including object accessors.
  160. // If you need to obtain additional information for each object in the path,
  161. // you should use get_fat_path() or traverse_path() instead (see below).
  162. struct PathElement;
  163. struct Path {
  164. TableKey top_table;
  165. ObjKey top_objkey;
  166. std::vector<PathElement> path_from_top;
  167. };
  168. Path get_path() const;
  169. // Get the fat path to this object expressed as a vector of fat path elements.
  170. // each Fat path elements include a Obj allowing for low cost access to the
  171. // objects data.
  172. // For a top-level object, the returned vector will be empty.
  173. // For an embedded object, the vector has the top object as first element,
  174. // and the embedded object itself is not included in the path.
  175. struct FatPathElement;
  176. using FatPath = std::vector<FatPathElement>;
  177. FatPath get_fat_path() const;
  178. // For an embedded object, traverse the path leading to this object.
  179. // The PathSizer is called first to set the size of the path
  180. // Then there is one call for each object on that path, starting with the top level object
  181. // The embedded object itself is not considered part of the path.
  182. // Note: You should never provide the path_index for calls to traverse_path.
  183. using Visitor = util::FunctionRef<void(const Obj&, ColKey, Mixed)>;
  184. using PathSizer = util::FunctionRef<void(size_t)>;
  185. void traverse_path(Visitor v, PathSizer ps, size_t path_index = 0) const;
  186. template <typename U>
  187. Obj& set(ColKey col_key, U value, bool is_default = false);
  188. // Create a new object and link it. If an embedded object
  189. // is already set, it will be removed. If a non-embedded
  190. // object is already set, we throw LogicError (to prevent
  191. // dangling objects, since they do not delete automatically
  192. // if they are not embedded...)
  193. Obj create_and_set_linked_object(ColKey col_key, bool is_default = false);
  194. // Clear all fields of a linked object returning it to its
  195. // default state. If the object does not exist, create a
  196. // new object and link it. (To Be Implemented)
  197. Obj clear_linked_object(ColKey col_key);
  198. Obj& set_any(ColKey col_key, Mixed value, bool is_default = false);
  199. Obj& set_any(StringData col_name, Mixed value, bool is_default = false)
  200. {
  201. return set_any(get_column_key(col_name), value, is_default);
  202. }
  203. template <typename U>
  204. Obj& set(StringData col_name, U value, bool is_default = false)
  205. {
  206. return set(get_column_key(col_name), value, is_default);
  207. }
  208. Obj& set_null(ColKey col_key, bool is_default = false);
  209. Obj& set_null(StringData col_name, bool is_default = false)
  210. {
  211. return set_null(get_column_key(col_name), is_default);
  212. }
  213. Obj& add_int(ColKey col_key, int64_t value);
  214. Obj& add_int(StringData col_name, int64_t value)
  215. {
  216. return add_int(get_column_key(col_name), value);
  217. }
  218. template <typename U>
  219. Obj& set_list_values(ColKey col_key, const std::vector<U>& values);
  220. template <typename U>
  221. std::vector<U> get_list_values(ColKey col_key) const;
  222. template <class Head, class... Tail>
  223. Obj& set_all(Head v, Tail... tail);
  224. // The main algorithm for handling schema migrations if we try to convert
  225. // from TopLevel* to Embedded, in this case all the orphan objects are deleted
  226. // and all the objects with multiple backlinks are cloned in order to avoid to
  227. // get schema violations during the migration.
  228. // By default this alogirithm is disabled. RealmConfig contains a boolean flag
  229. // to enable it.
  230. void handle_multiple_backlinks_during_schema_migration();
  231. Obj get_linked_object(ColKey link_col_key) const
  232. {
  233. return _get_linked_object(link_col_key, get_any(link_col_key));
  234. }
  235. Obj get_linked_object(StringData link_col_name) const
  236. {
  237. return get_linked_object(get_column_key(link_col_name));
  238. }
  239. Obj get_parent_object() const;
  240. template <typename U>
  241. Lst<U> get_list(ColKey col_key) const;
  242. template <typename U>
  243. LstPtr<U> get_list_ptr(ColKey col_key) const;
  244. template <typename U>
  245. Lst<U> get_list(StringData col_name) const
  246. {
  247. return get_list<U>(get_column_key(col_name));
  248. }
  249. LnkLst get_linklist(ColKey col_key) const;
  250. LnkLstPtr get_linklist_ptr(ColKey col_key) const;
  251. LnkLst get_linklist(StringData col_name) const;
  252. /// Get a type-erased list instance for the given list column.
  253. ///
  254. /// Note: For lists of links, this always returns a `LnkLst`, rather than a
  255. /// `Lst<ObjKey>`. Use `get_list_ptr<ObjKey>(col_key)` to get a list of
  256. /// links with uncondensed indices.
  257. LstBasePtr get_listbase_ptr(ColKey col_key) const;
  258. template <typename U>
  259. Set<U> get_set(StringData col_name) const
  260. {
  261. return get_set<U>(get_column_key(col_name));
  262. }
  263. template <typename U>
  264. Set<U> get_set(ColKey col_key) const;
  265. template <typename U>
  266. SetPtr<U> get_set_ptr(ColKey col_key) const;
  267. LnkSet get_linkset(ColKey col_key) const;
  268. LnkSet get_linkset(StringData col_name) const;
  269. LnkSetPtr get_linkset_ptr(ColKey col_key) const;
  270. SetBasePtr get_setbase_ptr(ColKey col_key) const;
  271. Dictionary get_dictionary(ColKey col_key) const;
  272. DictionaryPtr get_dictionary_ptr(ColKey col_key) const;
  273. Dictionary get_dictionary(StringData col_name) const;
  274. CollectionBasePtr get_collection_ptr(ColKey col_key) const;
  275. CollectionBasePtr get_collection_ptr(StringData col_name) const;
  276. LinkCollectionPtr get_linkcollection_ptr(ColKey col_key) const;
  277. void assign_pk_and_backlinks(const Obj& other);
  278. class Internal {
  279. friend class _impl::DeepChangeChecker;
  280. static ref_type get_ref(const Obj& obj, ColKey col_key);
  281. };
  282. private:
  283. friend class ArrayBacklink;
  284. friend class CascadeState;
  285. friend class Cluster;
  286. friend class ColumnListBase;
  287. friend class CollectionBase;
  288. friend class TableView;
  289. template <class, class>
  290. friend class Collection;
  291. template <class, class>
  292. friend class CollectionBaseImpl;
  293. template <class>
  294. friend class Lst;
  295. friend class LnkLst;
  296. friend class Dictionary;
  297. friend class LinkMap;
  298. template <class>
  299. friend class Set;
  300. friend class Table;
  301. friend class Transaction;
  302. mutable TableRef m_table;
  303. ObjKey m_key;
  304. mutable MemRef m_mem;
  305. mutable size_t m_row_ndx;
  306. mutable uint64_t m_storage_version;
  307. mutable bool m_valid;
  308. Allocator& _get_alloc() const noexcept;
  309. /// Update the accessor. Returns true when the accessor was updated to
  310. /// reflect new changes to the underlying state.
  311. bool update() const;
  312. // update if needed - with and without check of table instance version:
  313. bool update_if_needed() const;
  314. bool _update_if_needed() const; // no check, use only when already checked
  315. /// Update the accessor (and return `UpdateStatus::Detached` if the Obj is
  316. /// no longer valid, rather than throwing an exception).
  317. UpdateStatus update_if_needed_with_status() const;
  318. template <class T>
  319. bool do_is_null(ColKey::Idx col_ndx) const;
  320. const ClusterTree* get_tree_top() const;
  321. ColKey get_column_key(StringData col_name) const;
  322. ColKey get_primary_key_column() const;
  323. TableKey get_table_key() const;
  324. TableRef get_target_table(ObjLink link) const;
  325. const Spec& get_spec() const;
  326. template <typename U>
  327. U _get(ColKey::Idx col_ndx) const;
  328. ObjKey get_backlink(ColKey backlink_col, size_t backlink_ndx) const;
  329. // Return all backlinks from a specific backlink column
  330. std::vector<ObjKey> get_all_backlinks(ColKey backlink_col) const;
  331. // Return number of backlinks from a specific backlink column
  332. size_t get_backlink_cnt(ColKey backlink_col) const;
  333. ObjKey get_unfiltered_link(ColKey col_key) const;
  334. template <class Val>
  335. Obj& _set(size_t col_ndx, Val v);
  336. template <class Head, class... Tail>
  337. Obj& _set(size_t col_ndx, Head v, Tail... tail);
  338. ColKey spec_ndx2colkey(size_t col_ndx);
  339. size_t colkey2spec_ndx(ColKey);
  340. bool ensure_writeable();
  341. void sync(Node& arr);
  342. int_fast64_t bump_content_version();
  343. void bump_both_versions();
  344. template <class T>
  345. void do_set_null(ColKey col_key);
  346. // Dictionary support
  347. size_t get_row_ndx() const
  348. {
  349. return m_row_ndx;
  350. }
  351. Obj _get_linked_object(ColKey link_col_key, Mixed link) const;
  352. Obj _get_linked_object(StringData link_col_name, Mixed link) const
  353. {
  354. return _get_linked_object(get_column_key(link_col_name), link);
  355. }
  356. void set_int(ColKey col_key, int64_t value);
  357. void add_backlink(ColKey backlink_col, ObjKey origin_key);
  358. bool remove_one_backlink(ColKey backlink_col, ObjKey origin_key);
  359. void nullify_link(ColKey origin_col, ObjLink target_key) &&;
  360. // Used when inserting a new link. You will not remove existing links in this process
  361. void set_backlink(ColKey col_key, ObjLink new_link) const;
  362. // Used when replacing a link, return true if CascadeState contains objects to remove
  363. bool replace_backlink(ColKey col_key, ObjLink old_link, ObjLink new_link, CascadeState& state) const;
  364. // Used when removing a backlink, return true if CascadeState contains objects to remove
  365. bool remove_backlink(ColKey col_key, ObjLink old_link, CascadeState& state) const;
  366. template <class T>
  367. inline void set_spec(T&, ColKey);
  368. template <class ValueType>
  369. inline void nullify_single_link(ColKey col, ValueType target);
  370. void fix_linking_object_during_schema_migration(Obj linking_obj, Obj obj, ColKey opposite_col_key) const;
  371. };
  372. std::ostream& operator<<(std::ostream&, const Obj& obj);
  373. template <>
  374. int64_t Obj::get(ColKey) const;
  375. template <>
  376. bool Obj::get(ColKey) const;
  377. template <>
  378. int64_t Obj::_get(ColKey::Idx col_ndx) const;
  379. template <>
  380. StringData Obj::_get(ColKey::Idx col_ndx) const;
  381. template <>
  382. BinaryData Obj::_get(ColKey::Idx col_ndx) const;
  383. template <>
  384. ObjKey Obj::_get(ColKey::Idx col_ndx) const;
  385. struct Obj::FatPathElement {
  386. Obj obj; // Object which embeds...
  387. ColKey col_key; // Column holding link or link list which embeds...
  388. Mixed index; // index into link list or dictionary (or null)
  389. };
  390. struct Obj::PathElement {
  391. ColKey col_key; // Column holding link or link list which embeds...
  392. Mixed index; // index into link list or dictionary (or null)
  393. };
  394. template <>
  395. Obj& Obj::set(ColKey, int64_t value, bool is_default);
  396. template <>
  397. Obj& Obj::set(ColKey, ObjKey value, bool is_default);
  398. template <>
  399. Obj& Obj::set(ColKey, ObjLink value, bool is_default);
  400. template <>
  401. inline Obj& Obj::set(ColKey col_key, int value, bool is_default)
  402. {
  403. return set(col_key, int_fast64_t(value), is_default);
  404. }
  405. template <>
  406. inline Obj& Obj::set(ColKey col_key, uint_fast64_t value, bool is_default)
  407. {
  408. int_fast64_t value_2 = 0;
  409. if (REALM_UNLIKELY(util::int_cast_with_overflow_detect(value, value_2))) {
  410. REALM_TERMINATE("Unsigned integer too big.");
  411. }
  412. return set(col_key, value_2, is_default);
  413. }
  414. template <>
  415. inline Obj& Obj::set(ColKey col_key, const char* str, bool is_default)
  416. {
  417. return set(col_key, StringData(str), is_default);
  418. }
  419. template <>
  420. inline Obj& Obj::set(ColKey col_key, char* str, bool is_default)
  421. {
  422. return set(col_key, StringData(str), is_default);
  423. }
  424. template <>
  425. inline Obj& Obj::set(ColKey col_key, std::string str, bool is_default)
  426. {
  427. return set(col_key, StringData(str), is_default);
  428. }
  429. template <>
  430. inline Obj& Obj::set(ColKey col_key, realm::null, bool is_default)
  431. {
  432. return set_null(col_key, is_default);
  433. }
  434. template <>
  435. inline Obj& Obj::set(ColKey col_key, util::Optional<bool> value, bool is_default)
  436. {
  437. return value ? set(col_key, *value, is_default) : set_null(col_key, is_default);
  438. }
  439. template <>
  440. inline Obj& Obj::set(ColKey col_key, util::Optional<int64_t> value, bool is_default)
  441. {
  442. return value ? set(col_key, *value, is_default) : set_null(col_key, is_default);
  443. }
  444. template <>
  445. inline Obj& Obj::set(ColKey col_key, util::Optional<float> value, bool is_default)
  446. {
  447. return value ? set(col_key, *value, is_default) : set_null(col_key, is_default);
  448. }
  449. template <>
  450. inline Obj& Obj::set(ColKey col_key, util::Optional<double> value, bool is_default)
  451. {
  452. return value ? set(col_key, *value, is_default) : set_null(col_key, is_default);
  453. }
  454. template <>
  455. inline Obj& Obj::set(ColKey col_key, util::Optional<ObjectId> value, bool is_default)
  456. {
  457. return value ? set(col_key, *value, is_default) : set_null(col_key, is_default);
  458. }
  459. template <>
  460. inline Obj& Obj::set(ColKey col_key, util::Optional<UUID> value, bool is_default)
  461. {
  462. return value ? set(col_key, *value, is_default) : set_null(col_key, is_default);
  463. }
  464. template <typename U>
  465. Obj& Obj::set_list_values(ColKey col_key, const std::vector<U>& values)
  466. {
  467. size_t sz = values.size();
  468. auto list = get_list<U>(col_key);
  469. size_t list_sz = list.size();
  470. if (sz < list_sz) {
  471. list.resize(sz);
  472. list_sz = sz;
  473. }
  474. size_t i = 0;
  475. while (i < list_sz) {
  476. list.set(i, values[i]);
  477. i++;
  478. }
  479. while (i < sz) {
  480. list.add(values[i]);
  481. i++;
  482. }
  483. return *this;
  484. }
  485. template <typename U>
  486. std::vector<U> Obj::get_list_values(ColKey col_key) const
  487. {
  488. std::vector<U> values;
  489. auto list = get_list<U>(col_key);
  490. for (auto v : list)
  491. values.push_back(v);
  492. return values;
  493. }
  494. template <class Val>
  495. inline Obj& Obj::_set(size_t col_ndx, Val v)
  496. {
  497. return set(spec_ndx2colkey(col_ndx), v);
  498. }
  499. template <class Head, class... Tail>
  500. inline Obj& Obj::_set(size_t col_ndx, Head v, Tail... tail)
  501. {
  502. set(spec_ndx2colkey(col_ndx), v);
  503. return _set(col_ndx + 1, tail...);
  504. }
  505. template <class Head, class... Tail>
  506. inline Obj& Obj::set_all(Head v, Tail... tail)
  507. {
  508. size_t start_index = 0;
  509. // Avoid trying to set the PK column.
  510. if (get_primary_key_column()) {
  511. REALM_ASSERT(colkey2spec_ndx(get_primary_key_column()) == 0);
  512. start_index = 1;
  513. }
  514. return _set(start_index, v, tail...);
  515. }
  516. inline bool Obj::update_if_needed() const
  517. {
  518. auto current_version = get_alloc().get_storage_version();
  519. if (current_version != m_storage_version) {
  520. return update();
  521. }
  522. return false;
  523. }
  524. inline int_fast64_t Obj::bump_content_version()
  525. {
  526. Allocator& alloc = get_alloc();
  527. return alloc.bump_content_version();
  528. }
  529. inline void Obj::bump_both_versions()
  530. {
  531. Allocator& alloc = get_alloc();
  532. alloc.bump_content_version();
  533. alloc.bump_storage_version();
  534. }
  535. } // namespace realm
  536. #endif // REALM_OBJ_HPP