table.hpp 49 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394
  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_TABLE_HPP
  19. #define REALM_TABLE_HPP
  20. #include <algorithm>
  21. #include <map>
  22. #include <utility>
  23. #include <typeinfo>
  24. #include <memory>
  25. #include <mutex>
  26. #include <realm/util/features.h>
  27. #include <realm/util/function_ref.hpp>
  28. #include <realm/util/thread.hpp>
  29. #include <realm/table_ref.hpp>
  30. #include <realm/spec.hpp>
  31. #include <realm/query.hpp>
  32. #include <realm/table_cluster_tree.hpp>
  33. #include <realm/keys.hpp>
  34. #include <realm/global_key.hpp>
  35. // Only set this to one when testing the code paths that exercise object ID
  36. // hash collisions. It artificially limits the "optimistic" local ID to use
  37. // only the lower 15 bits of the ID rather than the lower 63 bits, making it
  38. // feasible to generate collisions within reasonable time.
  39. #define REALM_EXERCISE_OBJECT_ID_COLLISION 0
  40. namespace realm {
  41. class BacklinkColumn;
  42. template <class>
  43. class BacklinkCount;
  44. class BinaryColumy;
  45. class ConstTableView;
  46. class Group;
  47. class SortDescriptor;
  48. class StringIndex;
  49. class TableView;
  50. template <class>
  51. class Columns;
  52. template <class>
  53. class SubQuery;
  54. class ColKeys;
  55. struct GlobalKey;
  56. class LinkChain;
  57. class Subexpr;
  58. struct Link {
  59. };
  60. typedef Link BackLink;
  61. namespace _impl {
  62. class TableFriend;
  63. }
  64. namespace metrics {
  65. class QueryInfo;
  66. }
  67. namespace query_parser {
  68. class Arguments;
  69. class KeyPathMapping;
  70. class ParserDriver;
  71. } // namespace query_parser
  72. enum class ExpressionComparisonType : unsigned char {
  73. Any,
  74. All,
  75. None,
  76. };
  77. class Table {
  78. public:
  79. /// Construct a new freestanding top-level table with static
  80. /// lifetime. For debugging only.
  81. Table(Allocator& = Allocator::get_default());
  82. /// Construct a copy of the specified table as a new freestanding
  83. /// top-level table with static lifetime. For debugging only.
  84. Table(const Table&, Allocator& = Allocator::get_default());
  85. ~Table() noexcept;
  86. Allocator& get_alloc() const;
  87. /// Get the name of this table, if it has one. Only group-level tables have
  88. /// names. For a table of any other kind, this function returns the empty
  89. /// string.
  90. StringData get_name() const noexcept;
  91. const char* get_state() const noexcept;
  92. /// If this table is a group-level table, the parent group is returned,
  93. /// otherwise null is returned.
  94. Group* get_parent_group() const noexcept;
  95. // Whether or not elements can be null.
  96. bool is_nullable(ColKey col_key) const;
  97. // Whether or not the column is a list.
  98. bool is_list(ColKey col_key) const;
  99. //@{
  100. /// Conventience functions for inspecting the dynamic table type.
  101. ///
  102. bool is_embedded() const noexcept; // true if table holds embedded objects
  103. size_t get_column_count() const noexcept;
  104. DataType get_column_type(ColKey column_key) const;
  105. StringData get_column_name(ColKey column_key) const;
  106. ColumnAttrMask get_column_attr(ColKey column_key) const noexcept;
  107. DataType get_dictionary_key_type(ColKey column_key) const noexcept;
  108. ColKey get_column_key(StringData name) const noexcept;
  109. ColKeys get_column_keys() const;
  110. typedef util::Optional<std::pair<ConstTableRef, ColKey>> BacklinkOrigin;
  111. BacklinkOrigin find_backlink_origin(StringData origin_table_name, StringData origin_col_name) const noexcept;
  112. BacklinkOrigin find_backlink_origin(ColKey backlink_col) const noexcept;
  113. std::vector<std::pair<TableKey, ColKey>> get_incoming_link_columns() const noexcept;
  114. //@}
  115. // Primary key columns
  116. ColKey get_primary_key_column() const;
  117. void set_primary_key_column(ColKey col);
  118. void validate_primary_column();
  119. //@{
  120. /// Convenience functions for manipulating the dynamic table type.
  121. ///
  122. static const size_t max_column_name_length = 63;
  123. static const uint64_t max_num_columns = 0xFFFFUL; // <-- must be power of two -1
  124. ColKey add_column(DataType type, StringData name, bool nullable = false);
  125. ColKey add_column(Table& target, StringData name);
  126. ColKey add_column_list(DataType type, StringData name, bool nullable = false);
  127. ColKey add_column_list(Table& target, StringData name);
  128. ColKey add_column_set(DataType type, StringData name, bool nullable = false);
  129. ColKey add_column_set(Table& target, StringData name);
  130. ColKey add_column_dictionary(DataType type, StringData name, bool nullable = false,
  131. DataType key_type = type_String);
  132. ColKey add_column_dictionary(Table& target, StringData name, DataType key_type = type_String);
  133. [[deprecated("Use add_column(Table&) or add_column_list(Table&) instead.")]] //
  134. ColKey
  135. add_column_link(DataType type, StringData name, Table& target);
  136. void remove_column(ColKey col_key);
  137. void rename_column(ColKey col_key, StringData new_name);
  138. bool valid_column(ColKey col_key) const noexcept;
  139. void check_column(ColKey col_key) const;
  140. // Change the embedded property of a table. If switching to being embedded, the table must
  141. // not have a primary key and all objects must have exactly 1 backlink.
  142. void set_embedded(bool embedded);
  143. //@}
  144. /// True for `col_type_Link` and `col_type_LinkList`.
  145. static bool is_link_type(ColumnType) noexcept;
  146. //@{
  147. /// has_search_index() returns true if, and only if a search index has been
  148. /// added to the specified column. Rather than throwing, it returns false if
  149. /// the table accessor is detached or the specified index is out of range.
  150. ///
  151. /// add_search_index() adds a search index to the specified column of the
  152. /// table. It has no effect if a search index has already been added to the
  153. /// specified column (idempotency).
  154. ///
  155. /// remove_search_index() removes the search index from the specified column
  156. /// of the table. It has no effect if the specified column has no search
  157. /// index. The search index cannot be removed from the primary key of a
  158. /// table.
  159. ///
  160. /// \param col_key The key of a column of the table.
  161. bool has_search_index(ColKey col_key) const noexcept;
  162. void add_search_index(ColKey col_key);
  163. void remove_search_index(ColKey col_key);
  164. void enumerate_string_column(ColKey col_key);
  165. bool is_enumerated(ColKey col_key) const noexcept;
  166. bool contains_unique_values(ColKey col_key) const;
  167. //@}
  168. /// If the specified column is optimized to store only unique values, then
  169. /// this function returns the number of unique values currently
  170. /// stored. Otherwise it returns zero. This function is mainly intended for
  171. /// debugging purposes.
  172. size_t get_num_unique_values(ColKey col_key) const;
  173. template <class T>
  174. Columns<T> column(ColKey col_key, ExpressionComparisonType = ExpressionComparisonType::Any) const;
  175. template <class T>
  176. Columns<T> column(const Table& origin, ColKey origin_col_key) const;
  177. // BacklinkCount is a total count per row and therefore not attached to a specific column
  178. template <class T>
  179. BacklinkCount<T> get_backlink_count() const;
  180. template <class T>
  181. SubQuery<T> column(ColKey col_key, Query subquery) const;
  182. template <class T>
  183. SubQuery<T> column(const Table& origin, ColKey origin_col_key, Query subquery) const;
  184. // Table size and deletion
  185. bool is_empty() const noexcept;
  186. size_t size() const noexcept
  187. {
  188. return m_clusters.size();
  189. }
  190. size_t nb_unresolved() const noexcept
  191. {
  192. return m_tombstones ? m_tombstones->size() : 0;
  193. }
  194. //@{
  195. /// Object handling.
  196. // Create an object with key. If the key is omitted, a key will be generated by the system
  197. Obj create_object(ObjKey key = {}, const FieldValues& = {});
  198. // Create an object with specific GlobalKey - or return already existing object
  199. // Potential tombstone will be resurrected
  200. Obj create_object(GlobalKey object_id, const FieldValues& = {});
  201. // Create an object with primary key. If an object with the given primary key already exists, it
  202. // will be returned and did_create (if supplied) will be set to false.
  203. // Potential tombstone will be resurrected
  204. Obj create_object_with_primary_key(const Mixed& primary_key, FieldValues&&, bool* did_create = nullptr);
  205. Obj create_object_with_primary_key(const Mixed& primary_key, bool* did_create = nullptr)
  206. {
  207. return create_object_with_primary_key(primary_key, {{}}, did_create);
  208. }
  209. // Return key for existing object or return null key.
  210. ObjKey find_primary_key(Mixed value) const;
  211. // Return ObjKey for object identified by id. If objects does not exist, return null key
  212. // Important: This function must not be called for tables with primary keys.
  213. ObjKey get_objkey(GlobalKey id) const;
  214. // Return key for existing object or return unresolved key.
  215. // Important: This is to be used ONLY by the Sync client. SDKs should NEVER
  216. // observe an unresolved key. Ever.
  217. ObjKey get_objkey_from_primary_key(const Mixed& primary_key);
  218. // Return key for existing object or return unresolved key.
  219. // Important: This is to be used ONLY by the Sync client. SDKs should NEVER
  220. // observe an unresolved key. Ever.
  221. // Important (2): This function must not be called for tables with primary keys.
  222. ObjKey get_objkey_from_global_key(GlobalKey key);
  223. /// Create a number of objects and add corresponding keys to a vector
  224. void create_objects(size_t number, std::vector<ObjKey>& keys);
  225. /// Create a number of objects with keys supplied
  226. void create_objects(const std::vector<ObjKey>& keys);
  227. /// Does the key refer to an object within the table?
  228. bool is_valid(ObjKey key) const
  229. {
  230. return m_clusters.is_valid(key);
  231. }
  232. GlobalKey get_object_id(ObjKey key) const;
  233. Obj get_object(ObjKey key) const
  234. {
  235. REALM_ASSERT(!key.is_unresolved());
  236. return m_clusters.get(key);
  237. }
  238. Obj get_object(size_t ndx) const
  239. {
  240. return m_clusters.get(ndx);
  241. }
  242. // Get object based on primary key
  243. Obj get_object_with_primary_key(Mixed pk) const;
  244. // Get primary key based on ObjKey
  245. Mixed get_primary_key(ObjKey key);
  246. // Get logical index for object. This function is not very efficient
  247. size_t get_object_ndx(ObjKey key) const noexcept
  248. {
  249. return m_clusters.get_ndx(key);
  250. }
  251. void dump_objects();
  252. bool traverse_clusters(ClusterTree::TraverseFunction func) const
  253. {
  254. return m_clusters.traverse(func);
  255. }
  256. /// remove_object() removes the specified object from the table.
  257. /// Any links from the specified object into objects residing in an embedded
  258. /// table will cause those objects to be deleted as well, and so on recursively.
  259. void remove_object(ObjKey key);
  260. /// remove_object_recursive() will delete linked rows if the removed link was the
  261. /// last one holding on to the row in question. This will be done recursively.
  262. void remove_object_recursive(ObjKey key);
  263. // Invalidate object. To be used by the Sync client.
  264. // - turns the object into a tombstone if links exist
  265. // - otherwise works just as remove_object()
  266. ObjKey invalidate_object(ObjKey key);
  267. Obj get_tombstone(ObjKey key) const
  268. {
  269. REALM_ASSERT(key.is_unresolved());
  270. REALM_ASSERT(m_tombstones);
  271. return m_tombstones->get(key);
  272. }
  273. void clear();
  274. using Iterator = TableClusterTree::Iterator;
  275. Iterator begin() const;
  276. Iterator end() const;
  277. void remove_object(const Iterator& it)
  278. {
  279. remove_object(it->get_key());
  280. }
  281. //@}
  282. TableRef get_link_target(ColKey column_key) noexcept;
  283. ConstTableRef get_link_target(ColKey column_key) const noexcept;
  284. static const size_t max_string_size = 0xFFFFF8 - Array::header_size - 1;
  285. static const size_t max_binary_size = 0xFFFFF8 - Array::header_size;
  286. static constexpr int_fast64_t max_integer = std::numeric_limits<int64_t>::max();
  287. static constexpr int_fast64_t min_integer = std::numeric_limits<int64_t>::min();
  288. /// Only group-level unordered tables can be used as origins or targets of
  289. /// links.
  290. bool is_group_level() const noexcept;
  291. /// A Table accessor obtained from a frozen transaction is also frozen.
  292. bool is_frozen() const noexcept
  293. {
  294. return m_is_frozen;
  295. }
  296. /// If this table is a group-level table, then this function returns the
  297. /// index of this table within the group. Otherwise it returns realm::npos.
  298. size_t get_index_in_group() const noexcept;
  299. TableKey get_key() const noexcept;
  300. uint64_t allocate_sequence_number();
  301. // Used by upgrade
  302. void set_sequence_number(uint64_t seq);
  303. void set_collision_map(ref_type ref);
  304. // Get the key of this table directly, without needing a Table accessor.
  305. static TableKey get_key_direct(Allocator& alloc, ref_type top_ref);
  306. // Aggregate functions
  307. size_t count_int(ColKey col_key, int64_t value) const;
  308. size_t count_string(ColKey col_key, StringData value) const;
  309. size_t count_float(ColKey col_key, float value) const;
  310. size_t count_double(ColKey col_key, double value) const;
  311. size_t count_decimal(ColKey col_key, Decimal128 value) const;
  312. int64_t sum_int(ColKey col_key) const;
  313. double sum_float(ColKey col_key) const;
  314. double sum_double(ColKey col_key) const;
  315. Decimal128 sum_decimal(ColKey col_key) const;
  316. Decimal128 sum_mixed(ColKey col_key) const;
  317. int64_t maximum_int(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  318. float maximum_float(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  319. double maximum_double(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  320. Decimal128 maximum_decimal(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  321. Mixed maximum_mixed(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  322. Timestamp maximum_timestamp(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  323. int64_t minimum_int(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  324. float minimum_float(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  325. double minimum_double(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  326. Decimal128 minimum_decimal(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  327. Mixed minimum_mixed(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  328. Timestamp minimum_timestamp(ColKey col_key, ObjKey* return_ndx = nullptr) const;
  329. double average_int(ColKey col_key, size_t* value_count = nullptr) const;
  330. double average_float(ColKey col_key, size_t* value_count = nullptr) const;
  331. double average_double(ColKey col_key, size_t* value_count = nullptr) const;
  332. Decimal128 average_decimal(ColKey col_key, size_t* value_count = nullptr) const;
  333. Decimal128 average_mixed(ColKey col_key, size_t* value_count = nullptr) const;
  334. // Will return pointer to search index accessor. Will return nullptr if no index
  335. StringIndex* get_search_index(ColKey col) const noexcept
  336. {
  337. report_invalid_key(col);
  338. if (!has_search_index(col))
  339. return nullptr;
  340. return m_index_accessors[col.get_index().val];
  341. }
  342. template <class T>
  343. ObjKey find_first(ColKey col_key, T value) const;
  344. ObjKey find_first_int(ColKey col_key, int64_t value) const;
  345. ObjKey find_first_bool(ColKey col_key, bool value) const;
  346. ObjKey find_first_timestamp(ColKey col_key, Timestamp value) const;
  347. ObjKey find_first_object_id(ColKey col_key, ObjectId value) const;
  348. ObjKey find_first_float(ColKey col_key, float value) const;
  349. ObjKey find_first_double(ColKey col_key, double value) const;
  350. ObjKey find_first_decimal(ColKey col_key, Decimal128 value) const;
  351. ObjKey find_first_string(ColKey col_key, StringData value) const;
  352. ObjKey find_first_binary(ColKey col_key, BinaryData value) const;
  353. ObjKey find_first_null(ColKey col_key) const;
  354. ObjKey find_first_uuid(ColKey col_key, UUID value) const;
  355. // TableView find_all_link(Key target_key);
  356. // ConstTableView find_all_link(Key target_key) const;
  357. TableView find_all_int(ColKey col_key, int64_t value);
  358. ConstTableView find_all_int(ColKey col_key, int64_t value) const;
  359. TableView find_all_bool(ColKey col_key, bool value);
  360. ConstTableView find_all_bool(ColKey col_key, bool value) const;
  361. TableView find_all_float(ColKey col_key, float value);
  362. ConstTableView find_all_float(ColKey col_key, float value) const;
  363. TableView find_all_double(ColKey col_key, double value);
  364. ConstTableView find_all_double(ColKey col_key, double value) const;
  365. TableView find_all_string(ColKey col_key, StringData value);
  366. ConstTableView find_all_string(ColKey col_key, StringData value) const;
  367. TableView find_all_binary(ColKey col_key, BinaryData value);
  368. ConstTableView find_all_binary(ColKey col_key, BinaryData value) const;
  369. TableView find_all_null(ColKey col_key);
  370. ConstTableView find_all_null(ColKey col_key) const;
  371. TableView get_sorted_view(ColKey col_key, bool ascending = true);
  372. ConstTableView get_sorted_view(ColKey col_key, bool ascending = true) const;
  373. TableView get_sorted_view(SortDescriptor order);
  374. ConstTableView get_sorted_view(SortDescriptor order) const;
  375. // Report the current content version. This is a 64-bit value which is bumped whenever
  376. // the content in the table changes.
  377. uint_fast64_t get_content_version() const noexcept;
  378. // Report the current instance version. This is a 64-bit value which is bumped
  379. // whenever the table accessor is recycled.
  380. uint_fast64_t get_instance_version() const noexcept;
  381. // Report the current storage version. This is a 64-bit value which is bumped
  382. // whenever the location in memory of any part of the table changes.
  383. uint_fast64_t get_storage_version(uint64_t instance_version) const;
  384. uint_fast64_t get_storage_version() const;
  385. void bump_storage_version() const noexcept;
  386. void bump_content_version() const noexcept;
  387. // Change the nullability of the column identified by col_key.
  388. // This might result in the creation of a new column and deletion of the old.
  389. // The column key to use going forward is returned.
  390. // If the conversion is from nullable to non-nullable, throw_on_null determines
  391. // the reaction to encountering a null value: If clear, null values will be
  392. // converted to default values. If set, a 'column_not_nullable' is thrown and the
  393. // table is unchanged.
  394. ColKey set_nullability(ColKey col_key, bool nullable, bool throw_on_null);
  395. // Iterate through (subset of) columns. The supplied function may abort iteration
  396. // by returning 'true' (early out).
  397. template <typename Func>
  398. bool for_each_and_every_column(Func func) const
  399. {
  400. for (auto col_key : m_leaf_ndx2colkey) {
  401. if (!col_key)
  402. continue;
  403. if (func(col_key))
  404. return true;
  405. }
  406. return false;
  407. }
  408. template <typename Func>
  409. bool for_each_public_column(Func func) const
  410. {
  411. for (auto col_key : m_leaf_ndx2colkey) {
  412. if (!col_key)
  413. continue;
  414. if (col_key.get_type() == col_type_BackLink)
  415. continue;
  416. if (func(col_key))
  417. return true;
  418. }
  419. return false;
  420. }
  421. template <typename Func>
  422. bool for_each_backlink_column(Func func) const
  423. {
  424. // Could be optimized - to not iterate through all non-backlink columns:
  425. for (auto col_key : m_leaf_ndx2colkey) {
  426. if (!col_key)
  427. continue;
  428. if (col_key.get_type() != col_type_BackLink)
  429. continue;
  430. if (func(col_key))
  431. return true;
  432. }
  433. return false;
  434. }
  435. private:
  436. template <class T>
  437. TableView find_all(ColKey col_key, T value);
  438. void build_column_mapping();
  439. ColKey generate_col_key(ColumnType ct, ColumnAttrMask attrs);
  440. void convert_column(ColKey from, ColKey to, bool throw_on_null);
  441. template <class F, class T>
  442. void change_nullability(ColKey from, ColKey to, bool throw_on_null);
  443. template <class F, class T>
  444. void change_nullability_list(ColKey from, ColKey to, bool throw_on_null);
  445. Obj create_linked_object(GlobalKey = {});
  446. /// Changes embeddedness unconditionally. Called only from Group::do_get_or_add_table()
  447. void do_set_embedded(bool embedded);
  448. public:
  449. // mapping between index used in leaf nodes (leaf_ndx) and index used in spec (spec_ndx)
  450. // as well as the full column key. A leaf_ndx can be obtained directly from the column key
  451. size_t colkey2spec_ndx(ColKey key) const;
  452. size_t leaf_ndx2spec_ndx(ColKey::Idx idx) const;
  453. ColKey::Idx spec_ndx2leaf_ndx(size_t idx) const;
  454. ColKey leaf_ndx2colkey(ColKey::Idx idx) const;
  455. ColKey spec_ndx2colkey(size_t ndx) const;
  456. void report_invalid_key(ColKey col_key) const;
  457. // Queries
  458. // Using where(tv) is the new method to perform queries on TableView. The 'tv' can have any order; it does not
  459. // need to be sorted, and, resulting view retains its order.
  460. Query where(ConstTableView* tv = nullptr)
  461. {
  462. return Query(m_own_ref, tv);
  463. }
  464. Query where(ConstTableView* tv = nullptr) const
  465. {
  466. return Query(m_own_ref, tv);
  467. }
  468. // Perform queries on a Link Collection. The returned Query holds a reference to collection.
  469. Query where(const ObjList& list) const
  470. {
  471. return Query(m_own_ref, list);
  472. }
  473. Query where(const DictionaryLinkValues& dictionary_of_links) const;
  474. Query query(const std::string& query_string, const std::vector<Mixed>& arguments = {}) const;
  475. Query query(const std::string& query_string, const std::vector<Mixed>& arguments,
  476. const query_parser::KeyPathMapping& mapping) const;
  477. Query query(const std::string& query_string, query_parser::Arguments& arguments,
  478. const query_parser::KeyPathMapping&) const;
  479. //@{
  480. /// WARNING: The link() and backlink() methods will alter a state on the Table object and return a reference
  481. /// to itself. Be aware if assigning the return value of link() to a variable; this might be an error!
  482. /// This is an error:
  483. /// Table& cats = owners->link(1);
  484. /// auto& dogs = owners->link(2);
  485. /// Query q = person_table->where()
  486. /// .and_query(cats.column<String>(5).equal("Fido"))
  487. /// .Or()
  488. /// .and_query(dogs.column<String>(6).equal("Meowth"));
  489. /// Instead, do this:
  490. /// Query q = owners->where()
  491. /// .and_query(person_table->link(1).column<String>(5).equal("Fido"))
  492. /// .Or()
  493. /// .and_query(person_table->link(2).column<String>(6).equal("Meowth"));
  494. /// The two calls to link() in the erroneous example will append the two values 0 and 1 to an internal vector in
  495. /// the owners table, and we end up with three references to that same table: owners, cats and dogs. They are all
  496. /// the same table, its vector has the values {0, 1}, so a query would not make any sense.
  497. LinkChain link(ColKey link_column) const;
  498. LinkChain backlink(const Table& origin, ColKey origin_col_key) const;
  499. // Conversion
  500. void schema_to_json(std::ostream& out, const std::map<std::string, std::string>& renames) const;
  501. void to_json(std::ostream& out, size_t link_depth, const std::map<std::string, std::string>& renames,
  502. JSONOutputMode output_mode = output_mode_json) const;
  503. /// \brief Compare two tables for equality.
  504. ///
  505. /// Two tables are equal if they have equal descriptors
  506. /// (`Descriptor::operator==()`) and equal contents. Equal descriptors imply
  507. /// that the two tables have the same columns in the same order. Equal
  508. /// contents means that the two tables must have the same number of rows,
  509. /// and that for each row index, the two rows must have the same values in
  510. /// each column.
  511. ///
  512. /// In mixed columns, both the value types and the values are required to be
  513. /// equal.
  514. ///
  515. /// For a particular row and column, if the two values are themselves tables
  516. /// (subtable and mixed columns) value equality implies a recursive
  517. /// invocation of `Table::operator==()`.
  518. bool operator==(const Table&) const;
  519. /// \brief Compare two tables for inequality.
  520. ///
  521. /// See operator==().
  522. bool operator!=(const Table& t) const;
  523. /// Compute the sum of the sizes in number of bytes of all the array nodes
  524. /// that currently make up this table. See also
  525. /// Group::compute_aggregate_byte_size().
  526. ///
  527. /// If this table accessor is the detached state, this function returns
  528. /// zero.
  529. size_t compute_aggregated_byte_size() const noexcept;
  530. // Debug
  531. void verify() const;
  532. #ifdef REALM_DEBUG
  533. MemStats stats() const;
  534. #endif
  535. TableRef get_opposite_table(ColKey col_key) const;
  536. TableKey get_opposite_table_key(ColKey col_key) const;
  537. bool links_to_self(ColKey col_key) const;
  538. ColKey get_opposite_column(ColKey col_key) const;
  539. ColKey find_opposite_column(ColKey col_key) const;
  540. protected:
  541. /// Compare the objects of two tables under the assumption that the two tables
  542. /// have the same number of columns, and the same data type at each column
  543. /// index (as expressed through the DataType enum).
  544. bool compare_objects(const Table&) const;
  545. private:
  546. enum LifeCycleCookie {
  547. cookie_created = 0x1234,
  548. cookie_transaction_ended = 0xcafe,
  549. cookie_initialized = 0xbeef,
  550. cookie_removed = 0xbabe,
  551. cookie_void = 0x5678,
  552. cookie_deleted = 0xdead,
  553. };
  554. mutable WrappedAllocator m_alloc;
  555. Array m_top;
  556. void update_allocator_wrapper(bool writable)
  557. {
  558. m_alloc.update_from_underlying_allocator(writable);
  559. }
  560. void refresh_allocator_wrapper() const noexcept
  561. {
  562. m_alloc.refresh_ref_translation();
  563. }
  564. Spec m_spec; // 1st slot in m_top
  565. TableClusterTree m_clusters; // 3rd slot in m_top
  566. std::unique_ptr<TableClusterTree> m_tombstones; // 13th slot in m_top
  567. TableKey m_key; // 4th slot in m_top
  568. Array m_index_refs; // 5th slot in m_top
  569. Array m_opposite_table; // 7th slot in m_top
  570. Array m_opposite_column; // 8th slot in m_top
  571. std::vector<StringIndex*> m_index_accessors;
  572. ColKey m_primary_key_col;
  573. Replication* const* m_repl;
  574. static Replication* g_dummy_replication;
  575. bool m_is_frozen = false;
  576. util::Optional<bool> m_has_any_embedded_objects;
  577. TableRef m_own_ref;
  578. void batch_erase_rows(const KeyColumn& keys);
  579. size_t do_set_link(ColKey col_key, size_t row_ndx, size_t target_row_ndx);
  580. void populate_search_index(ColKey col_key);
  581. void erase_from_search_indexes(ObjKey key);
  582. void update_indexes(ObjKey key, const FieldValues& values);
  583. void clear_indexes();
  584. // Migration support
  585. void migrate_column_info();
  586. bool verify_column_keys();
  587. void migrate_indexes(ColKey pk_col_key);
  588. void migrate_subspec();
  589. void create_columns();
  590. bool migrate_objects(); // Returns true if there are no links to migrate
  591. void migrate_links();
  592. void finalize_migration(ColKey pk_col_key);
  593. /// Disable copying assignment.
  594. ///
  595. /// It could easily be implemented by calling assign(), but the
  596. /// non-checking nature of the low-level dynamically typed API
  597. /// makes it too risky to offer this feature as an
  598. /// operator.
  599. Table& operator=(const Table&) = delete;
  600. /// Create an uninitialized accessor whose lifetime is managed by Group
  601. Table(Replication* const* repl, Allocator&);
  602. void revive(Replication* const* repl, Allocator& new_allocator, bool writable);
  603. void init(ref_type top_ref, ArrayParent*, size_t ndx_in_parent, bool is_writable, bool is_frozen);
  604. void ensure_graveyard();
  605. void set_key(TableKey key);
  606. ColKey do_insert_column(ColKey col_key, DataType type, StringData name, Table* target_table,
  607. DataType key_type = DataType(0));
  608. struct InsertSubtableColumns;
  609. struct EraseSubtableColumns;
  610. struct RenameSubtableColumns;
  611. void erase_root_column(ColKey col_key);
  612. ColKey do_insert_root_column(ColKey col_key, ColumnType, StringData name, DataType key_type = DataType(0));
  613. void do_erase_root_column(ColKey col_key);
  614. void do_add_search_index(ColKey col_key);
  615. bool has_any_embedded_objects();
  616. void set_opposite_column(ColKey col_key, TableKey opposite_table, ColKey opposite_column);
  617. ColKey find_backlink_column(ColKey origin_col_key, TableKey origin_table) const;
  618. ColKey find_or_add_backlink_column(ColKey origin_col_key, TableKey origin_table);
  619. void do_set_primary_key_column(ColKey col_key);
  620. void validate_column_is_unique(ColKey col_key) const;
  621. ObjKey get_next_valid_key();
  622. /// Some Object IDs are generated as a tuple of the client_file_ident and a
  623. /// local sequence number. This function takes the next number in the
  624. /// sequence for the given table and returns an appropriate globally unique
  625. /// GlobalKey.
  626. GlobalKey allocate_object_id_squeezed();
  627. /// Find the local 64-bit object ID for the provided global 128-bit ID.
  628. ObjKey global_to_local_object_id_hashed(GlobalKey global_id) const;
  629. /// After a local ObjKey collision has been detected, this function may be
  630. /// called to obtain a non-colliding local ObjKey in such a way that subsequent
  631. /// calls to global_to_local_object_id() will return the correct local ObjKey
  632. /// for both \a incoming_id and \a colliding_id.
  633. ObjKey allocate_local_id_after_hash_collision(GlobalKey incoming_id, GlobalKey colliding_id,
  634. ObjKey colliding_local_id);
  635. /// Create a placeholder for a not yet existing object and return key to it
  636. Obj get_or_create_tombstone(ObjKey key, const FieldValues& values);
  637. /// Should be called when an object is deleted
  638. void free_local_id_after_hash_collision(ObjKey key);
  639. /// Should be called when last entry is removed - or when table is cleared
  640. void free_collision_table();
  641. /// Called in the context of Group::commit() to ensure that
  642. /// attached table accessors stay valid across a commit. Please
  643. /// note that this works only for non-transactional commits. Table
  644. /// accessors obtained during a transaction are always detached
  645. /// when the transaction ends.
  646. void update_from_parent() noexcept;
  647. // Detach accessor. This recycles the Table accessor and all subordinate
  648. // accessors become invalid.
  649. void detach(LifeCycleCookie) noexcept;
  650. void fully_detach() noexcept;
  651. ColumnType get_real_column_type(ColKey col_key) const noexcept;
  652. uint64_t get_sync_file_id() const noexcept;
  653. static size_t get_size_from_ref(ref_type top_ref, Allocator&) noexcept;
  654. static size_t get_size_from_ref(ref_type spec_ref, ref_type columns_ref, Allocator&) noexcept;
  655. /// Create an empty table with independent spec and return just
  656. /// the reference to the underlying memory.
  657. static ref_type create_empty_table(Allocator&, TableKey = TableKey());
  658. void nullify_links(CascadeState&);
  659. void remove_recursive(CascadeState&);
  660. /// Used by query. Follows chain of link columns and returns final target table
  661. const Table* get_link_chain_target(const std::vector<ColKey>&) const;
  662. Replication* get_repl() const noexcept;
  663. void set_ndx_in_parent(size_t ndx_in_parent) noexcept;
  664. /// Refresh the part of the accessor tree that is rooted at this
  665. /// table.
  666. void refresh_accessor_tree();
  667. void refresh_index_accessors();
  668. void refresh_content_version();
  669. void flush_for_commit();
  670. bool is_cross_table_link_target() const noexcept;
  671. template <typename T>
  672. void aggregate(QueryStateBase& st, ColKey col_key) const;
  673. template <typename T>
  674. double average(ColKey col_key, size_t* resultcount) const;
  675. std::vector<ColKey> m_leaf_ndx2colkey;
  676. std::vector<ColKey::Idx> m_spec_ndx2leaf_ndx;
  677. std::vector<size_t> m_leaf_ndx2spec_ndx;
  678. bool m_is_embedded = false;
  679. uint64_t m_in_file_version_at_transaction_boundary = 0;
  680. LifeCycleCookie m_cookie;
  681. static constexpr int top_position_for_spec = 0;
  682. static constexpr int top_position_for_columns = 1;
  683. static constexpr int top_position_for_cluster_tree = 2;
  684. static constexpr int top_position_for_key = 3;
  685. static constexpr int top_position_for_search_indexes = 4;
  686. static constexpr int top_position_for_column_key = 5;
  687. static constexpr int top_position_for_version = 6;
  688. static constexpr int top_position_for_opposite_table = 7;
  689. static constexpr int top_position_for_opposite_column = 8;
  690. static constexpr int top_position_for_sequence_number = 9;
  691. static constexpr int top_position_for_collision_map = 10;
  692. static constexpr int top_position_for_pk_col = 11;
  693. static constexpr int top_position_for_flags = 12;
  694. // flags contents: bit 0 - is table embedded?
  695. static constexpr int top_position_for_tombstones = 13;
  696. static constexpr int top_array_size = 14;
  697. enum { s_collision_map_lo = 0, s_collision_map_hi = 1, s_collision_map_local_id = 2, s_collision_map_num_slots };
  698. friend class SubtableNode;
  699. friend class _impl::TableFriend;
  700. friend class Query;
  701. friend class metrics::QueryInfo;
  702. template <class>
  703. friend class SimpleQuerySupport;
  704. friend class LangBindHelper;
  705. friend class ConstTableView;
  706. template <class T>
  707. friend class Columns;
  708. friend class Columns<StringData>;
  709. friend class ParentNode;
  710. friend struct util::serializer::SerialisationState;
  711. friend class LinkMap;
  712. friend class LinkView;
  713. friend class Group;
  714. friend class Transaction;
  715. friend class Cluster;
  716. friend class ClusterTree;
  717. friend class TableClusterTree;
  718. friend class ColKeyIterator;
  719. friend class Obj;
  720. friend class LnkLst;
  721. friend class Dictionary;
  722. friend class IncludeDescriptor;
  723. };
  724. class ColKeyIterator {
  725. public:
  726. bool operator!=(const ColKeyIterator& other)
  727. {
  728. return m_pos != other.m_pos;
  729. }
  730. ColKeyIterator& operator++()
  731. {
  732. ++m_pos;
  733. return *this;
  734. }
  735. ColKeyIterator operator++(int)
  736. {
  737. ColKeyIterator tmp(m_table, m_pos);
  738. ++m_pos;
  739. return tmp;
  740. }
  741. ColKey operator*()
  742. {
  743. if (m_pos < m_table->get_column_count()) {
  744. REALM_ASSERT(m_table->m_spec.get_key(m_pos) == m_table->spec_ndx2colkey(m_pos));
  745. return m_table->m_spec.get_key(m_pos);
  746. }
  747. return {};
  748. }
  749. private:
  750. friend class ColKeys;
  751. const Table* m_table;
  752. size_t m_pos;
  753. ColKeyIterator(const Table* t, size_t p)
  754. : m_table(t)
  755. , m_pos(p)
  756. {
  757. }
  758. };
  759. class ColKeys {
  760. public:
  761. ColKeys(const Table* t)
  762. : m_table(t)
  763. {
  764. }
  765. ColKeys()
  766. : m_table(nullptr)
  767. {
  768. }
  769. size_t size() const
  770. {
  771. return m_table->get_column_count();
  772. }
  773. bool empty() const
  774. {
  775. return size() == 0;
  776. }
  777. ColKey operator[](size_t p) const
  778. {
  779. return ColKeyIterator(m_table, p).operator*();
  780. }
  781. ColKeyIterator begin() const
  782. {
  783. return ColKeyIterator(m_table, 0);
  784. }
  785. ColKeyIterator end() const
  786. {
  787. return ColKeyIterator(m_table, size());
  788. }
  789. private:
  790. const Table* m_table;
  791. };
  792. // Class used to collect a chain of links when building up a Query following links.
  793. // It has member functions corresponding to the ones defined on Table.
  794. class LinkChain {
  795. public:
  796. LinkChain(ConstTableRef t = {}, ExpressionComparisonType type = ExpressionComparisonType::Any)
  797. : m_current_table(t)
  798. , m_base_table(t)
  799. , m_comparison_type(type)
  800. {
  801. }
  802. ConstTableRef get_base_table()
  803. {
  804. return m_base_table;
  805. }
  806. ConstTableRef get_current_table() const
  807. {
  808. return m_current_table;
  809. }
  810. ColKey get_current_col() const
  811. {
  812. return m_link_cols.back();
  813. }
  814. LinkChain& link(ColKey link_column)
  815. {
  816. add(link_column);
  817. return *this;
  818. }
  819. LinkChain& link(std::string col_name)
  820. {
  821. auto ck = m_current_table->get_column_key(col_name);
  822. if (!ck) {
  823. throw std::runtime_error(util::format("%1 has no property %2", m_current_table->get_name(), col_name));
  824. }
  825. add(ck);
  826. return *this;
  827. }
  828. LinkChain& backlink(const Table& origin, ColKey origin_col_key)
  829. {
  830. auto backlink_col_key = origin.get_opposite_column(origin_col_key);
  831. return link(backlink_col_key);
  832. }
  833. Subexpr* column(const std::string&);
  834. Subexpr* subquery(Query subquery);
  835. template <class T>
  836. inline Columns<T> column(ColKey col_key)
  837. {
  838. m_current_table->report_invalid_key(col_key);
  839. // Check if user-given template type equals Realm type.
  840. auto ct = col_key.get_type();
  841. if (ct == col_type_LinkList)
  842. ct = col_type_Link;
  843. if constexpr (std::is_same_v<T, Dictionary>) {
  844. if (!col_key.is_dictionary())
  845. throw LogicError(LogicError::type_mismatch);
  846. }
  847. else {
  848. if (ct != ColumnTypeTraits<T>::column_id)
  849. throw LogicError(LogicError::type_mismatch);
  850. }
  851. if (std::is_same<T, Link>::value || std::is_same<T, LnkLst>::value || std::is_same<T, BackLink>::value) {
  852. m_link_cols.push_back(col_key);
  853. }
  854. return Columns<T>(col_key, m_base_table, m_link_cols, m_comparison_type);
  855. }
  856. template <class T>
  857. Columns<T> column(const Table& origin, ColKey origin_col_key)
  858. {
  859. static_assert(std::is_same<T, BackLink>::value, "");
  860. auto backlink_col_key = origin.get_opposite_column(origin_col_key);
  861. m_link_cols.push_back(backlink_col_key);
  862. return Columns<T>(backlink_col_key, m_base_table, std::move(m_link_cols));
  863. }
  864. template <class T>
  865. SubQuery<T> column(ColKey col_key, Query subquery)
  866. {
  867. static_assert(std::is_same<T, Link>::value, "A subquery must involve a link list or backlink column");
  868. return SubQuery<T>(column<T>(col_key), std::move(subquery));
  869. }
  870. template <class T>
  871. SubQuery<T> column(const Table& origin, ColKey origin_col_key, Query subquery)
  872. {
  873. static_assert(std::is_same<T, BackLink>::value, "A subquery must involve a link list or backlink column");
  874. return SubQuery<T>(column<T>(origin, origin_col_key), std::move(subquery));
  875. }
  876. template <class T>
  877. BacklinkCount<T> get_backlink_count()
  878. {
  879. return BacklinkCount<T>(m_base_table, std::move(m_link_cols));
  880. }
  881. private:
  882. friend class Table;
  883. friend class query_parser::ParserDriver;
  884. std::vector<ColKey> m_link_cols;
  885. ConstTableRef m_current_table;
  886. ConstTableRef m_base_table;
  887. ExpressionComparisonType m_comparison_type;
  888. void add(ColKey ck);
  889. template <class T>
  890. Subexpr* create_subexpr(ColKey col_key)
  891. {
  892. return new Columns<T>(col_key, m_base_table, m_link_cols, m_comparison_type);
  893. }
  894. };
  895. // Implementation:
  896. inline ColKeys Table::get_column_keys() const
  897. {
  898. return ColKeys(this);
  899. }
  900. inline uint_fast64_t Table::get_content_version() const noexcept
  901. {
  902. return m_alloc.get_content_version();
  903. }
  904. inline uint_fast64_t Table::get_instance_version() const noexcept
  905. {
  906. return m_alloc.get_instance_version();
  907. }
  908. inline uint_fast64_t Table::get_storage_version(uint64_t instance_version) const
  909. {
  910. return m_alloc.get_storage_version(instance_version);
  911. }
  912. inline uint_fast64_t Table::get_storage_version() const
  913. {
  914. return m_alloc.get_storage_version();
  915. }
  916. inline TableKey Table::get_key() const noexcept
  917. {
  918. return m_key;
  919. }
  920. inline void Table::bump_storage_version() const noexcept
  921. {
  922. return m_alloc.bump_storage_version();
  923. }
  924. inline void Table::bump_content_version() const noexcept
  925. {
  926. m_alloc.bump_content_version();
  927. }
  928. inline size_t Table::get_column_count() const noexcept
  929. {
  930. return m_spec.get_public_column_count();
  931. }
  932. inline bool Table::is_embedded() const noexcept
  933. {
  934. return m_is_embedded;
  935. }
  936. inline StringData Table::get_column_name(ColKey column_key) const
  937. {
  938. auto spec_ndx = colkey2spec_ndx(column_key);
  939. REALM_ASSERT_3(spec_ndx, <, get_column_count());
  940. return m_spec.get_column_name(spec_ndx);
  941. }
  942. inline ColKey Table::get_column_key(StringData name) const noexcept
  943. {
  944. size_t spec_ndx = m_spec.get_column_index(name);
  945. if (spec_ndx == npos)
  946. return ColKey();
  947. return spec_ndx2colkey(spec_ndx);
  948. }
  949. inline ColumnType Table::get_real_column_type(ColKey col_key) const noexcept
  950. {
  951. return col_key.get_type();
  952. }
  953. inline DataType Table::get_column_type(ColKey column_key) const
  954. {
  955. return DataType(column_key.get_type());
  956. }
  957. inline ColumnAttrMask Table::get_column_attr(ColKey column_key) const noexcept
  958. {
  959. return column_key.get_attrs();
  960. }
  961. inline DataType Table::get_dictionary_key_type(ColKey column_key) const noexcept
  962. {
  963. auto spec_ndx = colkey2spec_ndx(column_key);
  964. REALM_ASSERT_3(spec_ndx, <, get_column_count());
  965. return m_spec.get_dictionary_key_type(spec_ndx);
  966. }
  967. inline Table::Table(Allocator& alloc)
  968. : m_alloc(alloc)
  969. , m_top(m_alloc)
  970. , m_spec(m_alloc)
  971. , m_clusters(this, m_alloc, top_position_for_cluster_tree)
  972. , m_index_refs(m_alloc)
  973. , m_opposite_table(m_alloc)
  974. , m_opposite_column(m_alloc)
  975. , m_repl(&g_dummy_replication)
  976. , m_own_ref(this, alloc.get_instance_version())
  977. {
  978. m_spec.set_parent(&m_top, top_position_for_spec);
  979. m_index_refs.set_parent(&m_top, top_position_for_search_indexes);
  980. m_opposite_table.set_parent(&m_top, top_position_for_opposite_table);
  981. m_opposite_column.set_parent(&m_top, top_position_for_opposite_column);
  982. ref_type ref = create_empty_table(m_alloc); // Throws
  983. ArrayParent* parent = nullptr;
  984. size_t ndx_in_parent = 0;
  985. init(ref, parent, ndx_in_parent, true, false);
  986. }
  987. inline Table::Table(Replication* const* repl, Allocator& alloc)
  988. : m_alloc(alloc)
  989. , m_top(m_alloc)
  990. , m_spec(m_alloc)
  991. , m_clusters(this, m_alloc, top_position_for_cluster_tree)
  992. , m_index_refs(m_alloc)
  993. , m_opposite_table(m_alloc)
  994. , m_opposite_column(m_alloc)
  995. , m_repl(repl)
  996. , m_own_ref(this, alloc.get_instance_version())
  997. {
  998. m_spec.set_parent(&m_top, top_position_for_spec);
  999. m_index_refs.set_parent(&m_top, top_position_for_search_indexes);
  1000. m_opposite_table.set_parent(&m_top, top_position_for_opposite_table);
  1001. m_opposite_column.set_parent(&m_top, top_position_for_opposite_column);
  1002. m_cookie = cookie_created;
  1003. }
  1004. inline void Table::revive(Replication* const* repl, Allocator& alloc, bool writable)
  1005. {
  1006. m_alloc.switch_underlying_allocator(alloc);
  1007. m_alloc.update_from_underlying_allocator(writable);
  1008. m_repl = repl;
  1009. m_own_ref = TableRef(this, m_alloc.get_instance_version());
  1010. // since we're rebinding to a new table, we'll bump version counters
  1011. // Possible optimization: save version counters along with the table data
  1012. // and restore them from there. Should decrease amount of non-necessary
  1013. // recomputations of any queries relying on this table.
  1014. bump_content_version();
  1015. bump_storage_version();
  1016. // we assume all other accessors are detached, so we're done.
  1017. }
  1018. inline Allocator& Table::get_alloc() const
  1019. {
  1020. return m_alloc;
  1021. }
  1022. // For use by queries
  1023. template <class T>
  1024. inline Columns<T> Table::column(ColKey col_key, ExpressionComparisonType cmp_type) const
  1025. {
  1026. LinkChain lc(m_own_ref, cmp_type);
  1027. return lc.column<T>(col_key);
  1028. }
  1029. template <class T>
  1030. inline Columns<T> Table::column(const Table& origin, ColKey origin_col_key) const
  1031. {
  1032. LinkChain lc(m_own_ref);
  1033. return lc.column<T>(origin, origin_col_key);
  1034. }
  1035. template <class T>
  1036. inline BacklinkCount<T> Table::get_backlink_count() const
  1037. {
  1038. return BacklinkCount<T>(this, {});
  1039. }
  1040. template <class T>
  1041. SubQuery<T> Table::column(ColKey col_key, Query subquery) const
  1042. {
  1043. LinkChain lc(m_own_ref);
  1044. return lc.column<T>(col_key, subquery);
  1045. }
  1046. template <class T>
  1047. SubQuery<T> Table::column(const Table& origin, ColKey origin_col_key, Query subquery) const
  1048. {
  1049. LinkChain lc(m_own_ref);
  1050. return lc.column<T>(origin, origin_col_key, subquery);
  1051. }
  1052. inline LinkChain Table::link(ColKey link_column) const
  1053. {
  1054. LinkChain lc(m_own_ref);
  1055. lc.add(link_column);
  1056. return lc;
  1057. }
  1058. inline LinkChain Table::backlink(const Table& origin, ColKey origin_col_key) const
  1059. {
  1060. auto backlink_col_key = origin.get_opposite_column(origin_col_key);
  1061. return link(backlink_col_key);
  1062. }
  1063. inline bool Table::is_empty() const noexcept
  1064. {
  1065. return size() == 0;
  1066. }
  1067. inline ConstTableRef Table::get_link_target(ColKey col_key) const noexcept
  1068. {
  1069. return const_cast<Table*>(this)->get_link_target(col_key);
  1070. }
  1071. inline bool Table::is_group_level() const noexcept
  1072. {
  1073. return bool(get_parent_group());
  1074. }
  1075. inline bool Table::operator==(const Table& t) const
  1076. {
  1077. return m_spec == t.m_spec && compare_objects(t); // Throws
  1078. }
  1079. inline bool Table::operator!=(const Table& t) const
  1080. {
  1081. return !(*this == t); // Throws
  1082. }
  1083. inline size_t Table::get_size_from_ref(ref_type top_ref, Allocator& alloc) noexcept
  1084. {
  1085. const char* top_header = alloc.translate(top_ref);
  1086. std::pair<int_least64_t, int_least64_t> p = Array::get_two(top_header, 0);
  1087. ref_type spec_ref = to_ref(p.first), columns_ref = to_ref(p.second);
  1088. return get_size_from_ref(spec_ref, columns_ref, alloc);
  1089. }
  1090. inline bool Table::is_link_type(ColumnType col_type) noexcept
  1091. {
  1092. return col_type == col_type_Link || col_type == col_type_LinkList;
  1093. }
  1094. inline Replication* Table::get_repl() const noexcept
  1095. {
  1096. return *m_repl;
  1097. }
  1098. inline void Table::set_ndx_in_parent(size_t ndx_in_parent) noexcept
  1099. {
  1100. REALM_ASSERT(m_top.is_attached());
  1101. m_top.set_ndx_in_parent(ndx_in_parent);
  1102. }
  1103. inline size_t Table::colkey2spec_ndx(ColKey key) const
  1104. {
  1105. auto leaf_idx = key.get_index();
  1106. REALM_ASSERT(leaf_idx.val < m_leaf_ndx2spec_ndx.size());
  1107. return m_leaf_ndx2spec_ndx[leaf_idx.val];
  1108. }
  1109. inline ColKey Table::spec_ndx2colkey(size_t spec_ndx) const
  1110. {
  1111. REALM_ASSERT(spec_ndx < m_spec_ndx2leaf_ndx.size());
  1112. return m_leaf_ndx2colkey[m_spec_ndx2leaf_ndx[spec_ndx].val];
  1113. }
  1114. inline void Table::report_invalid_key(ColKey col_key) const
  1115. {
  1116. if (col_key == ColKey())
  1117. throw LogicError(LogicError::column_does_not_exist);
  1118. auto idx = col_key.get_index();
  1119. if (idx.val >= m_leaf_ndx2colkey.size() || m_leaf_ndx2colkey[idx.val] != col_key)
  1120. throw LogicError(LogicError::column_does_not_exist);
  1121. }
  1122. inline size_t Table::leaf_ndx2spec_ndx(ColKey::Idx leaf_ndx) const
  1123. {
  1124. REALM_ASSERT(leaf_ndx.val < m_leaf_ndx2colkey.size());
  1125. return m_leaf_ndx2spec_ndx[leaf_ndx.val];
  1126. }
  1127. inline ColKey::Idx Table::spec_ndx2leaf_ndx(size_t spec_ndx) const
  1128. {
  1129. REALM_ASSERT(spec_ndx < m_spec_ndx2leaf_ndx.size());
  1130. return m_spec_ndx2leaf_ndx[spec_ndx];
  1131. }
  1132. inline ColKey Table::leaf_ndx2colkey(ColKey::Idx leaf_ndx) const
  1133. {
  1134. // this may be called with leaf indicies outside of the table. This can happen
  1135. // when a column is removed from the mapping, but space for it is still reserved
  1136. // at leaf level. Operations on Cluster and ClusterTree which walks the columns
  1137. // based on leaf indicies may ask for colkeys which are no longer valid.
  1138. if (leaf_ndx.val < m_leaf_ndx2spec_ndx.size())
  1139. return m_leaf_ndx2colkey[leaf_ndx.val];
  1140. else
  1141. return ColKey();
  1142. }
  1143. bool inline Table::valid_column(ColKey col_key) const noexcept
  1144. {
  1145. if (col_key == ColKey())
  1146. return false;
  1147. ColKey::Idx leaf_idx = col_key.get_index();
  1148. if (leaf_idx.val >= m_leaf_ndx2colkey.size())
  1149. return false;
  1150. return col_key == m_leaf_ndx2colkey[leaf_idx.val];
  1151. }
  1152. inline void Table::check_column(ColKey col_key) const
  1153. {
  1154. if (REALM_UNLIKELY(!valid_column(col_key)))
  1155. throw ColumnNotFound();
  1156. }
  1157. // The purpose of this class is to give internal access to some, but
  1158. // not all of the non-public parts of the Table class.
  1159. class _impl::TableFriend {
  1160. public:
  1161. static Spec& get_spec(Table& table) noexcept
  1162. {
  1163. return table.m_spec;
  1164. }
  1165. static const Spec& get_spec(const Table& table) noexcept
  1166. {
  1167. return table.m_spec;
  1168. }
  1169. static TableRef get_opposite_link_table(const Table& table, ColKey col_key);
  1170. static Group* get_parent_group(const Table& table) noexcept
  1171. {
  1172. return table.get_parent_group();
  1173. }
  1174. static void remove_recursive(Table& table, CascadeState& rows)
  1175. {
  1176. table.remove_recursive(rows); // Throws
  1177. }
  1178. static void batch_erase_rows(Table& table, const KeyColumn& keys)
  1179. {
  1180. table.batch_erase_rows(keys); // Throws
  1181. }
  1182. // Temporary hack
  1183. static Obj create_linked_object(Table& table, GlobalKey id)
  1184. {
  1185. return table.create_linked_object(id);
  1186. }
  1187. static ObjKey global_to_local_object_id_hashed(const Table& table, GlobalKey global_id)
  1188. {
  1189. return table.global_to_local_object_id_hashed(global_id);
  1190. }
  1191. };
  1192. } // namespace realm
  1193. #endif // REALM_TABLE_HPP