obj.hpp 18 KB

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