exceptions.hpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  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 <realm/status.hpp>
  21. #include <stdexcept>
  22. #include <system_error>
  23. namespace realm {
  24. class Exception : public std::exception {
  25. public:
  26. Exception(ErrorCodes::Error err, std::string_view str);
  27. explicit Exception(Status status);
  28. const char* what() const noexcept final;
  29. const Status& to_status() const;
  30. std::string_view reason() const noexcept;
  31. ErrorCodes::Error code() const noexcept;
  32. ErrorCategory category() const noexcept;
  33. std::string_view code_string() const noexcept;
  34. private:
  35. Status m_status;
  36. };
  37. /*
  38. * This will convert an exception in a catch(...) block into a Status. For `Exception`s, it returns the
  39. * status held in the exception directly. Otherwise it returns a status with an UnknownError error code and a
  40. * reason string holding the exception type and message.
  41. *
  42. * Currently this works for exceptions that derive from std::exception or Exception only.
  43. */
  44. Status exception_to_status() noexcept;
  45. /// The UnsupportedFileFormatVersion exception is thrown by DB::open()
  46. /// constructor when opening a database that uses a deprecated file format
  47. /// and/or a deprecated history schema which this version of Realm cannot
  48. /// upgrade from.
  49. struct UnsupportedFileFormatVersion : Exception {
  50. UnsupportedFileFormatVersion(int version);
  51. ~UnsupportedFileFormatVersion() noexcept override;
  52. /// The unsupported version of the file.
  53. int source_version = 0;
  54. };
  55. /// Thrown when a key is already existing when trying to create a new object
  56. struct KeyAlreadyUsed : Exception {
  57. KeyAlreadyUsed(std::string_view msg)
  58. : Exception(ErrorCodes::KeyAlreadyUsed, msg)
  59. {
  60. }
  61. ~KeyAlreadyUsed() noexcept override;
  62. };
  63. /// The \c LogicError exception class is intended to be thrown only when
  64. /// applications (or bindings) violate rules that are stated (or ought to have
  65. /// been stated) in the documentation of the public API, and only in cases
  66. /// where the violation could have been easily and efficiently predicted by the
  67. /// application. In other words, this exception class is for the cases where
  68. /// the error is due to incorrect use of the public API.
  69. ///
  70. /// This class is not supposed to be caught by applications. It is not even
  71. /// supposed to be considered part of the public API, and therefore the
  72. /// documentation of the public API should **not** mention the \c LogicError
  73. /// exception class by name. Note how this contrasts with other exception
  74. /// classes, such as \c NoSuchTable, which are part of the public API, and are
  75. /// supposed to be mentioned in the documentation by name. The \c LogicError
  76. /// exception is part of Realm's private API.
  77. ///
  78. /// In other words, the \c LogicError class should exclusively be used in
  79. /// replacement (or in addition to) asserts (debug or not) in order to
  80. /// guarantee program interruption, while still allowing for complete
  81. /// test-cases to be written and run.
  82. ///
  83. /// To this effect, the special `CHECK_LOGIC_ERROR()` macro is provided as a
  84. /// test framework plugin to allow unit tests to check that the functions in
  85. /// the public API do throw \c LogicError when rules are violated.
  86. ///
  87. /// The reason behind hiding this class from the public API is to prevent users
  88. /// from getting used to the idea that "Undefined Behaviour" equates a specific
  89. /// exception being thrown. The whole point of properly documenting "Undefined
  90. /// Behaviour" cases is to help the user know what the limits are, without
  91. /// constraining the database to handle every and any use-case thrown at it.
  92. struct LogicError : Exception {
  93. LogicError(ErrorCodes::Error code, std::string_view msg);
  94. ~LogicError() noexcept override;
  95. };
  96. struct RuntimeError : Exception {
  97. RuntimeError(ErrorCodes::Error code, std::string_view msg);
  98. ~RuntimeError() noexcept override;
  99. };
  100. /// Thrown when creating references that are too large to be contained in our ref_type (size_t)
  101. struct MaximumFileSizeExceeded : RuntimeError {
  102. MaximumFileSizeExceeded(std::string_view msg)
  103. : RuntimeError(ErrorCodes::MaximumFileSizeExceeded, msg)
  104. {
  105. }
  106. ~MaximumFileSizeExceeded() noexcept override;
  107. };
  108. /// Thrown when writing fails because the disk is full.
  109. struct OutOfDiskSpace : RuntimeError {
  110. OutOfDiskSpace(std::string_view msg)
  111. : RuntimeError(ErrorCodes::OutOfDiskSpace, msg)
  112. {
  113. }
  114. ~OutOfDiskSpace() noexcept override;
  115. };
  116. /// Thrown when a sync agent attempts to join a session in which there is
  117. /// already a sync agent. A session may only contain one sync agent at any given
  118. /// time.
  119. struct MultipleSyncAgents : RuntimeError {
  120. MultipleSyncAgents()
  121. : RuntimeError(ErrorCodes::MultipleSyncAgents, "Multiple sync agents attempted to join the same session")
  122. {
  123. }
  124. ~MultipleSyncAgents() noexcept override;
  125. };
  126. /// Thrown when memory can no longer be mapped to. When mmap/remap fails.
  127. struct AddressSpaceExhausted : RuntimeError {
  128. AddressSpaceExhausted(std::string_view msg)
  129. : RuntimeError(ErrorCodes::AddressSpaceExhausted, msg)
  130. {
  131. }
  132. ~AddressSpaceExhausted() noexcept override;
  133. };
  134. struct InvalidArgument : LogicError {
  135. InvalidArgument(std::string_view msg);
  136. InvalidArgument(ErrorCodes::Error code, std::string_view msg);
  137. ~InvalidArgument() noexcept override;
  138. };
  139. struct InvalidColumnKey : InvalidArgument {
  140. template <class T>
  141. InvalidColumnKey(const T& name)
  142. : InvalidArgument(ErrorCodes::InvalidProperty, util::format("Invalid property for object type %1", name))
  143. {
  144. }
  145. InvalidColumnKey()
  146. : InvalidArgument(ErrorCodes::InvalidProperty, "Invalid column key")
  147. {
  148. }
  149. ~InvalidColumnKey() noexcept override;
  150. };
  151. /// Thrown by various functions to indicate that a specified table does not
  152. /// exist.
  153. struct NoSuchTable : InvalidArgument {
  154. NoSuchTable()
  155. : InvalidArgument(ErrorCodes::NoSuchTable, "No such table exists")
  156. {
  157. }
  158. ~NoSuchTable() noexcept override;
  159. };
  160. /// Thrown by various functions to indicate that a specified table name is
  161. /// already in use.
  162. struct TableNameInUse : InvalidArgument {
  163. TableNameInUse()
  164. : InvalidArgument(ErrorCodes::TableNameInUse, "The specified table name is already in use")
  165. {
  166. }
  167. ~TableNameInUse() noexcept override;
  168. };
  169. /// Thrown when a key can not by found
  170. struct KeyNotFound : InvalidArgument {
  171. KeyNotFound(std::string_view msg)
  172. : InvalidArgument(ErrorCodes::KeyNotFound, msg)
  173. {
  174. }
  175. ~KeyNotFound() noexcept override;
  176. };
  177. struct NotNullable : InvalidArgument {
  178. NotNullable(std::string_view msg)
  179. : InvalidArgument(ErrorCodes::PropertyNotNullable, msg)
  180. {
  181. }
  182. template <class T, class U>
  183. NotNullable(const T& object_type, const U& property_name)
  184. : NotNullable(
  185. util::format("Invalid null value for non-nullable property '%1.%2'.", object_type, property_name))
  186. {
  187. }
  188. ~NotNullable() noexcept override;
  189. };
  190. struct PropertyTypeMismatch : InvalidArgument {
  191. template <class T, class U>
  192. PropertyTypeMismatch(const T& object_type, const U& property_name)
  193. : InvalidArgument(ErrorCodes::TypeMismatch,
  194. util::format("Type mismatch for property '%1.%2'.", object_type, property_name))
  195. {
  196. }
  197. ~PropertyTypeMismatch() noexcept override;
  198. };
  199. struct OutOfBounds : InvalidArgument {
  200. OutOfBounds(std::string_view msg, size_t idx, size_t sz);
  201. ~OutOfBounds() noexcept override;
  202. size_t index;
  203. size_t size;
  204. };
  205. struct InvalidEncryptionKey : InvalidArgument {
  206. InvalidEncryptionKey()
  207. : InvalidArgument(ErrorCodes::InvalidEncryptionKey, "Encryption key must be 64 bytes.")
  208. {
  209. }
  210. ~InvalidEncryptionKey() noexcept override;
  211. };
  212. struct StaleAccessor : LogicError {
  213. StaleAccessor(std::string_view msg)
  214. : LogicError(ErrorCodes::StaleAccessor, msg)
  215. {
  216. }
  217. ~StaleAccessor() noexcept override;
  218. };
  219. struct IllegalOperation : LogicError {
  220. IllegalOperation(std::string_view msg)
  221. : LogicError(ErrorCodes::IllegalOperation, msg)
  222. {
  223. }
  224. ~IllegalOperation() noexcept override;
  225. };
  226. struct NoSubscriptionForWrite : RuntimeError {
  227. NoSubscriptionForWrite(std::string_view msg)
  228. : RuntimeError(ErrorCodes::NoSubscriptionForWrite, msg)
  229. {
  230. }
  231. ~NoSubscriptionForWrite() noexcept override;
  232. };
  233. struct WrongTransactionState : LogicError {
  234. WrongTransactionState(std::string_view msg)
  235. : LogicError(ErrorCodes::WrongTransactionState, msg)
  236. {
  237. }
  238. ~WrongTransactionState() noexcept override;
  239. };
  240. struct InvalidTableRef : LogicError {
  241. InvalidTableRef(const char* cause)
  242. : LogicError(ErrorCodes::InvalidTableRef, cause)
  243. {
  244. }
  245. ~InvalidTableRef() noexcept override;
  246. };
  247. struct SerializationError : LogicError {
  248. SerializationError(std::string_view msg)
  249. : LogicError(ErrorCodes::SerializationError, msg)
  250. {
  251. }
  252. ~SerializationError() noexcept override;
  253. };
  254. struct NotImplemented : LogicError {
  255. NotImplemented()
  256. : LogicError(ErrorCodes::IllegalOperation, "Not implemented")
  257. {
  258. }
  259. ~NotImplemented() noexcept override;
  260. };
  261. struct MigrationFailed : LogicError {
  262. MigrationFailed(std::string_view msg)
  263. : LogicError(ErrorCodes::MigrationFailed, msg)
  264. {
  265. }
  266. ~MigrationFailed() noexcept override;
  267. };
  268. struct ObjectAlreadyExists : RuntimeError {
  269. template <class T, class U>
  270. ObjectAlreadyExists(const U& object_type, T pk_val)
  271. : RuntimeError(
  272. ErrorCodes::ObjectAlreadyExists,
  273. util::format("Attempting to create an object of type '%1' with an existing primary key value '%2'",
  274. object_type, pk_val))
  275. {
  276. }
  277. ~ObjectAlreadyExists() noexcept override;
  278. };
  279. // Thrown by functions that require a table to **not** be the target of link
  280. // columns, unless those link columns are part of the table itself.
  281. struct CrossTableLinkTarget : LogicError {
  282. template <class T>
  283. CrossTableLinkTarget(T table_name)
  284. : LogicError(ErrorCodes::CrossTableLinkTarget,
  285. util::format("Cannot remove %1 that is target of outside links", table_name))
  286. {
  287. }
  288. ~CrossTableLinkTarget() noexcept override;
  289. };
  290. /// Used for any I/O related exception. Note the derived exception
  291. /// types that are used for various specific types of errors.
  292. class FileAccessError : public RuntimeError {
  293. public:
  294. FileAccessError(ErrorCodes::Error code, std::string_view msg, std::string_view path, int err = 0);
  295. ~FileAccessError() noexcept override;
  296. /// Return the associated file system path, or the empty string if there is
  297. /// no associated file system path, or if the file system path is unknown.
  298. std::string_view get_path() const
  299. {
  300. return m_path;
  301. }
  302. int get_errno() const
  303. {
  304. return m_errno;
  305. }
  306. private:
  307. std::string m_path;
  308. int m_errno = 0;
  309. };
  310. struct SystemError : RuntimeError {
  311. SystemError(std::error_code err, std::string_view msg)
  312. : RuntimeError(ErrorCodes::SystemError, msg)
  313. {
  314. const_cast<Status&>(to_status()).set_std_error_code(err);
  315. }
  316. SystemError(int err_no, std::string_view msg)
  317. : SystemError(std::error_code(err_no, std::system_category()), msg)
  318. {
  319. }
  320. ~SystemError() noexcept override;
  321. std::error_code get_system_error() const
  322. {
  323. return to_status().get_std_error_code();
  324. }
  325. const std::error_category& get_category() const
  326. {
  327. return get_system_error().category();
  328. }
  329. };
  330. namespace query_parser {
  331. /// Exception thrown when parsing fails due to invalid syntax.
  332. struct SyntaxError : InvalidArgument {
  333. SyntaxError(std::string_view msg)
  334. : InvalidArgument(ErrorCodes::SyntaxError, msg)
  335. {
  336. }
  337. ~SyntaxError() noexcept override;
  338. };
  339. /// Exception thrown when binding a syntactically valid query string in a
  340. /// context where it does not make sense.
  341. struct InvalidQueryError : InvalidArgument {
  342. InvalidQueryError(std::string_view msg)
  343. : InvalidArgument(ErrorCodes::InvalidQuery, msg)
  344. {
  345. }
  346. ~InvalidQueryError() noexcept override;
  347. };
  348. /// Exception thrown when there is a problem accessing the arguments in a query string
  349. struct InvalidQueryArgError : InvalidArgument {
  350. InvalidQueryArgError(std::string_view msg)
  351. : InvalidArgument(ErrorCodes::InvalidQueryArg, msg)
  352. {
  353. }
  354. ~InvalidQueryArgError() noexcept override;
  355. };
  356. } // namespace query_parser
  357. } // namespace realm
  358. #endif // REALM_EXCEPTIONS_HPP