replication.hpp 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610
  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_REPLICATION_HPP
  19. #define REALM_REPLICATION_HPP
  20. #include <algorithm>
  21. #include <limits>
  22. #include <memory>
  23. #include <exception>
  24. #include <string>
  25. #include <realm/util/assert.hpp>
  26. #include <realm/util/safe_int_ops.hpp>
  27. #include <realm/util/buffer.hpp>
  28. #include <realm/impl/cont_transact_hist.hpp>
  29. #include <realm/impl/transact_log.hpp>
  30. namespace realm {
  31. namespace util {
  32. class Logger;
  33. }
  34. // FIXME: Be careful about the possibility of one modification function being called by another where both do
  35. // transaction logging.
  36. /// Replication is enabled by passing an instance of an implementation of this
  37. /// class to the DB constructor.
  38. class Replication {
  39. public:
  40. virtual ~Replication() = default;
  41. // Formerly Replication:
  42. virtual void add_class(TableKey table_key, StringData table_name, Table::Type table_type);
  43. virtual void add_class_with_primary_key(TableKey, StringData table_name, DataType pk_type, StringData pk_field,
  44. bool nullable, Table::Type table_type);
  45. virtual void prepare_erase_class(TableKey table_key);
  46. virtual void erase_class(TableKey table_key, size_t num_tables);
  47. virtual void rename_class(TableKey table_key, StringData new_name);
  48. virtual void insert_column(const Table*, ColKey col_key, DataType type, StringData name, Table* target_table);
  49. virtual void erase_column(const Table*, ColKey col_key);
  50. virtual void rename_column(const Table*, ColKey col_key, StringData name);
  51. virtual void add_int(const Table*, ColKey col_key, ObjKey key, int_fast64_t value);
  52. virtual void set(const Table*, ColKey col_key, ObjKey key, Mixed value,
  53. _impl::Instruction variant = _impl::instr_Set);
  54. virtual void list_set(const CollectionBase& list, size_t list_ndx, Mixed value);
  55. virtual void list_insert(const CollectionBase& list, size_t list_ndx, Mixed value, size_t prior_size);
  56. virtual void list_move(const CollectionBase&, size_t from_link_ndx, size_t to_link_ndx);
  57. virtual void list_erase(const CollectionBase&, size_t link_ndx);
  58. virtual void list_clear(const CollectionBase&);
  59. virtual void set_insert(const CollectionBase& set, size_t list_ndx, Mixed value);
  60. virtual void set_erase(const CollectionBase& set, size_t list_ndx, Mixed value);
  61. virtual void set_clear(const CollectionBase& set);
  62. virtual void dictionary_insert(const CollectionBase& dict, size_t dict_ndx, Mixed key, Mixed value);
  63. virtual void dictionary_set(const CollectionBase& dict, size_t dict_ndx, Mixed key, Mixed value);
  64. virtual void dictionary_erase(const CollectionBase& dict, size_t dict_ndx, Mixed key);
  65. virtual void create_object(const Table*, GlobalKey);
  66. virtual void create_object_with_primary_key(const Table*, ObjKey, Mixed);
  67. virtual void remove_object(const Table*, ObjKey);
  68. virtual void typed_link_change(const Table*, ColKey, TableKey);
  69. //@{
  70. /// Implicit nullifications due to removal of target row. This is redundant
  71. /// information from the point of view of replication, as the removal of the
  72. /// target row will reproduce the implicit nullifications in the target
  73. /// Realm anyway. The purpose of this instruction is to allow observers
  74. /// (reactor pattern) to be explicitly notified about the implicit
  75. /// nullifications.
  76. virtual void nullify_link(const Table*, ColKey col_key, ObjKey key);
  77. virtual void link_list_nullify(const Lst<ObjKey>&, size_t link_ndx);
  78. // Be sure to keep this type aligned with what is actually used in DB.
  79. using version_type = _impl::History::version_type;
  80. using InputStream = util::NoCopyInputStream;
  81. class TransactLogApplier;
  82. class Interrupted; // Exception
  83. class SimpleIndexTranslator;
  84. std::string get_database_path() const;
  85. /// Called during construction of the associated DB object.
  86. ///
  87. /// \param db The associated DB object.
  88. virtual void initialize(DB& db);
  89. /// \defgroup replication_transactions
  90. //@{
  91. /// From the point of view of the Replication class, a transaction is
  92. /// initiated when, and only when the associated Transaction object calls
  93. /// initiate_transact() and the call is successful. The associated
  94. /// Transaction object must terminate every initiated transaction either by
  95. /// calling finalize_commit() or by calling abort_transact(). It may only
  96. /// call finalize_commit(), however, after calling prepare_commit(), and
  97. /// only when prepare_commit() succeeds. If prepare_commit() fails (i.e.,
  98. /// throws) abort_transact() must still be called.
  99. ///
  100. /// The associated Transaction object is supposed to terminate a transaction
  101. /// as soon as possible, and is required to terminate it before attempting
  102. /// to initiate a new one.
  103. ///
  104. /// initiate_transact() is called by the associated Transaction object as
  105. /// part of the initiation of a transaction, and at a time where the caller
  106. /// has acquired exclusive write access to the local Realm. The Replication
  107. /// implementation is allowed to perform "precursor transactions" on the
  108. /// local Realm at this time. During the initiated transaction, the
  109. /// associated DB object must inform the Replication object of all
  110. /// modifying operations by calling set_value() and friends.
  111. ///
  112. /// FIXME: There is currently no way for implementations to perform
  113. /// precursor transactions, since a regular transaction would cause a dead
  114. /// lock when it tries to acquire a write lock. Consider giving access to
  115. /// special non-locking precursor transactions via an extra argument to this
  116. /// function.
  117. ///
  118. /// prepare_commit() serves as the first phase of a two-phase commit. This
  119. /// function is called by the associated Transaction object immediately
  120. /// before the commit operation on the local Realm. The associated
  121. /// Transaction object will then, as the second phase, either call
  122. /// finalize_commit() or abort_transact() depending on whether the commit
  123. /// operation succeeded or not. The Replication implementation is allowed to
  124. /// modify the Realm via the associated Transaction object at this time
  125. /// (important to in-Realm histories).
  126. ///
  127. /// initiate_transact() and prepare_commit() are allowed to block the
  128. /// calling thread if, for example, they need to communicate over the
  129. /// network. If a calling thread is blocked in one of these functions, it
  130. /// must be possible to interrupt the blocking operation by having another
  131. /// thread call interrupt(). The contract is as follows: When interrupt() is
  132. /// called, then any execution of initiate_transact() or prepare_commit(),
  133. /// initiated before the interruption, must complete without blocking, or
  134. /// the execution must be aborted by throwing an Interrupted exception. If
  135. /// initiate_transact() or prepare_commit() throws Interrupted, it counts as
  136. /// a failed operation.
  137. ///
  138. /// finalize_commit() is called by the associated Transaction object
  139. /// immediately after a successful commit operation on the local Realm. This
  140. /// happens at a time where modification of the Realm is no longer possible
  141. /// via the associated Transaction object. In the case of in-Realm
  142. /// histories, the changes are automatically finalized as part of the commit
  143. /// operation performed by the caller prior to the invocation of
  144. /// finalize_commit(), so in that case, finalize_commit() might not need to
  145. /// do anything.
  146. ///
  147. /// abort_transact() is called by the associated Transaction object to
  148. /// terminate a transaction without committing. That is, any transaction
  149. /// that is not terminated by finalize_commit() is terminated by
  150. /// abort_transact(). This could be due to an explicit rollback, or due to a
  151. /// failed commit attempt.
  152. ///
  153. /// Note that finalize_commit() and abort_transact() are not allowed to
  154. /// throw.
  155. ///
  156. /// \param current_version The version of the snapshot that the current
  157. /// transaction is based on.
  158. ///
  159. /// \param history_updated Pass true only when the history has already been
  160. /// updated to reflect the currently bound snapshot, such as when
  161. /// _impl::History::update_early_from_top_ref() was called during the
  162. /// transition from a read transaction to the current write transaction.
  163. ///
  164. /// \throw Interrupted Thrown by initiate_transact() and prepare_commit() if
  165. /// a blocking operation was interrupted.
  166. void initiate_transact(Group& group, version_type current_version, bool history_updated);
  167. /// \param current_version The version of the snapshot that the current
  168. /// transaction is based on.
  169. /// \return prepare_commit() returns the version of the new snapshot
  170. /// produced by the transaction.
  171. version_type prepare_commit(version_type current_version);
  172. void finalize_commit() noexcept;
  173. //@}
  174. /// Get the list of uncommitted changes accumulated so far in the current
  175. /// write transaction.
  176. ///
  177. /// The callee retains ownership of the referenced memory. The ownership is
  178. /// not handed over to the caller.
  179. ///
  180. /// This function may be called only during a write transaction (prior to
  181. /// initiation of commit operation). In that case, the caller may assume that the
  182. /// returned memory reference stays valid for the remainder of the transaction (up
  183. /// until initiation of the commit operation).
  184. BinaryData get_uncommitted_changes() const noexcept;
  185. /// CAUTION: These values are stored in Realm files, so value reassignment
  186. /// is not allowed.
  187. enum HistoryType {
  188. /// No history available. No support for either continuous transactions
  189. /// or inter-client synchronization.
  190. hist_None = 0,
  191. /// Out-of-Realm history supporting continuous transactions.
  192. ///
  193. /// NOTE: This history type is no longer in use. The value needs to stay
  194. /// reserved in case someone tries to open an old Realm file.
  195. hist_OutOfRealm = 1,
  196. /// In-Realm history supporting continuous transactions
  197. /// (make_in_realm_history()).
  198. hist_InRealm = 2,
  199. /// In-Realm history supporting continuous transactions and client-side
  200. /// synchronization protocol (realm::sync::ClientHistory).
  201. hist_SyncClient = 3,
  202. /// In-Realm history supporting continuous transactions and server-side
  203. /// synchronization protocol (realm::_impl::ServerHistory).
  204. hist_SyncServer = 4
  205. };
  206. static const char* history_type_name(int);
  207. /// Returns the type of history maintained by this Replication
  208. /// implementation, or \ref hist_None if no history is maintained by it.
  209. ///
  210. /// This type is used to ensure that all session participants agree on
  211. /// history type, and that the Realm file contains a compatible type of
  212. /// history, at the beginning of a new session.
  213. ///
  214. /// As a special case, if there is no top array (Group::m_top) at the
  215. /// beginning of a new session, then the history type is still undecided and
  216. /// all history types (as returned by get_history_type()) are threfore
  217. /// allowed for the session initiator. Note that this case only arises if
  218. /// there was no preceding session, or if no transaction was sucessfully
  219. /// committed during any of the preceding sessions. As soon as a transaction
  220. /// is successfully committed, the Realm contains at least a top array, and
  221. /// from that point on, the history type is generally fixed, although still
  222. /// subject to certain allowed changes (as mentioned below).
  223. ///
  224. /// For the sake of backwards compatibility with older Realm files that does
  225. /// not store any history type, the following rule shall apply:
  226. ///
  227. /// - If the top array of a Realm file (Group::m_top) does not contain a
  228. /// history type, because it is too short, it shall be understood as
  229. /// implicitly storing the type \ref hist_None.
  230. ///
  231. /// Note: In what follows, the meaning of *preceding session* is: The last
  232. /// preceding session that modified the Realm by sucessfully committing a
  233. /// new snapshot.
  234. ///
  235. /// It shall be allowed to switch to a \ref hist_InRealm history if the
  236. /// stored history type is \ref hist_None. This can be done simply by adding
  237. /// a new history to the Realm file. This is possible because histories of
  238. /// this type a transient in nature, and need not survive from one session
  239. /// to the next.
  240. ///
  241. /// On the other hand, as soon as a history of type \ref hist_InRealm is
  242. /// added to a Realm file, that history type is binding for all subsequent
  243. /// sessions. In theory, this constraint is not necessary, and a later
  244. /// switch to \ref hist_None would be possible because of the transient
  245. /// nature of it, however, because the \ref hist_InRealm history remains in
  246. /// the Realm file, there are practical complications, and for that reason,
  247. /// such switching shall not be supported.
  248. ///
  249. /// The \ref hist_SyncClient history type can only be used if the stored
  250. /// history type is also \ref hist_SyncClient, or when there is no top array
  251. /// yet. Likewise, the \ref hist_SyncServer history type can only be used if
  252. /// the stored history type is also \ref hist_SyncServer, or when there is
  253. /// no top array yet. Additionally, when the stored history type is \ref
  254. /// hist_SyncClient or \ref hist_SyncServer, then all subsequent sessions
  255. /// must have the same type. These restrictions apply because such a history
  256. /// needs to be maintained persistently across sessions.
  257. ///
  258. /// In general, if there is no stored history type (no top array) at the
  259. /// beginning of a new session, or if the stored type disagrees with what is
  260. /// returned by get_history_type() (which is possible due to particular
  261. /// allowed changes of history type), the actual history type (as returned
  262. /// by get_history_type()) used during that session, must be stored in the
  263. /// Realm during the first successfully committed transaction in that
  264. /// session. But note that there is still no need to expand the top array to
  265. /// store the history type \ref hist_None, due to the rule mentioned above.
  266. ///
  267. /// This function must return \ref hist_None when, and only when
  268. /// get_history() returns null.
  269. virtual HistoryType get_history_type() const noexcept
  270. {
  271. return HistoryType::hist_None;
  272. }
  273. /// Returns the schema version of the history maintained by this Replication
  274. /// implementation, or 0 if no history is maintained by it. All session
  275. /// participants must agree on history schema version.
  276. ///
  277. /// Must return 0 if get_history_type() returns \ref hist_None.
  278. virtual int get_history_schema_version() const noexcept
  279. {
  280. return 0;
  281. }
  282. /// Implementation may assume that this function is only ever called with a
  283. /// stored schema version that is less than what was returned by
  284. /// get_history_schema_version().
  285. virtual bool is_upgradable_history_schema(int /* stored_schema_version */) const noexcept
  286. {
  287. return false;
  288. }
  289. /// The implementation may assume that this function is only ever called if
  290. /// is_upgradable_history_schema() was called with the same stored schema
  291. /// version, and returned true. This implies that the specified stored
  292. /// schema version is always strictly less than what was returned by
  293. /// get_history_schema_version().
  294. virtual void upgrade_history_schema(int /* stored_schema_version */) {}
  295. /// Returns an object that gives access to the history of changesets
  296. /// used by writers. All writers can share the same object as all write
  297. /// transactions are serialized.
  298. ///
  299. /// This function must return null when, and only when get_history_type()
  300. /// returns \ref hist_None.
  301. virtual _impl::History* _get_history_write()
  302. {
  303. return nullptr;
  304. }
  305. /// Returns an object that gives access to the history of changesets in a
  306. /// way that allows for continuous transactions to work. All readers must
  307. /// get their own exclusive object as readers are not blocking each other.
  308. /// (Group::advance_transact() in particular).
  309. ///
  310. /// This function must return null when, and only when get_history_type()
  311. /// returns \ref hist_None.
  312. virtual std::unique_ptr<_impl::History> _create_history_read()
  313. {
  314. return nullptr;
  315. }
  316. void set_logger(util::Logger* logger)
  317. {
  318. m_logger = logger;
  319. }
  320. util::Logger* get_logger() const noexcept
  321. {
  322. return m_logger;
  323. }
  324. protected:
  325. Replication() = default;
  326. //@{
  327. /// do_initiate_transact() is called by initiate_transact(), and likewise
  328. /// for do_prepare_commit()
  329. ///
  330. /// With respect to exception safety, the Replication implementation has two
  331. /// options: It can prepare to accept the accumulated changeset in
  332. /// do_prepapre_commit() by allocating all required resources, and delay the
  333. /// actual acceptance to finalize_commit(), which requires that the final
  334. /// acceptance can be done without any risk of failure. Alternatively, the
  335. /// Replication implementation can fully accept the changeset in
  336. /// do_prepapre_commit() (allowing for failure), and then discard that
  337. /// changeset during the next invocation of do_initiate_transact() if
  338. /// `current_version` indicates that the previous transaction failed.
  339. virtual void do_initiate_transact(Group& group, version_type current_version, bool history_updated);
  340. //@}
  341. // Formerly part of TrivialReplication:
  342. virtual version_type prepare_changeset(const char*, size_t, version_type orig_version)
  343. {
  344. return orig_version + 1;
  345. }
  346. virtual void finalize_changeset() noexcept {}
  347. private:
  348. struct CollectionId {
  349. TableKey table_key;
  350. ObjKey object_key;
  351. ColKey col_id;
  352. CollectionId() = default;
  353. CollectionId(const CollectionBase& list)
  354. : table_key(list.get_table()->get_key())
  355. , object_key(list.get_owner_key())
  356. , col_id(list.get_col_key())
  357. {
  358. }
  359. CollectionId(TableKey t, ObjKey k, ColKey c)
  360. : table_key(t)
  361. , object_key(k)
  362. , col_id(c)
  363. {
  364. }
  365. bool operator!=(const CollectionId& other)
  366. {
  367. return object_key != other.object_key || table_key != other.table_key || col_id != other.col_id;
  368. }
  369. };
  370. _impl::TransactLogBufferStream m_stream;
  371. _impl::TransactLogEncoder m_encoder{m_stream};
  372. util::Logger* m_logger = nullptr;
  373. mutable const Table* m_selected_table = nullptr;
  374. mutable CollectionId m_selected_list;
  375. void unselect_all() noexcept;
  376. void select_table(const Table*); // unselects link list
  377. void select_collection(const CollectionBase&);
  378. void do_select_table(const Table*);
  379. void do_select_collection(const CollectionBase&);
  380. void do_set(const Table*, ColKey col_key, ObjKey key, _impl::Instruction variant = _impl::instr_Set);
  381. size_t transact_log_size();
  382. };
  383. class Replication::Interrupted : public std::exception {
  384. public:
  385. const char* what() const noexcept override
  386. {
  387. return "Interrupted";
  388. }
  389. };
  390. // Implementation:
  391. inline void Replication::initiate_transact(Group& group, version_type current_version, bool history_updated)
  392. {
  393. if (auto hist = _get_history_write()) {
  394. hist->set_group(&group, history_updated);
  395. }
  396. do_initiate_transact(group, current_version, history_updated);
  397. unselect_all();
  398. }
  399. inline void Replication::finalize_commit() noexcept
  400. {
  401. finalize_changeset();
  402. }
  403. inline BinaryData Replication::get_uncommitted_changes() const noexcept
  404. {
  405. const char* data = m_stream.get_data();
  406. size_t size = m_encoder.write_position() - data;
  407. return BinaryData(data, size);
  408. }
  409. inline size_t Replication::transact_log_size()
  410. {
  411. return m_encoder.write_position() - m_stream.get_data();
  412. }
  413. inline void Replication::unselect_all() noexcept
  414. {
  415. m_selected_table = nullptr;
  416. m_selected_list = CollectionId();
  417. }
  418. inline void Replication::select_table(const Table* table)
  419. {
  420. if (table != m_selected_table)
  421. do_select_table(table); // Throws
  422. m_selected_list = CollectionId();
  423. }
  424. inline void Replication::select_collection(const CollectionBase& list)
  425. {
  426. if (CollectionId(list) != m_selected_list) {
  427. do_select_collection(list); // Throws
  428. }
  429. }
  430. inline void Replication::prepare_erase_class(TableKey) {}
  431. inline void Replication::erase_class(TableKey table_key, size_t)
  432. {
  433. unselect_all();
  434. m_encoder.erase_class(table_key); // Throws
  435. }
  436. inline void Replication::rename_class(TableKey table_key, StringData)
  437. {
  438. unselect_all();
  439. m_encoder.rename_class(table_key); // Throws
  440. }
  441. inline void Replication::insert_column(const Table* t, ColKey col_key, DataType, StringData, Table*)
  442. {
  443. select_table(t); // Throws
  444. m_encoder.insert_column(col_key); // Throws
  445. }
  446. inline void Replication::erase_column(const Table* t, ColKey col_key)
  447. {
  448. select_table(t); // Throws
  449. m_encoder.erase_column(col_key); // Throws
  450. }
  451. inline void Replication::rename_column(const Table* t, ColKey col_key, StringData)
  452. {
  453. select_table(t); // Throws
  454. m_encoder.rename_column(col_key); // Throws
  455. }
  456. inline void Replication::do_set(const Table* t, ColKey col_key, ObjKey key, _impl::Instruction variant)
  457. {
  458. if (variant != _impl::Instruction::instr_SetDefault) {
  459. select_table(t); // Throws
  460. m_encoder.modify_object(col_key, key); // Throws
  461. }
  462. }
  463. inline void Replication::set(const Table* t, ColKey col_key, ObjKey key, Mixed, _impl::Instruction variant)
  464. {
  465. do_set(t, col_key, key, variant); // Throws
  466. }
  467. inline void Replication::add_int(const Table* t, ColKey col_key, ObjKey key, int_fast64_t)
  468. {
  469. do_set(t, col_key, key); // Throws
  470. }
  471. inline void Replication::nullify_link(const Table* t, ColKey col_key, ObjKey key)
  472. {
  473. select_table(t); // Throws
  474. m_encoder.modify_object(col_key, key); // Throws
  475. }
  476. inline void Replication::list_set(const CollectionBase& list, size_t list_ndx, Mixed)
  477. {
  478. select_collection(list); // Throws
  479. m_encoder.list_set(list.translate_index(list_ndx)); // Throws
  480. }
  481. inline void Replication::list_insert(const CollectionBase& list, size_t list_ndx, Mixed, size_t)
  482. {
  483. select_collection(list); // Throws
  484. m_encoder.list_insert(list.translate_index(list_ndx)); // Throws
  485. }
  486. inline void Replication::set_insert(const CollectionBase& set, size_t set_ndx, Mixed)
  487. {
  488. select_collection(set); // Throws
  489. m_encoder.set_insert(set_ndx); // Throws
  490. }
  491. inline void Replication::set_erase(const CollectionBase& set, size_t set_ndx, Mixed)
  492. {
  493. select_collection(set); // Throws
  494. m_encoder.set_erase(set_ndx); // Throws
  495. }
  496. inline void Replication::set_clear(const CollectionBase& set)
  497. {
  498. select_collection(set); // Throws
  499. m_encoder.set_clear(set.size()); // Throws
  500. }
  501. inline void Replication::remove_object(const Table* t, ObjKey key)
  502. {
  503. select_table(t); // Throws
  504. m_encoder.remove_object(key); // Throws
  505. }
  506. inline void Replication::list_move(const CollectionBase& list, size_t from_link_ndx, size_t to_link_ndx)
  507. {
  508. select_collection(list); // Throws
  509. m_encoder.list_move(list.translate_index(from_link_ndx), list.translate_index(to_link_ndx)); // Throws
  510. }
  511. inline void Replication::list_erase(const CollectionBase& list, size_t link_ndx)
  512. {
  513. select_collection(list); // Throws
  514. m_encoder.list_erase(list.translate_index(link_ndx)); // Throws
  515. }
  516. inline void Replication::typed_link_change(const Table* source_table, ColKey col, TableKey dest_table)
  517. {
  518. select_table(source_table);
  519. m_encoder.typed_link_change(col, dest_table);
  520. }
  521. } // namespace realm
  522. #endif // REALM_REPLICATION_HPP