exceptions.hpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  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_EXCEPTIONS_HPP
  19. #define REALM_EXCEPTIONS_HPP
  20. #include <stdexcept>
  21. #include <realm/util/features.h>
  22. #include <realm/util/backtrace.hpp>
  23. #include <realm/util/to_string.hpp>
  24. namespace realm {
  25. using util::ExceptionWithBacktrace;
  26. /// Thrown by various functions to indicate that a specified table does not
  27. /// exist.
  28. class NoSuchTable : public ExceptionWithBacktrace<std::exception> {
  29. public:
  30. const char* message() const noexcept override;
  31. };
  32. class InvalidTableRef : public ExceptionWithBacktrace<std::exception> {
  33. public:
  34. InvalidTableRef(const char* cause)
  35. : m_message(cause)
  36. {
  37. }
  38. const char* message() const noexcept override
  39. {
  40. return m_message.c_str();
  41. }
  42. std::string m_message;
  43. };
  44. /// Thrown by various functions to indicate that a specified table name is
  45. /// already in use.
  46. class TableNameInUse : public ExceptionWithBacktrace<std::exception> {
  47. public:
  48. const char* message() const noexcept override;
  49. };
  50. // Thrown by functions that require a table to **not** be the target of link
  51. // columns, unless those link columns are part of the table itself.
  52. class CrossTableLinkTarget : public ExceptionWithBacktrace<std::exception> {
  53. public:
  54. const char* message() const noexcept override;
  55. };
  56. /// Thrown by various functions to indicate that the dynamic type of a table
  57. /// does not match a particular other table type (dynamic or static).
  58. class DescriptorMismatch : public ExceptionWithBacktrace<std::exception> {
  59. public:
  60. const char* message() const noexcept override;
  61. };
  62. /// The UnsupportedFileFormatVersion exception is thrown by DB::open()
  63. /// constructor when opening a database that uses a deprecated file format
  64. /// and/or a deprecated history schema which this version of Realm cannot
  65. /// upgrade from.
  66. class UnsupportedFileFormatVersion : public ExceptionWithBacktrace<> {
  67. public:
  68. UnsupportedFileFormatVersion(int source_version);
  69. /// The unsupported version of the file.
  70. int source_version = 0;
  71. };
  72. /// Thrown when a sync agent attempts to join a session in which there is
  73. /// already a sync agent. A session may only contain one sync agent at any given
  74. /// time.
  75. class MultipleSyncAgents : public ExceptionWithBacktrace<std::exception> {
  76. public:
  77. const char* message() const noexcept override;
  78. };
  79. /// Thrown when memory can no longer be mapped to. When mmap/remap fails.
  80. class AddressSpaceExhausted : public std::runtime_error {
  81. public:
  82. AddressSpaceExhausted(const std::string& msg);
  83. /// runtime_error::what() returns the msg provided in the constructor.
  84. };
  85. /// Thrown when creating references that are too large to be contained in our ref_type (size_t)
  86. class MaximumFileSizeExceeded : public std::runtime_error {
  87. public:
  88. MaximumFileSizeExceeded(const std::string& msg);
  89. /// runtime_error::what() returns the msg provided in the constructor.
  90. };
  91. /// Thrown when writing fails because the disk is full.
  92. class OutOfDiskSpace : public std::runtime_error {
  93. public:
  94. OutOfDiskSpace(const std::string& msg);
  95. /// runtime_error::what() returns the msg provided in the constructor.
  96. };
  97. /// Thrown when a key can not by found
  98. class KeyNotFound : public std::runtime_error {
  99. public:
  100. KeyNotFound(const std::string& msg)
  101. : std::runtime_error(msg)
  102. {
  103. }
  104. };
  105. /// Thrown when a column can not by found
  106. class ColumnNotFound : public std::runtime_error {
  107. public:
  108. ColumnNotFound()
  109. : std::runtime_error("Column not found")
  110. {
  111. }
  112. };
  113. /// Thrown when a column key is already used
  114. class ColumnAlreadyExists : public std::runtime_error {
  115. public:
  116. ColumnAlreadyExists()
  117. : std::runtime_error("Column already exists")
  118. {
  119. }
  120. };
  121. /// Thrown when a key is already existing when trying to create a new object
  122. class KeyAlreadyUsed : public std::runtime_error {
  123. public:
  124. KeyAlreadyUsed(const std::string& msg)
  125. : std::runtime_error(msg)
  126. {
  127. }
  128. };
  129. // SerialisationError intentionally does not inherit ExceptionWithBacktrace
  130. // because the query-based-sync permissions queries generated on the server
  131. // use a LinksToNode which is not currently serialisable (this limitation can
  132. // be lifted in core 6 given stable ids). Coupled with query metrics which
  133. // serialize all queries, the capturing of the stack for these frequent
  134. // permission queries shows up in performance profiles.
  135. class SerialisationError : public std::runtime_error {
  136. public:
  137. SerialisationError(const std::string& msg);
  138. /// runtime_error::what() returns the msg provided in the constructor.
  139. };
  140. // thrown when a user constructed link path is not a valid input
  141. class InvalidPathError : public std::runtime_error {
  142. public:
  143. InvalidPathError(const std::string& msg);
  144. /// runtime_error::what() returns the msg provided in the constructor.
  145. };
  146. class DuplicatePrimaryKeyValueException : public std::logic_error {
  147. public:
  148. DuplicatePrimaryKeyValueException(std::string object_type, std::string property);
  149. std::string const& object_type() const
  150. {
  151. return m_object_type;
  152. }
  153. std::string const& property() const
  154. {
  155. return m_property;
  156. }
  157. private:
  158. std::string m_object_type;
  159. std::string m_property;
  160. };
  161. /// The \c LogicError exception class is intended to be thrown only when
  162. /// applications (or bindings) violate rules that are stated (or ought to have
  163. /// been stated) in the documentation of the public API, and only in cases
  164. /// where the violation could have been easily and efficiently predicted by the
  165. /// application. In other words, this exception class is for the cases where
  166. /// the error is due to incorrect use of the public API.
  167. ///
  168. /// This class is not supposed to be caught by applications. It is not even
  169. /// supposed to be considered part of the public API, and therefore the
  170. /// documentation of the public API should **not** mention the \c LogicError
  171. /// exception class by name. Note how this contrasts with other exception
  172. /// classes, such as \c NoSuchTable, which are part of the public API, and are
  173. /// supposed to be mentioned in the documentation by name. The \c LogicError
  174. /// exception is part of Realm's private API.
  175. ///
  176. /// In other words, the \c LogicError class should exclusively be used in
  177. /// replacement (or in addition to) asserts (debug or not) in order to
  178. /// guarantee program interruption, while still allowing for complete
  179. /// test-cases to be written and run.
  180. ///
  181. /// To this effect, the special `CHECK_LOGIC_ERROR()` macro is provided as a
  182. /// test framework plugin to allow unit tests to check that the functions in
  183. /// the public API do throw \c LogicError when rules are violated.
  184. ///
  185. /// The reason behind hiding this class from the public API is to prevent users
  186. /// from getting used to the idea that "Undefined Behaviour" equates a specific
  187. /// exception being thrown. The whole point of properly documenting "Undefined
  188. /// Behaviour" cases is to help the user know what the limits are, without
  189. /// constraining the database to handle every and any use-case thrown at it.
  190. class LogicError : public ExceptionWithBacktrace<std::exception> {
  191. public:
  192. enum ErrorKind {
  193. string_too_big,
  194. binary_too_big,
  195. table_name_too_long,
  196. column_name_too_long,
  197. column_name_in_use,
  198. invalid_column_name,
  199. table_index_out_of_range,
  200. row_index_out_of_range,
  201. column_index_out_of_range,
  202. string_position_out_of_range,
  203. link_index_out_of_range,
  204. bad_version,
  205. illegal_type,
  206. /// Indicates that an argument has a value that is illegal in combination
  207. /// with another argument, or with the state of an involved object.
  208. illegal_combination,
  209. /// Indicates a data type mismatch, such as when `Table::find_pkey_int()` is
  210. /// called and the type of the primary key is not `type_Int`.
  211. type_mismatch,
  212. /// Indicates that two involved tables are not in the same group.
  213. group_mismatch,
  214. /// Indicates that an involved descriptor is of the wrong kind, i.e., if
  215. /// it is a subtable descriptor, and the function requires a root table
  216. /// descriptor.
  217. wrong_kind_of_descriptor,
  218. /// Indicates that an involved table is of the wrong kind, i.e., if it
  219. /// is a subtable, and the function requires a root table, or if it is a
  220. /// free-standing table, and the function requires a group-level table.
  221. wrong_kind_of_table,
  222. /// Indicates that an involved accessor is was detached, i.e., was not
  223. /// attached to an underlying object.
  224. detached_accessor,
  225. /// Indicates that a specified row index of a target table (a link) is
  226. /// out of range. This is used for disambiguation in cases such as
  227. /// Table::set_link() where one specifies both a row index of the origin
  228. /// table, and a row index of the target table.
  229. target_row_index_out_of_range,
  230. // Indicates that an involved column lacks a search index.
  231. no_search_index,
  232. /// Indicates that a modification was attempted that would have produced a
  233. /// duplicate primary value.
  234. unique_constraint_violation,
  235. /// User attempted to insert null in non-nullable column
  236. column_not_nullable,
  237. /// Group::open() is called on a group accessor that is already in the
  238. /// attached state. Or Group::open() or Group::commit() is called on a
  239. /// group accessor that is managed by a DB object.
  240. wrong_group_state,
  241. /// No active transaction on a particular Transaction object (e.g. after commit)
  242. /// or the Transaction object is of the wrong type (write to a read-only transaction)
  243. wrong_transact_state,
  244. /// Attempted use of a continuous transaction through a DB
  245. /// object with no history. See Replication::get_history().
  246. no_history,
  247. /// Durability setting (as passed to the DB constructor) was
  248. /// not consistent across the session.
  249. mixed_durability,
  250. /// History type (as specified by the Replication implementation passed
  251. /// to the DB constructor) was not consistent across the
  252. /// session.
  253. mixed_history_type,
  254. /// History schema version (as specified by the Replication
  255. /// implementation passed to the DB constructor) was not
  256. /// consistent across the session.
  257. mixed_history_schema_version,
  258. /// Adding rows to a table with no columns is not supported.
  259. table_has_no_columns,
  260. /// Referring to a column that has been deleted.
  261. column_does_not_exist,
  262. /// You can not add index on a subtable of a subtable
  263. subtable_of_subtable_index,
  264. /// You try to instantiate a collection object not matching column type
  265. collection_type_mismatch
  266. };
  267. LogicError(ErrorKind message);
  268. const char* message() const noexcept override;
  269. ErrorKind kind() const noexcept;
  270. private:
  271. ErrorKind m_kind;
  272. };
  273. // Implementation:
  274. // LCOV_EXCL_START (Wording of what() strings are not to be tested)
  275. inline const char* NoSuchTable::message() const noexcept
  276. {
  277. return "No such table exists";
  278. }
  279. inline const char* TableNameInUse::message() const noexcept
  280. {
  281. return "The specified table name is already in use";
  282. }
  283. inline const char* CrossTableLinkTarget::message() const noexcept
  284. {
  285. return "Table is target of cross-table link columns";
  286. }
  287. inline const char* DescriptorMismatch::message() const noexcept
  288. {
  289. return "Table descriptor mismatch";
  290. }
  291. inline UnsupportedFileFormatVersion::UnsupportedFileFormatVersion(int version)
  292. : ExceptionWithBacktrace<>(
  293. util::format("Database has an unsupported version (%1) and cannot be upgraded", version))
  294. , source_version(version)
  295. {
  296. }
  297. inline const char* MultipleSyncAgents::message() const noexcept
  298. {
  299. return "Multiple sync agents attempted to join the same session";
  300. }
  301. // LCOV_EXCL_STOP
  302. inline AddressSpaceExhausted::AddressSpaceExhausted(const std::string& msg)
  303. : std::runtime_error(msg)
  304. {
  305. }
  306. inline MaximumFileSizeExceeded::MaximumFileSizeExceeded(const std::string& msg)
  307. : std::runtime_error(msg)
  308. {
  309. }
  310. inline OutOfDiskSpace::OutOfDiskSpace(const std::string& msg)
  311. : std::runtime_error(msg)
  312. {
  313. }
  314. inline SerialisationError::SerialisationError(const std::string& msg)
  315. : std::runtime_error(msg)
  316. {
  317. }
  318. inline InvalidPathError::InvalidPathError(const std::string& msg)
  319. : runtime_error(msg)
  320. {
  321. }
  322. inline LogicError::LogicError(LogicError::ErrorKind k)
  323. : m_kind(k)
  324. {
  325. }
  326. inline LogicError::ErrorKind LogicError::kind() const noexcept
  327. {
  328. return m_kind;
  329. }
  330. } // namespace realm
  331. #endif // REALM_EXCEPTIONS_HPP