status.hpp 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. /*************************************************************************
  2. *
  3. * Copyright 2021 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. #pragma once
  19. #include <atomic>
  20. #include <cstdint>
  21. #include <iosfwd>
  22. #include <string>
  23. #include "realm/error_codes.hpp"
  24. #include "realm/string_data.hpp"
  25. #include "realm/util/bind_ptr.hpp"
  26. #include "realm/util/features.h"
  27. namespace realm {
  28. class REALM_NODISCARD Status {
  29. public:
  30. /*
  31. * This is the best way to construct a Status that represents a non-error condition.
  32. */
  33. static inline Status OK();
  34. /*
  35. * You can construct a Status from strings, C strings, and StringData's.
  36. */
  37. Status(ErrorCodes::Error code, StringData reason);
  38. Status(ErrorCodes::Error code, const std::string& reason);
  39. Status(ErrorCodes::Error code, const char* reason);
  40. /*
  41. * Copying a Status is just copying an intrusive pointer - i.e. very cheap. Moving them is similarly cheap.
  42. */
  43. inline Status(const Status& other);
  44. inline Status& operator=(const Status& other);
  45. inline Status(Status&& other) noexcept;
  46. inline Status& operator=(Status&& other) noexcept;
  47. inline bool is_ok() const noexcept;
  48. inline const std::string& reason() const noexcept;
  49. inline ErrorCodes::Error code() const noexcept;
  50. inline StringData code_string() const noexcept;
  51. /*
  52. * This class is marked nodiscard so that we always handle errors. If there is a place where we need
  53. * to explicitly ignore an error, you can call this function, which does nothing, to satisfy the compiler.
  54. */
  55. void ignore() const noexcept {}
  56. private:
  57. Status() = default;
  58. struct ErrorInfo {
  59. mutable std::atomic<uint32_t> m_refs;
  60. const ErrorCodes::Error m_code;
  61. const std::string m_reason;
  62. static util::bind_ptr<ErrorInfo> create(ErrorCodes::Error code, StringData reason);
  63. protected:
  64. template <typename>
  65. friend class ::realm::util::bind_ptr;
  66. inline void bind_ptr() const noexcept
  67. {
  68. m_refs.fetch_add(1, std::memory_order_relaxed);
  69. }
  70. inline void unbind_ptr() const noexcept
  71. {
  72. if (m_refs.fetch_sub(1, std::memory_order_acq_rel) == 1) {
  73. delete this;
  74. }
  75. }
  76. private:
  77. ErrorInfo(ErrorCodes::Error code, StringData reason);
  78. };
  79. util::bind_ptr<ErrorInfo> m_error = {};
  80. };
  81. class ExceptionForStatus : public std::exception {
  82. public:
  83. const char* what() const noexcept final
  84. {
  85. return reason().c_str();
  86. }
  87. const Status& to_status() const
  88. {
  89. return m_status;
  90. }
  91. const std::string& reason() const noexcept
  92. {
  93. return m_status.reason();
  94. }
  95. ErrorCodes::Error code() const noexcept
  96. {
  97. return m_status.code();
  98. }
  99. StringData code_string() const noexcept
  100. {
  101. return m_status.code_string();
  102. }
  103. ExceptionForStatus(ErrorCodes::Error err, StringData str)
  104. : m_status(err, str)
  105. {
  106. }
  107. explicit ExceptionForStatus(Status status)
  108. : m_status(std::move(status))
  109. {
  110. }
  111. private:
  112. Status m_status;
  113. };
  114. /*
  115. * This will convert an exception in a catch(...) block into a Status. For ExceptionForStatus's, it returns the
  116. * status held in the exception directly. Otherwise it returns a status with an UnknownError error code and a
  117. * reason string holding the exception type and message.
  118. *
  119. * Currently this works for exceptions that derive from std::exception or ExceptionForStatus only.
  120. */
  121. Status exception_to_status() noexcept;
  122. std::ostream& operator<<(std::ostream& out, const Status& val);
  123. inline bool operator==(const Status& lhs, const Status& rhs) noexcept
  124. {
  125. return lhs.code() == rhs.code();
  126. }
  127. inline bool operator!=(const Status& lhs, const Status& rhs) noexcept
  128. {
  129. return lhs.code() != rhs.code();
  130. }
  131. inline bool operator==(const Status& lhs, ErrorCodes::Error rhs) noexcept
  132. {
  133. return lhs.code() == rhs;
  134. }
  135. inline bool operator!=(const Status& lhs, ErrorCodes::Error rhs) noexcept
  136. {
  137. return lhs.code() != rhs;
  138. }
  139. inline Status Status::OK()
  140. {
  141. // Returns a status with m_error set to nullptr.
  142. return Status{};
  143. }
  144. inline Status::Status(const Status& other)
  145. : m_error(other.m_error)
  146. {
  147. }
  148. inline Status& Status::operator=(const Status& other)
  149. {
  150. m_error = other.m_error;
  151. return *this;
  152. }
  153. inline Status::Status(Status&& other) noexcept
  154. : m_error(std::move(other.m_error))
  155. {
  156. }
  157. inline Status& Status::operator=(Status&& other) noexcept
  158. {
  159. m_error = std::move(other.m_error);
  160. return *this;
  161. }
  162. inline bool Status::is_ok() const noexcept
  163. {
  164. return !m_error;
  165. }
  166. inline const std::string& Status::reason() const noexcept
  167. {
  168. static const std::string empty;
  169. return m_error ? m_error->m_reason : empty;
  170. }
  171. inline ErrorCodes::Error Status::code() const noexcept
  172. {
  173. return m_error ? m_error->m_code : ErrorCodes::OK;
  174. }
  175. inline StringData Status::code_string() const noexcept
  176. {
  177. return ErrorCodes::error_string(code());
  178. }
  179. } // namespace realm