Diag-Client-Lib
result.h
Go to the documentation of this file.
1 /* Diagnostic Client library
2  * Copyright (C) 2024 Avijit Dey
3  *
4  * This Source Code Form is subject to the terms of the Mozilla Public
5  * License, v. 2.0. If a copy of the MPL was not distributed with this
6  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
7 */
8 #ifndef DIAG_CLIENT_LIB_LIB_PLATFORM_CORE_RESULT_H_
9 #define DIAG_CLIENT_LIB_LIB_PLATFORM_CORE_RESULT_H_
10 
11 #include <optional>
12 #include <type_traits>
13 #include <utility>
14 #include <variant>
15 
17 
18 namespace core_type {
19 
28 template<typename T, typename E = ErrorCode>
29 class Result final {
30  public:
34  using value_type = T;
35 
39  using error_type = E;
40 
48  static Result FromValue(T &t) noexcept { return Result{t}; }
49 
57  static Result FromValue(T &&t) noexcept { return Result{std::move(t)}; }
58 
71  template<typename... Args>
72  static Result FromValue(Args &&...args) noexcept {
73  return Result{std::forward<Args>(args)...};
74  }
75 
83  static Result FromError(const E &e) noexcept { return Result{e}; }
84 
92  static Result FromError(E &&e) noexcept { return Result{std::move(e)}; }
93 
106  template<typename... Args>
107  static Result FromError(Args &&...args) noexcept {
108  return Result{std::forward<Args>(args)...};
109  }
110 
116  constexpr explicit Result(const T &t) noexcept : storage_{t} {}
117 
123  constexpr explicit Result(T &&t) noexcept : storage_{std::move(t)} {}
124 
130  constexpr explicit Result(const E &e) noexcept : storage_{e} {}
131 
137  constexpr explicit Result(E &&e) noexcept : storage_{std::move(e)} {}
138 
144  Result(const Result &other) = default;
145 
153  Result &operator=(const Result &other) = default;
154 
160  Result(Result &&other) noexcept(std::is_nothrow_move_constructible<T>::value
161  &&std::is_nothrow_move_constructible<E>::value) = default;
162 
170  Result &operator=(Result &&other) noexcept(
171  std::is_nothrow_move_constructible<T>::value &&std::is_nothrow_move_assignable<T>::value
172  &&std::is_nothrow_move_constructible<E>::value
173  &&std::is_nothrow_move_assignable<E>::value) = default;
174 
178  ~Result() noexcept = default;
179 
187  template<typename... Args>
188  void EmplaceValue(Args &&...args) noexcept {
189  storage_.template emplace<value_type>(std::forward<Args>(args)...);
190  }
191 
199  template<typename... Args>
200  void EmplaceError(Args &&...args) noexcept {
201  storage_.template emplace<error_type>(std::forward<Args>(args)...);
202  }
203 
209  bool HasValue() const noexcept { return std::holds_alternative<value_type>(storage_); }
210 
216  explicit operator bool() const noexcept { return HasValue(); }
217 
224  const T &operator*() const &noexcept { return std::get<value_type>(storage_); }
225 
232  T &&operator*() &&noexcept { return std::move(std::get<value_type>(storage_)); }
233 
240  const T *operator->() const noexcept { return std::get_if<value_type>(storage_); }
241 
248  const T &Value() const &noexcept { return std::get<value_type>(storage_); }
249 
256  T &&Value() &&noexcept { return std::move(std::get<value_type>(storage_)); }
257 
264  const E &Error() const &noexcept { return std::get<error_type>(storage_); }
265 
272  E &&Error() &&noexcept { return std::move(std::get<error_type>(storage_)); }
273 
279  std::optional<T> Ok() const &noexcept {
280  std::optional<T> opt_val{};
281  if (HasValue()) { opt_val.emplace(Value()); }
282  return opt_val;
283  }
284 
290  std::optional<T> Ok() &&noexcept {
291  std::optional<T> opt_val{};
292  if (HasValue()) { opt_val.emplace(std::move(Value())); }
293  return opt_val;
294  }
295 
301  std::optional<E> Err() const &noexcept {
302  std::optional<E> opt_err{};
303  if (!HasValue()) { opt_err.emplace(Error()); }
304  return opt_err;
305  }
306 
312  std::optional<E> Err() &&noexcept {
313  std::optional<E> opt_err{};
314  if (!HasValue()) { opt_err.emplace(std::move(Error())); }
315  return opt_err;
316  }
317 
328  template<typename F, typename R2 = typename std::invoke_result_t<F, value_type &&>>
329  auto AndThen(F &&fn) &&noexcept -> R2 {
330  return HasValue() ? std::forward<F>(fn)(std::move(*this).Value())
331  : R2{std::move(*this).Error()};
332  }
333 
344  template<typename F, typename E2 = std::invoke_result_t<F, E>>
346  return HasValue() ? Result<T, E2>{std::move(*this).Value()}
347  : Result<T, E2>{fn(std::move(*this).Error())};
348  }
349 
360  template<typename F>
361  Result OrElse(F &&fn) &&noexcept {
362  return HasValue() ? Result{std::move(*this)} : Result{fn(std::move(*this).Error())};
363  }
364 
375  template<typename U>
376  T ValueOr(U &&defaultValue) const &noexcept {
377  return HasValue() ? Result{*this} : static_cast<T>(defaultValue);
378  }
379 
390  template<typename U>
391  T ValueOr(U &&defaultValue) &&noexcept {
392  return HasValue() ? Result{std::move(*this)} : static_cast<T>(defaultValue);
393  }
394 
405  template<typename G>
406  E ErrorOr(G &&defaultError) const &noexcept {
407  return !HasValue() ? Result{*this} : static_cast<E>(defaultError);
408  }
409 
420  template<typename G>
421  E ErrorOr(G &&defaultError) &&noexcept {
422  return !HasValue() ? Result{std::move(*this)} : static_cast<E>(defaultError);
423  }
424 
437  template<typename F>
438  T Resolve(F &&f) const {
439  return HasValue() ? Result{*this} : f(Error());
440  }
441 
442  private:
446  std::variant<T, E> storage_;
447 };
448 
455 template<typename E>
456 class Result<void, E> final {
457  public:
461  using value_type = void;
462 
466  using error_type = E;
467 
473  static Result FromValue() noexcept { return Result{}; }
474 
482  static Result FromError(const E &e) noexcept { return Result{e}; }
483 
491  static Result FromError(E &&e) noexcept { return Result{std::move(e)}; }
492 
505  template<typename... Args>
506  static Result FromError(Args &&...args) noexcept {
507  return Result{std::forward<Args>(args)...};
508  }
509 
513  constexpr explicit Result() noexcept : storage_{empty_value{}} {}
514 
520  constexpr explicit Result(const E &e) noexcept : storage_{e} {}
521 
527  constexpr explicit Result(E &&e) noexcept : storage_{std::move(e)} {}
528 
534  Result(const Result &other) = default;
535 
543  Result &operator=(const Result &other) = default;
544 
550  Result(Result &&other) noexcept(std::is_nothrow_move_constructible<E>::value) = default;
551 
559  Result &operator=(Result &&other) noexcept(
560  std::is_nothrow_move_constructible<E>::value &&std::is_nothrow_move_assignable<E>::value) =
561  default;
562 
566  ~Result() noexcept { static_assert(std::is_trivially_destructible<E>::value); }
567 
571  void EmplaceValue() noexcept { storage_.EmplaceValue(empty_value{}); }
572 
580  template<typename... Args>
581  void EmplaceError(Args &&...args) noexcept {
582  storage_.EmplaceError(std::forward<Args>(args)...);
583  }
584 
590  bool HasValue() const noexcept { return storage_.HasValue(); }
591 
597  explicit operator bool() const noexcept { return HasValue(); }
598 
605  const E &Error() const &noexcept { return storage_.Error(); }
606 
613  E &&Error() &&noexcept { return std::move(storage_.Error()); }
614 
620  std::optional<E> Err() const &noexcept {
621  std::optional<E> opt_err{};
622  if (!HasValue()) { opt_err.emplace(Error()); }
623  return opt_err;
624  }
625 
631  std::optional<E> Err() &&noexcept {
632  std::optional<E> opt_err{};
633  if (!HasValue()) { opt_err.emplace(std::move(Error())); }
634  return opt_err;
635  }
636 
647  template<typename F, typename E2 = std::invoke_result_t<F, E>>
649  return HasValue() ? Result<void, E2>{} : Result<void, E2>{fn(Error())};
650  }
651 
662  template<typename F, typename E2 = std::invoke_result_t<F, E>>
664  return HasValue() ? Result<void, E2>{} : Result<void, E2>{fn(Error())};
665  }
666 
677  template<typename F>
678  Result AndThen(F &&fn) &&noexcept {
679  if (HasValue()) { fn(); }
680  return Result{std::move(*this)};
681  }
682 
693  template<typename F>
694  Result OrElse(F &&fn) &&noexcept {
695  return HasValue() ? Result{std::move(*this)} : Result{fn(Error())};
696  }
697 
708  template<typename G>
709  E ErrorOr(G &&defaultError) const &noexcept {
710  return !HasValue() ? Result{*this} : static_cast<E>(defaultError);
711  }
712 
723  template<typename G>
724  E ErrorOr(G &&defaultError) &&noexcept {
725  return !HasValue() ? Result{std::move(*this)} : static_cast<E>(defaultError);
726  }
727 
739  template<typename F>
740  void Resolve(F &&f) const {
741  f(Error());
742  }
743 
744  private:
748  struct empty_value {};
749 
754 };
755 
756 } // namespace core_type
757 
758 #endif // DIAG_CLIENT_LIB_LIB_PLATFORM_CORE_RESULT_H_
static Result FromError(E &&e) noexcept
Build a new Result from the specified error (given as rvalue)
Definition: result.h:491
Result & operator=(const Result &other)=default
Copy-assign another Result to this instance.
constexpr Result() noexcept
Construct a new Result with a "void" value.
Definition: result.h:513
Result< void, E2 > CheckError(F &&fn)
Returns the result of the given function on the contained value if it exists; otherwise,...
Definition: result.h:663
static Result FromValue() noexcept
Build a new Result with "void" as value.
Definition: result.h:473
void EmplaceError(Args &&...args) noexcept
Put a new error into this instance, constructed in-place from the given arguments.
Definition: result.h:581
E error_type
Type alias for the type E of errors.
Definition: result.h:466
const E & Error() const &noexcept
Access the contained error.
Definition: result.h:605
Result AndThen(F &&fn) &&noexcept
Returns the result of the given function on the contained value if it exists; otherwise,...
Definition: result.h:678
Result OrElse(F &&fn) &&noexcept
Returns the result itself if it contains a value; otherwise, returns the result of the given function...
Definition: result.h:694
void value_type
Type alias for the type T of values, always "void" for this specialization.
Definition: result.h:461
E && Error() &&noexcept
Access the contained error.
Definition: result.h:613
Result< void, E2 > MapError(F &&fn)
Returns the result of the given function on the contained value if it exists; otherwise,...
Definition: result.h:648
bool HasValue() const noexcept
Check whether *this contains a value.
Definition: result.h:590
static Result FromError(const E &e) noexcept
Build a new Result from the specified error (given as lvalue)
Definition: result.h:482
Result(Result &&other) noexcept(std::is_nothrow_move_constructible< E >::value)=default
Move-construct a new Result from another instance.
Result & operator=(Result &&other) noexcept(std::is_nothrow_move_constructible< E >::value &&std::is_nothrow_move_assignable< E >::value)=default
Move-assign another Result to this instance.
E ErrorOr(G &&defaultError) &&noexcept
Return the contained error or the given default error.
Definition: result.h:724
constexpr Result(const E &e) noexcept
Construct a new Result from the specified error (given as lvalue)
Definition: result.h:520
E ErrorOr(G &&defaultError) const &noexcept
Return the contained error or the given default error.
Definition: result.h:709
static Result FromError(Args &&...args) noexcept
Build a new Result from an error that is constructed in-place from the given arguments.
Definition: result.h:506
void Resolve(F &&f) const
Do nothing or call a function.
Definition: result.h:740
std::optional< E > Err() &&noexcept
Return the contained error as an Optional.
Definition: result.h:631
std::optional< E > Err() const &noexcept
Return the contained error as an Optional.
Definition: result.h:620
Result< empty_value, E > storage_
Storage to contain an empty value or error of type E.
Definition: result.h:753
void EmplaceValue() noexcept
Put a new value into this instance, constructed in-place from the given arguments.
Definition: result.h:571
~Result() noexcept
Destruct an instance of Result.
Definition: result.h:566
Result(const Result &other)=default
Copy-construct a new Result from another instance.
constexpr Result(E &&e) noexcept
Construct a new Result from the specified error (given as rvalue)
Definition: result.h:527
Class type to contains a value (of type ValueType), or an error (of type ErrorType)
Definition: result.h:29
constexpr Result(const T &t) noexcept
Construct a new Result from the specified value (given as lvalue)
Definition: result.h:116
const T & operator*() const &noexcept
Access the contained value.
Definition: result.h:224
constexpr Result(T &&t) noexcept
Construct a new Result from the specified value (given as rvalue)
Definition: result.h:123
Result & operator=(Result &&other) noexcept(std::is_nothrow_move_constructible< T >::value &&std::is_nothrow_move_assignable< T >::value &&std::is_nothrow_move_constructible< E >::value &&std::is_nothrow_move_assignable< E >::value)=default
Move-assign another Result to this instance.
static Result FromValue(T &&t) noexcept
Build a new Result from the specified value (given as rvalue)
Definition: result.h:57
std::optional< E > Err() const &noexcept
Return the contained error as an Optional.
Definition: result.h:301
std::optional< T > Ok() const &noexcept
Return the contained value as an Optional.
Definition: result.h:279
T ValueOr(U &&defaultValue) const &noexcept
Return the contained value or the given default value.
Definition: result.h:376
void EmplaceError(Args &&...args) noexcept
Put a new error into this instance, constructed in-place from the given arguments.
Definition: result.h:200
Result< T, E2 > MapError(F &&fn)
Returns the result of the given function on the contained value if it exists; otherwise,...
Definition: result.h:345
E ErrorOr(G &&defaultError) &&noexcept
Return the contained error or the given default error.
Definition: result.h:421
std::variant< T, E > storage_
Storage to contain value of type T or error of type E.
Definition: result.h:446
std::optional< T > Ok() &&noexcept
Return the contained value as an Optional.
Definition: result.h:290
Result(const Result &other)=default
Copy-construct a new Result from another instance.
Result(Result &&other) noexcept(std::is_nothrow_move_constructible< T >::value &&std::is_nothrow_move_constructible< E >::value)=default
Move-construct a new Result from another instance.
Result & operator=(const Result &other)=default
Copy-assign another Result to this instance.
const T & Value() const &noexcept
Access the contained value.
Definition: result.h:248
static Result FromError(const E &e) noexcept
Build a new Result from the specified error (given as lvalue)
Definition: result.h:83
auto AndThen(F &&fn) &&noexcept -> R2
Returns the result of the given function on the contained value if it exists; otherwise,...
Definition: result.h:329
const E & Error() const &noexcept
Access the contained error.
Definition: result.h:264
static Result FromError(Args &&...args) noexcept
Build a new Result from an error that is constructed in-place from the given arguments.
Definition: result.h:107
E ErrorOr(G &&defaultError) const &noexcept
Return the contained error or the given default error.
Definition: result.h:406
E && Error() &&noexcept
Access the contained error.
Definition: result.h:272
T Resolve(F &&f) const
Return the contained value or return the result of a function call.
Definition: result.h:438
constexpr Result(const E &e) noexcept
Construct a new Result from the specified error (given as lvalue)
Definition: result.h:130
bool HasValue() const noexcept
Check whether *this contains a value.
Definition: result.h:209
~Result() noexcept=default
Destruct an instance of Result.
T && Value() &&noexcept
Access the contained value.
Definition: result.h:256
T && operator*() &&noexcept
Access the contained value.
Definition: result.h:232
Result OrElse(F &&fn) &&noexcept
Returns the result itself if it contains a value; otherwise, returns the result of the given function...
Definition: result.h:361
const T * operator->() const noexcept
Access the contained value.
Definition: result.h:240
T ValueOr(U &&defaultValue) &&noexcept
Return the contained value or the given default value.
Definition: result.h:391
E error_type
Type alias for the type E of errors.
Definition: result.h:39
constexpr Result(E &&e) noexcept
Construct a new Result from the specified error (given as rvalue)
Definition: result.h:137
T value_type
Type alias for the type T of values.
Definition: result.h:34
static Result FromValue(Args &&...args) noexcept
Build a new Result from a value that is constructed in-place from the given arguments.
Definition: result.h:72
static Result FromError(E &&e) noexcept
Build a new Result from the specified error (given as rvalue)
Definition: result.h:92
void EmplaceValue(Args &&...args) noexcept
Put a new value into this instance, constructed in-place from the given arguments.
Definition: result.h:188
static Result FromValue(T &t) noexcept
Build a new Result from the specified value (given as lvalue)
Definition: result.h:48
std::optional< E > Err() &&noexcept
Return the contained error as an Optional.
Definition: result.h:312