group.hpp 52 KB

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