status_with.hpp 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213
  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 <type_traits>
  20. #include "realm/status.hpp"
  21. #include "realm/util/assert.hpp"
  22. #include "realm/util/features.h"
  23. #include "realm/util/optional.hpp"
  24. namespace realm {
  25. template <typename T>
  26. class StatusWith;
  27. template <typename T>
  28. constexpr bool is_status_with = false;
  29. template <typename T>
  30. constexpr bool is_status_with<StatusWith<T>> = true;
  31. template <typename T>
  32. constexpr bool is_status_or_status_with = std::is_same_v<T, ::realm::Status> || is_status_with<T>;
  33. template <typename T>
  34. using StatusOrStatusWith = std::conditional_t<std::is_void_v<T>, Status, StatusWith<T>>;
  35. /**
  36. * StatusWith is used to return an error or a value.
  37. * This class is designed to make exception-free code cleaner by not needing as many out
  38. * parameters.
  39. *
  40. * Example:
  41. * StatusWith<int> fib( int n ) {
  42. * if ( n < 0 )
  43. * return StatusWith<int>( ErrorCodes::BadValue, "parameter to fib has to be >= 0" );
  44. * if ( n <= 1 ) return StatusWith<int>( 1 );
  45. * StatusWith<int> a = fib( n - 1 );
  46. * StatusWith<int> b = fib( n - 2 );
  47. * if ( !a.isOK() ) return a;
  48. * if ( !b.isOK() ) return b;
  49. * return StatusWith<int>( a.getValue() + b.getValue() );
  50. * }
  51. */
  52. template <typename T>
  53. class REALM_NODISCARD StatusWith {
  54. static_assert(!is_status_or_status_with<T>, "StatusWith<Status> and StatusWith<StatusWith<T>> are banned.");
  55. public:
  56. using value_type = T;
  57. template <typename Reason, std::enable_if_t<std::is_constructible_v<std::string_view, Reason>, int> = 0>
  58. StatusWith(ErrorCodes::Error code, Reason reason)
  59. : m_status(code, reason)
  60. {
  61. }
  62. StatusWith(Status status)
  63. : m_status(std::move(status))
  64. {
  65. }
  66. StatusWith(T value)
  67. : m_status(Status::OK())
  68. , m_value(std::move(value))
  69. {
  70. }
  71. bool is_ok() const
  72. {
  73. return m_status.is_ok();
  74. }
  75. const T& get_value() const
  76. {
  77. REALM_ASSERT_DEBUG(is_ok());
  78. REALM_ASSERT_RELEASE(m_value);
  79. return *m_value;
  80. }
  81. T& get_value()
  82. {
  83. REALM_ASSERT_DEBUG(is_ok());
  84. REALM_ASSERT_RELEASE(m_value);
  85. return *m_value;
  86. }
  87. const Status& get_status() const
  88. {
  89. return m_status;
  90. }
  91. private:
  92. Status m_status;
  93. util::Optional<T> m_value;
  94. };
  95. template <typename T, typename... Args>
  96. StatusWith<T> make_status_with(Args&&... args)
  97. {
  98. return StatusWith<T>{T(std::forward<Args>(args)...)};
  99. }
  100. template <typename T>
  101. auto operator<<(std::ostream& stream, const StatusWith<T>& sw)
  102. -> decltype(stream << sw.get_value()) // SFINAE on T streamability.
  103. {
  104. if (sw.is_ok())
  105. return stream << sw.get_value();
  106. return stream << sw.get_status();
  107. }
  108. //
  109. // EqualityComparable(StatusWith<T>, T). Intentionally not providing an ordering relation.
  110. //
  111. template <typename T>
  112. bool operator==(const StatusWith<T>& sw, const T& val)
  113. {
  114. return sw.is_ok() && sw.get_value() == val;
  115. }
  116. template <typename T>
  117. bool operator==(const T& val, const StatusWith<T>& sw)
  118. {
  119. return sw.is_ok() && val == sw.get_value();
  120. }
  121. template <typename T>
  122. bool operator!=(const StatusWith<T>& sw, const T& val)
  123. {
  124. return !(sw == val);
  125. }
  126. template <typename T>
  127. bool operator!=(const T& val, const StatusWith<T>& sw)
  128. {
  129. return !(val == sw);
  130. }
  131. //
  132. // EqualityComparable(StatusWith<T>, Status)
  133. //
  134. template <typename T>
  135. bool operator==(const StatusWith<T>& sw, const Status& status)
  136. {
  137. return sw.get_status() == status;
  138. }
  139. template <typename T>
  140. bool operator==(const Status& status, const StatusWith<T>& sw)
  141. {
  142. return status == sw.get_status();
  143. }
  144. template <typename T>
  145. bool operator!=(const StatusWith<T>& sw, const Status& status)
  146. {
  147. return !(sw == status);
  148. }
  149. template <typename T>
  150. bool operator!=(const Status& status, const StatusWith<T>& sw)
  151. {
  152. return !(status == sw);
  153. }
  154. //
  155. // EqualityComparable(StatusWith<T>, ErrorCode)
  156. //
  157. template <typename T>
  158. bool operator==(const StatusWith<T>& sw, const ErrorCodes::Error code)
  159. {
  160. return sw.get_status() == code;
  161. }
  162. template <typename T>
  163. bool operator==(const ErrorCodes::Error code, const StatusWith<T>& sw)
  164. {
  165. return code == sw.get_status();
  166. }
  167. template <typename T>
  168. bool operator!=(const StatusWith<T>& sw, const ErrorCodes::Error code)
  169. {
  170. return !(sw == code);
  171. }
  172. template <typename T>
  173. bool operator!=(const ErrorCodes::Error code, const StatusWith<T>& sw)
  174. {
  175. return !(code == sw);
  176. }
  177. } // namespace realm