group.hpp 51 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357
  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_GROUP_HPP
  19. #define REALM_GROUP_HPP
  20. #include <functional>
  21. #include <map>
  22. #include <optional>
  23. #include <stdexcept>
  24. #include <string>
  25. #include <vector>
  26. #include <chrono>
  27. #include <realm/alloc_slab.hpp>
  28. #include <realm/exceptions.hpp>
  29. #include <realm/impl/cont_transact_hist.hpp>
  30. #include <realm/impl/output_stream.hpp>
  31. #include <realm/metrics/metrics.hpp>
  32. #include <realm/table.hpp>
  33. #include <realm/util/features.h>
  34. #include <realm/util/input_stream.hpp>
  35. namespace realm {
  36. class DB;
  37. class TableKeys;
  38. namespace _impl {
  39. class GroupFriend;
  40. class TransactLogParser;
  41. } // namespace _impl
  42. /// A group is a collection of named tables.
  43. ///
  44. class Group : public ArrayParent {
  45. public:
  46. /// Construct a free-standing group. This group instance will be
  47. /// in the attached state, but neither associated with a file, nor
  48. /// with an external memory buffer.
  49. Group();
  50. /// Attach this Group instance to the specified database file.
  51. ///
  52. /// The specified file is opened in read-only mode. This allows opening
  53. /// a file even when the caller lacks permission to write to that file.
  54. /// The opened group may still be modified freely, but the changes cannot be
  55. /// written back to the same file. Tt is an error if the specified
  56. /// file does not already exist in the file system.
  57. ///
  58. /// The file must contain a valid Realm database. In many cases invalidity
  59. /// will be detected and cause the InvalidDatabase exception to be thrown,
  60. /// but you should not rely on it.
  61. ///
  62. /// You may call write() to write the entire database to a new file. Writing
  63. /// the database to a new file does not end, or in any other way
  64. /// change the association between the Group instance and the file
  65. /// that was specified in the call to open().
  66. ///
  67. /// A Realm file that contains a history (see Replication::HistoryType) may
  68. /// be opened via Group::open(), as long as the application can ensure that
  69. /// there is no concurrent access to the file (see below for more on
  70. /// concurrency).
  71. ///
  72. /// A file that is passed to Group::open(), may not be modified by
  73. /// a third party until after the Group object is
  74. /// destroyed. Behavior is undefined if a file is modified by a
  75. /// third party while any Group object is associated with it.
  76. ///
  77. /// Calling open() on a Group instance that is already in the
  78. /// attached state has undefined behavior.
  79. ///
  80. /// Accessing a Realm database file through manual construction
  81. /// of a Group object does not offer any level of thread safety or
  82. /// transaction safety. When any of those kinds of safety are a
  83. /// concern, consider using a DB instead. When accessing
  84. /// a database file in read/write mode through a manually
  85. /// constructed Group object, it is entirely the responsibility of
  86. /// the application that the file is not accessed in any way by a
  87. /// third party during the life-time of that group object. It is,
  88. /// on the other hand, safe to concurrently access a database file
  89. /// by multiple manually created Group objects, as long as all of
  90. /// them are opened in read-only mode, and there is no other party
  91. /// that modifies the file concurrently.
  92. ///
  93. /// Do not call this function on a group instance that is managed
  94. /// by a shared group. Doing so will result in undefined behavior.
  95. ///
  96. /// Even if this function throws, it may have the side-effect of
  97. /// creating the specified file, and the file may get left behind
  98. /// in an invalid state. Of course, this can only happen if
  99. /// read/write mode (mode_ReadWrite) was requested, and the file
  100. /// did not already exist.
  101. ///
  102. /// \param file File system path to a Realm database file.
  103. ///
  104. /// \param encryption_key 32-byte key used to encrypt and decrypt
  105. /// the database file, or nullptr to disable encryption.
  106. ///
  107. /// \throw FileAccessError If the file could not be
  108. /// opened. If the reason corresponds to one of the exception
  109. /// types that are derived from FileAccessError, the
  110. /// derived exception type is thrown. Note that InvalidDatabase is
  111. /// among these derived exception types.
  112. explicit Group(const std::string& file, const char* encryption_key = nullptr);
  113. /// Attach this Group instance to the specified memory buffer.
  114. ///
  115. /// This is similar to constructing a group from a file except
  116. /// that in this case the database is assumed to be stored in the
  117. /// specified memory buffer.
  118. ///
  119. /// If \a take_ownership is `true`, you pass the ownership of the
  120. /// specified buffer to the group. In this case the buffer will
  121. /// eventually be freed using std::free(), so the buffer you pass,
  122. /// must have been allocated using std::malloc().
  123. ///
  124. /// On the other hand, if \a take_ownership is set to `false`, it
  125. /// is your responsibility to keep the memory buffer alive during
  126. /// the lifetime of the group, and in case the buffer needs to be
  127. /// deallocated afterwards, that is your responsibility too.
  128. ///
  129. /// If this function throws, the ownership of the memory buffer
  130. /// will remain with the caller, regardless of whether \a
  131. /// take_ownership is set to `true` or `false`.
  132. ///
  133. /// Calling open() on a Group instance that is already in the
  134. /// attached state has undefined behavior.
  135. ///
  136. /// Do not call this function on a group instance that is managed
  137. /// by a shared group. Doing so will result in undefined behavior.
  138. ///
  139. /// \throw InvalidDatabase If the specified buffer does not appear
  140. /// to contain a valid database.
  141. /// Note that if this constructor throws, the
  142. /// ownership of the memory buffer will remain with the caller,
  143. /// regardless of whether \a take_ownership is set to `true` or
  144. /// `false`.
  145. explicit Group(BinaryData, bool take_ownership = true);
  146. Group(const Group&) = delete;
  147. Group& operator=(const Group&) = delete;
  148. ~Group() noexcept override;
  149. /// A group may be created in the unattached state, and then later
  150. /// attached to a file with a call to open(). Calling any method
  151. /// other than open(), and is_attached() on an unattached instance
  152. /// results in undefined behavior.
  153. bool is_attached() const noexcept;
  154. /// A group is frozen only if it is actually a frozen transaction.
  155. virtual bool is_frozen() const noexcept
  156. {
  157. return false;
  158. }
  159. /// Returns true if, and only if the number of tables in this
  160. /// group is zero.
  161. bool is_empty() const noexcept;
  162. size_t size() const noexcept;
  163. static int get_current_file_format_version()
  164. {
  165. return g_current_file_format_version;
  166. }
  167. int get_history_schema_version() noexcept;
  168. Replication* get_replication() const
  169. {
  170. return *get_repl();
  171. }
  172. /// The sync file id is set when a client synchronizes with the server for the
  173. /// first time. It is used when generating GlobalKeys for tables without a primary
  174. /// key, where it is used as the "hi" part. This ensures global uniqueness of
  175. /// GlobalKeys.
  176. uint64_t get_sync_file_id() const noexcept;
  177. void set_sync_file_id(uint64_t id);
  178. /// Returns the keys for all tables in this group.
  179. TableKeys get_table_keys() const;
  180. /// \defgroup group_table_access Table Accessors
  181. ///
  182. /// has_table() returns true if, and only if this group contains a table
  183. /// with the specified name.
  184. ///
  185. /// find_table() returns the key of the first table in this group with the
  186. /// specified name, or `realm::not_found` if this group does not contain a
  187. /// table with the specified name.
  188. ///
  189. /// get_table_name() returns the name of table with the specified key.
  190. ///
  191. /// The versions of get_table(), that accepts a \a name argument, return a
  192. /// table with the specified name, or null if no such table exists.
  193. ///
  194. /// add_table() adds a table with the specified name to this group. It
  195. /// throws TableNameInUse if \a require_unique_name is true and \a name
  196. /// clashes with the name of an existing table. If \a require_unique_name is
  197. /// false, it is possible to add more than one table with the same
  198. /// name. Whenever a table is added the key assigned to it is returned.
  199. ///
  200. /// get_or_add_table() checks if a table exists in this group with the specified
  201. /// name. If it doesn't exist, a table is created.
  202. ///
  203. /// remove_table() removes the specified table from this group. A table can
  204. /// be removed only when it is not the target of a link column of a
  205. /// different table.
  206. ///
  207. /// rename_table() changes the name of a preexisting table. If \a
  208. /// require_unique_name is false, it becomes possible to have more than one
  209. /// table with a given name in a single group.
  210. ///
  211. /// The template functions work exactly like their non-template namesakes
  212. /// except as follows: The template versions of get_table() and
  213. /// get_or_add_table() throw DescriptorMismatch if the dynamic type of the
  214. /// specified table does not match the statically specified custom table
  215. /// type. The template versions of add_table() and get_or_add_table() set
  216. /// the dynamic type (descriptor) to match the statically specified custom
  217. /// table type.
  218. ///
  219. /// \param key Key of table in this group.
  220. ///
  221. /// \param name Name of table. All strings are valid table names as long as
  222. /// they are valid UTF-8 encodings and the number of bytes does not exceed
  223. /// `max_table_name_length`. A call to add_table() or get_or_add_table()
  224. /// with a name that is longer than `max_table_name_length` will cause an
  225. /// exception to be thrown.
  226. ///
  227. /// \param new_name New name for preexisting table.
  228. ///
  229. /// \param require_unique_name When set to true (the default), it becomes
  230. /// impossible to add a table with a name that is already in use, or to
  231. /// rename a table to a name that is already in use.
  232. ///
  233. /// \param was_added When specified, the boolean variable is set to true if
  234. /// the table was added, and to false otherwise. If the function throws, the
  235. /// boolean variable retains its original value.
  236. ///
  237. /// \return get_table(), add_table(), and get_or_add_table() return a table
  238. /// accessor attached to the requested (or added) table. get_table() may
  239. /// return null.
  240. ///
  241. /// \throw DescriptorMismatch Thrown by get_table() and get_or_add_table()
  242. /// tf the dynamic table type does not match the statically specified custom
  243. /// table type (\a T).
  244. ///
  245. /// \throw NoSuchTable Thrown by remove_table() and rename_table() if there
  246. /// is no table with the specified \a name.
  247. ///
  248. /// \throw TableNameInUse Thrown by add_table() if \a require_unique_name is
  249. /// true and \a name clashes with the name of a preexisting table. Thrown by
  250. /// rename_table() if \a require_unique_name is true and \a new_name clashes
  251. /// with the name of a preexisting table.
  252. ///
  253. /// \throw CrossTableLinkTarget Thrown by remove_table() if the specified
  254. /// table is the target of a link column of a different table.
  255. ///
  256. //@{
  257. static const size_t max_table_name_length = 63;
  258. bool has_table(StringData name) const noexcept;
  259. TableKey find_table(StringData name) const noexcept;
  260. StringData get_table_name(TableKey key) const;
  261. StringData get_class_name(TableKey key) const
  262. {
  263. return table_name_to_class_name(get_table_name(key));
  264. }
  265. bool table_is_public(TableKey key) const;
  266. static StringData table_name_to_class_name(StringData table_name)
  267. {
  268. if (table_name.begins_with(g_class_name_prefix)) {
  269. return table_name.substr(g_class_name_prefix.size());
  270. }
  271. return table_name;
  272. }
  273. using TableNameBuffer = std::array<char, max_table_name_length>;
  274. static StringData class_name_to_table_name(StringData class_name, TableNameBuffer& buffer)
  275. {
  276. char* p = std::copy_n(g_class_name_prefix.data(), g_class_name_prefix.size(), buffer.data());
  277. size_t len = std::min(class_name.size(), buffer.size() - g_class_name_prefix.size());
  278. std::copy_n(class_name.data(), len, p);
  279. return StringData(buffer.data(), g_class_name_prefix.size() + len);
  280. }
  281. TableRef get_table(TableKey key);
  282. ConstTableRef get_table(TableKey key) const;
  283. // Catch some implicit conversions
  284. TableRef get_table(int) = delete;
  285. ConstTableRef get_table(int) const = delete;
  286. TableRef get_table(StringData name);
  287. ConstTableRef get_table(StringData name) const;
  288. TableRef add_table(StringData name, Table::Type table_type = Table::Type::TopLevel);
  289. TableRef add_table_with_primary_key(StringData name, DataType pk_type, StringData pk_name, bool nullable = false,
  290. Table::Type table_type = Table::Type::TopLevel);
  291. TableRef get_or_add_table(StringData name, Table::Type table_type = Table::Type::TopLevel,
  292. bool* was_added = nullptr);
  293. TableRef get_or_add_table_with_primary_key(StringData name, DataType pk_type, StringData pk_name,
  294. bool nullable = false, Table::Type table_type = Table::Type::TopLevel);
  295. void remove_table(TableKey key);
  296. void remove_table(StringData name);
  297. void rename_table(TableKey key, StringData new_name, bool require_unique_name = true);
  298. void rename_table(StringData name, StringData new_name, bool require_unique_name = true);
  299. Obj get_object(ObjLink link);
  300. void validate(ObjLink link) const;
  301. //@}
  302. // Serialization
  303. /// Write this database to the specified output stream.
  304. ///
  305. /// \param out The destination output stream to write to.
  306. ///
  307. /// \param pad If true, the file is padded to ensure the footer is aligned
  308. /// to the end of a page
  309. void write(std::ostream& out, bool pad = false) const;
  310. /// Write this database to a new file. It is an error to specify a
  311. /// file that already exists. This is to protect against
  312. /// overwriting a database file that is currently open, which
  313. /// would cause undefined behaviour.
  314. ///
  315. /// \param path A filesystem path to the file you want to write to.
  316. ///
  317. /// \param encryption_key 32-byte key used to encrypt the database file,
  318. /// or nullptr to disable encryption.
  319. ///
  320. /// \param version If different from 0, the new file will be a full fledged
  321. /// realm file with free list and history info. The version of the commit
  322. /// will be set to the value given here.
  323. ///
  324. /// \param write_history Indicates if you want the Sync Client History to
  325. /// be written to the file (only relevant for synchronized files).
  326. /// \throw FileAccessError If the file could not be
  327. /// opened. If the reason corresponds to one of the exception
  328. /// types that are derived from FileAccessError, the
  329. /// derived exception type is thrown. In particular,
  330. /// util::File::Exists will be thrown if the file exists already.
  331. void write(const std::string& path, const char* encryption_key = nullptr, uint64_t version = 0,
  332. bool write_history = true) const;
  333. /// Write this database to a memory buffer.
  334. ///
  335. /// Ownership of the returned buffer is transferred to the
  336. /// caller. The memory will have been allocated using
  337. /// std::malloc().
  338. BinaryData write_to_mem() const;
  339. //@{
  340. /// Some operations on Tables in a Group can cause indirect changes to other
  341. /// fields, including in other Tables in the same Group. Specifically,
  342. /// removing a row will set any links to that row to null, and if it had the
  343. /// last strong links to other rows, will remove those rows. When this
  344. /// happens, The cascade notification handler will be called with a
  345. /// CascadeNotification containing information about what indirect changes
  346. /// will occur, before any changes are made.
  347. ///
  348. /// has_cascade_notification_handler() returns true if and only if there is
  349. /// currently a non-null notification handler registered.
  350. ///
  351. /// set_cascade_notification_handler() replaces the current handler (if any)
  352. /// with the passed in handler. Pass in nullptr to remove the current handler
  353. /// without registering a new one.
  354. ///
  355. /// CascadeNotification contains a vector of rows which will be removed and
  356. /// a vector of links which will be set to null (or removed, for entries in
  357. /// LinkLists).
  358. struct CascadeNotification {
  359. struct row {
  360. /// Key identifying a group-level table.
  361. TableKey table_key;
  362. /// Key identifying object to be removed.
  363. ObjKey key;
  364. row() = default;
  365. row(TableKey tk, ObjKey k)
  366. : table_key(tk)
  367. , key(k)
  368. {
  369. }
  370. bool operator==(const row& r) const noexcept
  371. {
  372. return table_key == r.table_key && key == r.key;
  373. }
  374. bool operator!=(const row& r) const noexcept
  375. {
  376. return !(*this == r);
  377. }
  378. /// Trivial lexicographic order
  379. bool operator<(const row& r) const noexcept
  380. {
  381. return table_key < r.table_key || (table_key == r.table_key && key < r.key);
  382. }
  383. };
  384. struct link {
  385. link() = default;
  386. link(TableKey tk, ColKey ck, ObjKey k, ObjKey otk)
  387. : origin_table(tk)
  388. , origin_col_key(ck)
  389. , origin_key(k)
  390. , old_target_key(otk)
  391. {
  392. }
  393. TableKey origin_table; ///< A group-level table.
  394. ColKey origin_col_key; ///< Link column being nullified.
  395. ObjKey origin_key; ///< Row in column being nullified.
  396. /// The target row index which is being removed. Mostly relevant for
  397. /// LinkList (to know which entries are being removed), but also
  398. /// valid for Link.
  399. ObjKey old_target_key;
  400. };
  401. /// A sorted list of rows which will be removed by the current operation.
  402. std::vector<row> rows;
  403. /// An unordered list of links which will be nullified by the current
  404. /// operation.
  405. std::vector<link> links;
  406. };
  407. bool has_cascade_notification_handler() const noexcept;
  408. void
  409. set_cascade_notification_handler(util::UniqueFunction<void(const CascadeNotification&)> new_handler) noexcept;
  410. //@}
  411. //@{
  412. /// During sync operation, schema changes may happen at runtime as connected
  413. /// clients update their schema as part of an app update. Since this is a
  414. /// relatively rare event, no attempt is made at limiting the amount of work
  415. /// the handler is required to do to update its information about table and
  416. /// column indices (i.e., all table and column indices must be recalculated).
  417. ///
  418. /// At the time of writing, only additive schema changes may occur in that
  419. /// scenario.
  420. ///
  421. /// has_schema_change_notification_handler() returns true iff there is currently
  422. /// a non-null notification handler registered.
  423. ///
  424. /// set_schema_change_notification_handler() replaces the current handler (if any)
  425. /// with the passed in handler. Pass in nullptr to remove the current handler
  426. /// without registering a new one.
  427. bool has_schema_change_notification_handler() const noexcept;
  428. void set_schema_change_notification_handler(util::UniqueFunction<void()> new_handler) noexcept;
  429. //@}
  430. // Conversion
  431. void schema_to_json(std::ostream& out, std::map<std::string, std::string>* renames = nullptr) const;
  432. void to_json(std::ostream& out, size_t link_depth = 0, std::map<std::string, std::string>* renames = nullptr,
  433. JSONOutputMode output_mode = output_mode_json) const;
  434. /// Compare two groups for equality. Two groups are equal if, and
  435. /// only if, they contain the same tables in the same order, that
  436. /// is, for each table T at index I in one of the groups, there is
  437. /// a table at index I in the other group that is equal to T.
  438. /// Tables are equal if they have the same content and the same table name.
  439. bool operator==(const Group&) const;
  440. /// Compare two groups for inequality. See operator==().
  441. bool operator!=(const Group& g) const
  442. {
  443. return !(*this == g);
  444. }
  445. /// Return the size taken up by the current snapshot. This is in contrast to
  446. /// the number returned by DB::get_stats() which will return the
  447. /// size of the last snapshot done in that DB. If the snapshots are
  448. /// identical, the numbers will of course be equal.
  449. size_t get_used_space() const noexcept;
  450. /// check that an already attached realm file is valid for read only access.
  451. /// if not detach the file and throw a FileFormatUpgradeRequired.
  452. /// return the file format version.
  453. static int read_only_version_check(SlabAlloc& alloc, ref_type top_ref, const std::string& path);
  454. void verify() const;
  455. void validate_primary_columns();
  456. #ifdef REALM_DEBUG
  457. void print() const;
  458. void print_free() const;
  459. MemStats get_stats();
  460. void enable_mem_diagnostics(bool enable = true)
  461. {
  462. m_alloc.enable_debug(enable);
  463. }
  464. #endif
  465. protected:
  466. static constexpr size_t s_table_name_ndx = 0;
  467. static constexpr size_t s_table_refs_ndx = 1;
  468. static constexpr size_t s_file_size_ndx = 2;
  469. static constexpr size_t s_free_pos_ndx = 3;
  470. static constexpr size_t s_free_size_ndx = 4;
  471. static constexpr size_t s_free_version_ndx = 5;
  472. static constexpr size_t s_version_ndx = 6;
  473. static constexpr size_t s_hist_type_ndx = 7;
  474. static constexpr size_t s_hist_ref_ndx = 8;
  475. static constexpr size_t s_hist_version_ndx = 9;
  476. static constexpr size_t s_sync_file_id_ndx = 10;
  477. static constexpr size_t s_evacuation_point_ndx = 11;
  478. static constexpr size_t s_group_max_size = 12;
  479. virtual Replication* const* get_repl() const
  480. {
  481. return &Table::g_dummy_replication;
  482. }
  483. private:
  484. static constexpr StringData g_class_name_prefix = "class_";
  485. struct ToDeleteRef {
  486. ToDeleteRef(TableKey tk, ObjKey k)
  487. : table_key(tk)
  488. , obj_key(k)
  489. , ttl(std::chrono::steady_clock::now())
  490. {
  491. }
  492. TableKey table_key;
  493. ObjKey obj_key;
  494. std::chrono::steady_clock::time_point ttl;
  495. };
  496. // nullptr, if we're sharing an allocator provided during initialization
  497. std::unique_ptr<SlabAlloc> m_local_alloc;
  498. // in-use allocator. If local, then equal to m_local_alloc.
  499. SlabAlloc& m_alloc;
  500. int m_file_format_version;
  501. /// `m_top` is the root node (or top array) of the Realm, and has the
  502. /// following layout:
  503. ///
  504. /// <pre>
  505. ///
  506. /// Introduced in file
  507. /// Slot Value format version
  508. /// ---------------------------------------------------------------------
  509. /// 1st m_table_names
  510. /// 2nd m_tables
  511. /// 3rd Logical file size
  512. /// 4th GroupWriter::m_free_positions (optional)
  513. /// 5th GroupWriter::m_free_lengths (optional)
  514. /// 6th GroupWriter::m_free_versions (optional)
  515. /// 7th Transaction number / version (optional)
  516. /// 8th History type (optional) 4
  517. /// 9th History ref (optional) 4
  518. /// 10th History version (optional) 7
  519. /// 11th Sync File Id (optional) 10
  520. /// 12th Evacuation point (optional) 22
  521. ///
  522. /// </pre>
  523. ///
  524. /// The 'History type' slot stores a value of type
  525. /// Replication::HistoryType. The 'History version' slot stores a history
  526. /// schema version as returned by Replication::get_history_schema_version().
  527. ///
  528. /// The first three entries are mandatory. In files created by
  529. /// Group::write(), none of the optional entries are present and the size of
  530. /// `m_top` is 3. In files updated by Group::commit(), the 4th and 5th entry
  531. /// are present, and the size of `m_top` is 5. In files updated by way of a
  532. /// transaction (Transaction::commit()), the 4th, 5th, 6th, and 7th entry
  533. /// are present, and the size of `m_top` is 7. In files that contain a
  534. /// changeset history, the 8th, 9th, and 10th entry are present. The 11th entry
  535. /// will be present if the file is syncked and the client has received a client
  536. /// file id from the server.
  537. ///
  538. /// When a group accessor is attached to a newly created file or an empty
  539. /// memory buffer where there is no top array yet, `m_top`, `m_tables`, and
  540. /// `m_table_names` will be left in the detached state until the initiation
  541. /// of the first write transaction. In particular, they will remain in the
  542. /// detached state during read transactions that precede the first write
  543. /// transaction.
  544. Array m_top;
  545. Array m_tables;
  546. ArrayStringShort m_table_names;
  547. uint64_t m_last_seen_mapping_version = 0;
  548. typedef std::vector<Table*> TableAccessors;
  549. mutable TableAccessors m_table_accessors;
  550. mutable std::mutex m_accessor_mutex;
  551. mutable int m_num_tables = 0;
  552. bool m_attached = false;
  553. bool m_is_writable = true;
  554. static std::optional<int> fake_target_file_format;
  555. util::UniqueFunction<void(const CascadeNotification&)> m_notify_handler;
  556. util::UniqueFunction<void()> m_schema_change_handler;
  557. std::shared_ptr<metrics::Metrics> m_metrics;
  558. std::vector<ToDeleteRef> m_objects_to_delete;
  559. size_t m_total_rows;
  560. Group(SlabAlloc* alloc) noexcept;
  561. void init_array_parents() noexcept;
  562. void open(ref_type top_ref, const std::string& file_path);
  563. // If the underlying memory mappings have been extended, this method is used
  564. // to update all the tables' allocator wrappers. The allocator wrappers are
  565. // configure to either allow or deny changes.
  566. void update_allocator_wrappers(bool writable);
  567. /// If `top_ref` is not zero, attach this group accessor to the specified
  568. /// underlying node structure. If `top_ref` is zero and \a
  569. /// create_group_when_missing is true, create a new node structure that
  570. /// represents an empty group, and attach this group accessor to it.
  571. void attach(ref_type top_ref, bool writable, bool create_group_when_missing, size_t file_size = -1,
  572. uint_fast64_t version = -1);
  573. /// Detach this group accessor from the underlying node structure. If this
  574. /// group accessors is already in the detached state, this function does
  575. /// nothing (idempotency).
  576. void detach() noexcept;
  577. /// \param writable Must be set to true when, and only when attaching for a
  578. /// write transaction.
  579. void attach_shared(ref_type new_top_ref, size_t new_file_size, bool writable, VersionID version);
  580. void create_empty_group();
  581. void remove_table(size_t table_ndx, TableKey key);
  582. void reset_free_space_tracking();
  583. void remap_and_update_refs(ref_type new_top_ref, size_t new_file_size, bool writable);
  584. /// Recursively update refs stored in all cached array
  585. /// accessors. This includes cached array accessors in any
  586. /// currently attached table accessors. This ensures that the
  587. /// group instance itself, as well as any attached table accessor
  588. /// that exists across Transaction::commit() will remain valid. This
  589. /// function is not appropriate for use in conjunction with
  590. /// commits via shared group.
  591. void update_refs(ref_type top_ref) noexcept;
  592. // Overriding method in ArrayParent
  593. void update_child_ref(size_t, ref_type) override;
  594. // Overriding method in ArrayParent
  595. ref_type get_child_ref(size_t) const noexcept override;
  596. class TableWriter;
  597. class DefaultTableWriter;
  598. static void write(std::ostream&, int file_format_version, TableWriter&, bool no_top_array,
  599. bool pad_for_encryption, uint_fast64_t version_number);
  600. Table* do_get_table(size_t ndx);
  601. const Table* do_get_table(size_t ndx) const;
  602. Table* do_get_table(StringData name);
  603. const Table* do_get_table(StringData name) const;
  604. Table* do_add_table(StringData name, Table::Type table_type, bool do_repl = true);
  605. void create_and_insert_table(TableKey key, StringData name);
  606. Table* create_table_accessor(size_t table_ndx);
  607. void recycle_table_accessor(Table*);
  608. void detach_table_accessors() noexcept; // Idempotent
  609. void mark_all_table_accessors() noexcept;
  610. void write(util::File& file, const char* encryption_key, uint_fast64_t version_number, TableWriter& writer) const;
  611. void write(std::ostream&, bool pad, uint_fast64_t version_numer, TableWriter& writer) const;
  612. std::shared_ptr<metrics::Metrics> get_metrics() const noexcept;
  613. void set_metrics(std::shared_ptr<metrics::Metrics> other) noexcept;
  614. void update_num_objects();
  615. class TransactAdvancer;
  616. /// Memory mappings must have been updated to reflect any growth in filesize before
  617. /// calling advance_transact()
  618. void advance_transact(ref_type new_top_ref, util::NoCopyInputStream&, bool writable);
  619. void refresh_dirty_accessors();
  620. void flush_accessors_for_commit();
  621. /// \brief The version of the format of the node structure (in file or in
  622. /// memory) in use by Realm objects associated with this group.
  623. ///
  624. /// Every group contains a file format version field, which is returned
  625. /// by this function. The file format version field is set to the file format
  626. /// version specified by the attached file (or attached memory buffer) at the
  627. /// time of attachment and the value is used to determine if a file format
  628. /// upgrade is required.
  629. ///
  630. /// A value of zero means that the file format is not yet decided. This is
  631. /// only possible for empty Realms where top-ref is zero. (When group is created
  632. /// with the unattached_tag). The version number will then be determined in the
  633. /// subsequent call to Group::open.
  634. ///
  635. /// In shared mode (when a Realm file is opened via a DB instance)
  636. /// it can happen that the file format is upgraded asyncronously (via
  637. /// another DB instance), and in that case the file format version
  638. /// field can get out of date, but only for a short while. It is always
  639. /// guaranteed to be, and remain up to date after the opening process completes
  640. /// (when DB::do_open() returns).
  641. ///
  642. /// An empty Realm file (one whose top-ref is zero) may specify a file
  643. /// format version of zero to indicate that the format is not yet
  644. /// decided. In that case the file format version must be changed to a proper
  645. /// before the opening process completes (Group::open() or DB::open()).
  646. ///
  647. /// File format versions:
  648. ///
  649. /// 1 Initial file format version
  650. ///
  651. /// 2 Various changes.
  652. ///
  653. /// 3 Supporting null on string columns broke the file format in following
  654. /// way: Index appends an 'X' character to all strings except the null
  655. /// string, to be able to distinguish between null and empty
  656. /// string. Bumped to 3 because of null support of String columns and
  657. /// because of new format of index.
  658. ///
  659. /// 4 Introduction of optional in-Realm history of changes (additional
  660. /// entries in Group::m_top). Since this change is not forward
  661. /// compatible, the file format version had to be bumped. This change is
  662. /// implemented in a way that achieves backwards compatibility with
  663. /// version 3 (and in turn with version 2).
  664. ///
  665. /// 5 Introduced the new Timestamp column type that replaces DateTime.
  666. /// When opening an older database file, all DateTime columns will be
  667. /// automatically upgraded Timestamp columns.
  668. ///
  669. /// 6 Introduced a new structure for the StringIndex. Moved the commit
  670. /// logs into the Realm file. Changes to the transaction log format
  671. /// including reshuffling instructions. This is the format used in
  672. /// milestone 2.0.0.
  673. ///
  674. /// 7 Introduced "history schema version" as 10th entry in top array.
  675. ///
  676. /// 8 Subtables can now have search index.
  677. ///
  678. /// 9 Replication instruction values shuffled, instr_MoveRow added.
  679. ///
  680. /// 10 Cluster based table layout. Memory mapping changes which require
  681. /// special treatment of large files of preceding versions.
  682. ///
  683. /// 11 Same as 10, but version 11 files will have search index added on
  684. /// string primary key columns.
  685. ///
  686. /// 12 - 19 Room for new file formats in legacy code.
  687. ///
  688. /// 20 New data types: Decimal128 and ObjectId. Embedded tables. Search index
  689. /// is removed from primary key columns.
  690. ///
  691. /// 21 New data types: UUID, Mixed, Set and Dictionary.
  692. ///
  693. /// 22 Object keys are no longer generated from primary key values. Search index
  694. /// reintroduced.
  695. ///
  696. /// 23 Layout of Set and Dictionary changed.
  697. ///
  698. /// IMPORTANT: When introducing a new file format version, be sure to review
  699. /// the file validity checks in Group::open() and DB::do_open, the file
  700. /// format selection logic in
  701. /// Group::get_target_file_format_version_for_session(), and the file format
  702. /// upgrade logic in Group::upgrade_file_format(), AND the lists of accepted
  703. /// file formats and the version deletion list residing in "backup_restore.cpp"
  704. static constexpr int g_current_file_format_version = 23;
  705. int get_file_format_version() const noexcept;
  706. void set_file_format_version(int) noexcept;
  707. int get_committed_file_format_version() const noexcept;
  708. /// The specified history type must be a value of Replication::HistoryType.
  709. static int get_target_file_format_version_for_session(int current_file_format_version, int history_type) noexcept;
  710. void send_cascade_notification(const CascadeNotification& notification) const;
  711. void send_schema_change_notification() const;
  712. static void get_version_and_history_info(const Array& top, _impl::History::version_type& version,
  713. int& history_type, int& history_schema_version) noexcept;
  714. static ref_type get_history_ref(const Array& top) noexcept;
  715. static size_t get_logical_file_size(const Array& top) noexcept;
  716. size_t get_logical_file_size() const noexcept
  717. {
  718. return get_logical_file_size(m_top);
  719. }
  720. void clear_history();
  721. void set_history_schema_version(int version);
  722. template <class Accessor>
  723. void set_history_parent(Accessor& history_root) noexcept;
  724. void prepare_top_for_history(int history_type, int history_schema_version, uint64_t file_ident);
  725. template <class Accessor>
  726. void prepare_history_parent(Accessor& history_root, int history_type, int history_schema_version,
  727. uint64_t file_ident);
  728. static void validate_top_array(const Array& arr, const SlabAlloc& alloc,
  729. std::optional<size_t> read_lock_file_size = util::none,
  730. std::optional<uint_fast64_t> read_lock_version = util::none);
  731. size_t find_table_index(StringData name) const noexcept;
  732. TableKey ndx2key(size_t ndx) const;
  733. size_t key2ndx(TableKey key) const;
  734. size_t key2ndx_checked(TableKey key) const;
  735. void set_size() const noexcept;
  736. std::map<TableRef, ColKey> get_primary_key_columns_from_pk_table(TableRef pk_table);
  737. void check_table_name_uniqueness(StringData name)
  738. {
  739. if (m_table_names.find_first(name) != not_found)
  740. throw TableNameInUse();
  741. }
  742. void check_attached() const
  743. {
  744. if (!is_attached())
  745. throw StaleAccessor("Stale transaction");
  746. }
  747. friend class Table;
  748. friend class GroupWriter;
  749. friend class DB;
  750. friend class _impl::GroupFriend;
  751. friend class _impl::TransactLogParser;
  752. friend class metrics::QueryInfo;
  753. friend class metrics::Metrics;
  754. friend class Transaction;
  755. friend class TableKeyIterator;
  756. friend class CascadeState;
  757. friend class SlabAlloc;
  758. };
  759. class TableKeyIterator {
  760. public:
  761. bool operator!=(const TableKeyIterator& other)
  762. {
  763. return m_pos != other.m_pos;
  764. }
  765. TableKeyIterator& operator++();
  766. TableKey operator*();
  767. private:
  768. friend class TableKeys;
  769. const Group* m_group;
  770. size_t m_pos;
  771. size_t m_index_in_group = 0;
  772. TableKey m_table_key;
  773. TableKeyIterator(const Group* g, size_t p)
  774. : m_group(g)
  775. , m_pos(p)
  776. {
  777. }
  778. void load_key();
  779. };
  780. class TableKeys {
  781. public:
  782. TableKeys(const Group* g)
  783. : m_iter(g, 0)
  784. {
  785. }
  786. size_t size() const
  787. {
  788. return m_iter.m_group->size();
  789. }
  790. bool empty() const
  791. {
  792. return size() == 0;
  793. }
  794. TableKey operator[](size_t p) const;
  795. TableKeyIterator begin() const
  796. {
  797. return TableKeyIterator(m_iter.m_group, 0);
  798. }
  799. TableKeyIterator end() const
  800. {
  801. return TableKeyIterator(m_iter.m_group, size());
  802. }
  803. private:
  804. mutable TableKeyIterator m_iter;
  805. };
  806. // Implementation
  807. inline TableKeys Group::get_table_keys() const
  808. {
  809. return TableKeys(this);
  810. }
  811. inline bool Group::is_attached() const noexcept
  812. {
  813. return m_attached;
  814. }
  815. inline bool Group::is_empty() const noexcept
  816. {
  817. if (!is_attached())
  818. return false;
  819. return size() == 0;
  820. }
  821. inline size_t Group::key2ndx(TableKey key) const
  822. {
  823. size_t idx = key.value & 0xFFFF;
  824. return idx;
  825. }
  826. inline StringData Group::get_table_name(TableKey key) const
  827. {
  828. size_t table_ndx = key2ndx_checked(key);
  829. return m_table_names.get(table_ndx);
  830. }
  831. inline bool Group::table_is_public(TableKey key) const
  832. {
  833. return get_table_name(key).begins_with(g_class_name_prefix);
  834. }
  835. inline bool Group::has_table(StringData name) const noexcept
  836. {
  837. size_t ndx = find_table_index(name);
  838. return ndx != not_found;
  839. }
  840. inline size_t Group::find_table_index(StringData name) const noexcept
  841. {
  842. if (m_table_names.is_attached())
  843. return m_table_names.find_first(name);
  844. return not_found;
  845. }
  846. inline TableKey Group::find_table(StringData name) const noexcept
  847. {
  848. if (!is_attached())
  849. return TableKey();
  850. size_t ndx = find_table_index(name);
  851. return (ndx != npos) ? ndx2key(ndx) : TableKey{};
  852. }
  853. inline TableRef Group::get_table(TableKey key)
  854. {
  855. check_attached();
  856. auto ndx = key2ndx_checked(key);
  857. Table* table = do_get_table(ndx); // Throws
  858. return TableRef(table, table ? table->m_alloc.get_instance_version() : 0);
  859. }
  860. inline ConstTableRef Group::get_table(TableKey key) const
  861. {
  862. check_attached();
  863. auto ndx = key2ndx_checked(key);
  864. const Table* table = do_get_table(ndx); // Throws
  865. return ConstTableRef(table, table ? table->m_alloc.get_instance_version() : 0);
  866. }
  867. inline TableRef Group::get_table(StringData name)
  868. {
  869. check_attached();
  870. Table* table = do_get_table(name); // Throws
  871. return TableRef(table, table ? table->m_alloc.get_instance_version() : 0);
  872. }
  873. inline ConstTableRef Group::get_table(StringData name) const
  874. {
  875. check_attached();
  876. const Table* table = do_get_table(name); // Throws
  877. return ConstTableRef(table, table ? table->m_alloc.get_instance_version() : 0);
  878. }
  879. inline TableRef Group::add_table(StringData name, Table::Type table_type)
  880. {
  881. check_attached();
  882. check_table_name_uniqueness(name);
  883. Table* table = do_add_table(name, table_type); // Throws
  884. return TableRef(table, table->m_alloc.get_instance_version());
  885. }
  886. inline TableRef Group::get_or_add_table(StringData name, Table::Type table_type, bool* was_added)
  887. {
  888. REALM_ASSERT(table_type != Table::Type::Embedded);
  889. check_attached();
  890. auto table = do_get_table(name);
  891. if (was_added)
  892. *was_added = !table;
  893. if (!table) {
  894. table = do_add_table(name, table_type);
  895. }
  896. return TableRef(table, table->m_alloc.get_instance_version());
  897. }
  898. inline TableRef Group::get_or_add_table_with_primary_key(StringData name, DataType pk_type, StringData pk_name,
  899. bool nullable, Table::Type table_type)
  900. {
  901. REALM_ASSERT(table_type != Table::Type::Embedded);
  902. if (TableRef table = get_table(name)) {
  903. if (!table->get_primary_key_column() || table->get_column_name(table->get_primary_key_column()) != pk_name ||
  904. table->is_nullable(table->get_primary_key_column()) != nullable ||
  905. table->get_table_type() != table_type) {
  906. return {};
  907. }
  908. return table;
  909. }
  910. else {
  911. return add_table_with_primary_key(name, pk_type, pk_name, nullable, table_type);
  912. }
  913. }
  914. inline void Group::init_array_parents() noexcept
  915. {
  916. m_table_names.set_parent(&m_top, 0);
  917. m_tables.set_parent(&m_top, 1);
  918. }
  919. inline void Group::update_child_ref(size_t child_ndx, ref_type new_ref)
  920. {
  921. m_tables.set(child_ndx, new_ref);
  922. }
  923. inline ref_type Group::get_child_ref(size_t child_ndx) const noexcept
  924. {
  925. return m_tables.get_as_ref(child_ndx);
  926. }
  927. inline bool Group::has_cascade_notification_handler() const noexcept
  928. {
  929. return !!m_notify_handler;
  930. }
  931. inline void
  932. Group::set_cascade_notification_handler(util::UniqueFunction<void(const CascadeNotification&)> new_handler) noexcept
  933. {
  934. m_notify_handler = std::move(new_handler);
  935. }
  936. inline void Group::send_cascade_notification(const CascadeNotification& notification) const
  937. {
  938. REALM_ASSERT_DEBUG(m_notify_handler);
  939. m_notify_handler(notification);
  940. }
  941. inline bool Group::has_schema_change_notification_handler() const noexcept
  942. {
  943. return !!m_schema_change_handler;
  944. }
  945. inline void Group::set_schema_change_notification_handler(util::UniqueFunction<void()> new_handler) noexcept
  946. {
  947. m_schema_change_handler = std::move(new_handler);
  948. }
  949. inline void Group::send_schema_change_notification() const
  950. {
  951. if (m_schema_change_handler)
  952. m_schema_change_handler();
  953. }
  954. inline ref_type Group::get_history_ref(const Array& top) noexcept
  955. {
  956. bool has_history = (top.is_attached() && top.size() > s_hist_type_ndx);
  957. if (has_history) {
  958. // This function is only used is shared mode (from DB)
  959. REALM_ASSERT(top.size() > s_hist_version_ndx);
  960. return top.get_as_ref(s_hist_ref_ndx);
  961. }
  962. return 0;
  963. }
  964. inline size_t Group::get_logical_file_size(const Array& top) noexcept
  965. {
  966. if (top.is_attached() && top.size() > s_file_size_ndx) {
  967. return (size_t)top.get_as_ref_or_tagged(s_file_size_ndx).get_as_int();
  968. }
  969. return 0;
  970. }
  971. inline void Group::set_sync_file_id(uint64_t id)
  972. {
  973. while (m_top.size() < s_sync_file_id_ndx + 1)
  974. m_top.add(0);
  975. m_top.set(s_sync_file_id_ndx, RefOrTagged::make_tagged(id));
  976. }
  977. inline void Group::set_history_schema_version(int version)
  978. {
  979. while (m_top.size() < s_hist_version_ndx + 1)
  980. m_top.add(0);
  981. m_top.set(s_hist_version_ndx, RefOrTagged::make_tagged(unsigned(version))); // Throws
  982. }
  983. template <class Accessor>
  984. inline void Group::set_history_parent(Accessor& history_root) noexcept
  985. {
  986. history_root.set_parent(&m_top, 8);
  987. }
  988. template <class Accessor>
  989. void Group::prepare_history_parent(Accessor& history_root, int history_type, int history_schema_version,
  990. uint64_t file_ident)
  991. {
  992. prepare_top_for_history(history_type, history_schema_version, file_ident);
  993. set_history_parent(history_root);
  994. }
  995. class Group::TableWriter {
  996. public:
  997. struct HistoryInfo {
  998. ref_type ref = 0;
  999. int type = 0;
  1000. int version = 0;
  1001. uint64_t sync_file_id = 0;
  1002. };
  1003. virtual ref_type write_names(_impl::OutputStream&) = 0;
  1004. virtual ref_type write_tables(_impl::OutputStream&) = 0;
  1005. virtual HistoryInfo write_history(_impl::OutputStream&) = 0;
  1006. virtual ~TableWriter() noexcept {}
  1007. void set_group(const Group* g)
  1008. {
  1009. m_group = g;
  1010. }
  1011. protected:
  1012. const Group* m_group = nullptr;
  1013. };
  1014. class Group::DefaultTableWriter : public Group::TableWriter {
  1015. public:
  1016. DefaultTableWriter(bool should_write_history = true)
  1017. : m_should_write_history(should_write_history)
  1018. {
  1019. }
  1020. ref_type write_names(_impl::OutputStream& out) override;
  1021. ref_type write_tables(_impl::OutputStream& out) override;
  1022. HistoryInfo write_history(_impl::OutputStream& out) override;
  1023. private:
  1024. bool m_should_write_history;
  1025. };
  1026. inline const Table* Group::do_get_table(size_t ndx) const
  1027. {
  1028. return const_cast<Group*>(this)->do_get_table(ndx); // Throws
  1029. }
  1030. inline const Table* Group::do_get_table(StringData name) const
  1031. {
  1032. return const_cast<Group*>(this)->do_get_table(name); // Throws
  1033. }
  1034. inline void Group::reset_free_space_tracking()
  1035. {
  1036. // if used whith a shared allocator, free space should never be reset through
  1037. // Group, but rather through the proper owner of the allocator, which is the DB object.
  1038. REALM_ASSERT(m_local_alloc);
  1039. m_alloc.reset_free_space_tracking(); // Throws
  1040. }
  1041. inline std::shared_ptr<metrics::Metrics> Group::get_metrics() const noexcept
  1042. {
  1043. return m_metrics;
  1044. }
  1045. inline void Group::set_metrics(std::shared_ptr<metrics::Metrics> shared) noexcept
  1046. {
  1047. m_metrics = shared;
  1048. }
  1049. // The purpose of this class is to give internal access to some, but
  1050. // not all of the non-public parts of the Group class.
  1051. class _impl::GroupFriend {
  1052. public:
  1053. static Allocator& get_alloc(const Group& group) noexcept
  1054. {
  1055. return group.m_alloc;
  1056. }
  1057. static ref_type get_top_ref(const Group& group) noexcept
  1058. {
  1059. return group.m_top.get_ref();
  1060. }
  1061. static ref_type get_history_ref(Allocator& alloc, ref_type top_ref) noexcept
  1062. {
  1063. Array top(alloc);
  1064. if (top_ref != 0)
  1065. top.init_from_ref(top_ref);
  1066. return Group::get_history_ref(top);
  1067. }
  1068. static ref_type get_history_ref(const Group& group) noexcept
  1069. {
  1070. return Group::get_history_ref(group.m_top);
  1071. }
  1072. static int get_file_format_version(const Group& group) noexcept
  1073. {
  1074. return group.get_file_format_version();
  1075. }
  1076. static void get_version_and_history_info(const Allocator& alloc, ref_type top_ref,
  1077. _impl::History::version_type& version, int& history_type,
  1078. int& history_schema_version) noexcept
  1079. {
  1080. Array top{const_cast<Allocator&>(alloc)};
  1081. if (top_ref != 0)
  1082. top.init_from_ref(top_ref);
  1083. Group::get_version_and_history_info(top, version, history_type, history_schema_version);
  1084. }
  1085. static void set_history_schema_version(Group& group, int version)
  1086. {
  1087. group.set_history_schema_version(version); // Throws
  1088. }
  1089. template <class Accessor>
  1090. static void set_history_parent(Group& group, Accessor& history_root) noexcept
  1091. {
  1092. group.set_history_parent(history_root);
  1093. }
  1094. template <class Accessor>
  1095. static void prepare_history_parent(Group& group, Accessor& history_root, int history_type,
  1096. int history_schema_version, uint64_t file_ident = 0)
  1097. {
  1098. group.prepare_history_parent(history_root, history_type, history_schema_version, file_ident); // Throws
  1099. }
  1100. // This is used by upgrade functions in Sync
  1101. static Table* get_table_by_ndx(Group& group, size_t ndx)
  1102. {
  1103. return group.do_get_table(ndx);
  1104. }
  1105. static int get_target_file_format_version_for_session(int current_file_format_version, int history_type) noexcept
  1106. {
  1107. return Group::get_target_file_format_version_for_session(current_file_format_version, history_type);
  1108. }
  1109. static void fake_target_file_format(const std::optional<int> format) noexcept;
  1110. };
  1111. class CascadeState {
  1112. public:
  1113. enum class Mode {
  1114. /// If we remove the last link to an object, delete that object, even if
  1115. /// the link we removed was not a strong link
  1116. All,
  1117. /// If we remove the last link to an object, delete that object only if
  1118. /// the link we removed was a strong link
  1119. Strong,
  1120. /// Never delete objects due to removing links
  1121. None
  1122. };
  1123. struct Link {
  1124. TableKey origin_table; ///< A group-level table.
  1125. ColKey origin_col_key; ///< Link column being nullified.
  1126. ObjKey origin_key; ///< Row in column being nullified.
  1127. /// The target row index which is being removed. Mostly relevant for
  1128. /// LinkList (to know which entries are being removed), but also
  1129. /// valid for Link.
  1130. ObjLink old_target_link;
  1131. };
  1132. CascadeState(Mode mode = Mode::Strong, Group* g = nullptr) noexcept
  1133. : m_mode(mode)
  1134. , m_group(g)
  1135. {
  1136. }
  1137. /// Indicate which links to take action on. Either all, strong or none.
  1138. Mode m_mode;
  1139. std::vector<std::pair<TableKey, ObjKey>> m_to_be_deleted;
  1140. std::vector<Link> m_to_be_nullified;
  1141. Group* m_group = nullptr;
  1142. bool notification_handler() const noexcept
  1143. {
  1144. return m_group && m_group->has_cascade_notification_handler();
  1145. }
  1146. void send_notifications(Group::CascadeNotification& notifications) const
  1147. {
  1148. REALM_ASSERT_DEBUG(notification_handler());
  1149. m_group->send_cascade_notification(notifications);
  1150. }
  1151. bool enqueue_for_cascade(const Obj& target_obj, bool link_is_strong, bool last_removed)
  1152. {
  1153. // Check if the object should be cascade deleted
  1154. if (m_mode == Mode::None || !last_removed) {
  1155. return false;
  1156. }
  1157. if (m_mode == Mode::All || link_is_strong) {
  1158. bool has_backlinks = target_obj.has_backlinks(m_mode == Mode::Strong);
  1159. if (!has_backlinks) {
  1160. // Object has no more backlinks - add to list for deletion
  1161. m_to_be_deleted.emplace_back(target_obj.get_table()->get_key(), target_obj.get_key());
  1162. return true;
  1163. }
  1164. }
  1165. return false;
  1166. }
  1167. void enqueue_for_nullification(Table& src_table, ColKey src_col_key, ObjKey origin_key, ObjLink target_link)
  1168. {
  1169. // Nullify immediately if we don't need to send cascade notifications
  1170. if (!notification_handler()) {
  1171. src_table.get_object(origin_key).nullify_link(src_col_key, target_link);
  1172. return;
  1173. }
  1174. // Otherwise enqueue it
  1175. m_to_be_nullified.push_back({src_table.get_key(), src_col_key, origin_key, target_link});
  1176. }
  1177. void send_notifications()
  1178. {
  1179. if (!notification_handler()) {
  1180. return;
  1181. }
  1182. Group::CascadeNotification notification;
  1183. for (auto& o : m_to_be_deleted)
  1184. notification.rows.emplace_back(o.first, o.second);
  1185. for (auto& l : m_to_be_nullified)
  1186. notification.links.emplace_back(l.origin_table, l.origin_col_key, l.origin_key,
  1187. l.old_target_link.get_obj_key());
  1188. send_notifications(notification);
  1189. }
  1190. };
  1191. } // namespace realm
  1192. #endif // REALM_GROUP_HPP