| // Copyright 2021 The ChromiumOS Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef MISSIVE_UTIL_STATUS_MACROS_H_ |
| #define MISSIVE_UTIL_STATUS_MACROS_H_ |
| |
| #include <optional> |
| #include <utility> |
| |
| #include <base/types/always_false.h> |
| #include <base/types/expected.h> |
| |
| #include "missive/util/status.h" |
| #include "missive/util/statusor.h" |
| |
| // Nocompile tests for this file are missing in this repo because platform2 |
| // packages do not yet have nocompile test support. See |
| // https://groups.google.com/a/google.com/g/chromeos-build-discuss/c/xy1Onm4YskM/m/LJKXy8rrBgAJ. |
| // Please sync code to components/reporting/util and add nocompile tests there. |
| |
| namespace reporting::internal { |
| |
| // Helper functions for the macro RETURN_IF_ERROR_STATUS. Overloads of the |
| // following functions to return if the given status is OK. If yes, the return |
| // value is nullopt. If not, the desired return value is returned. |
| std::optional<Status> ShouldReturnStatus(const Status& status); |
| std::optional<Status> ShouldReturnStatus(Status&& status); |
| std::optional<base::unexpected<Status>> ShouldReturnStatus( |
| const base::unexpected<Status>& status); |
| std::optional<base::unexpected<Status>> ShouldReturnStatus( |
| base::unexpected<Status>&& status); |
| |
| template <typename T> |
| void ShouldReturnStatus(T) { |
| static_assert(base::AlwaysFalse<T>, |
| "RETURN_IF_ERROR_STATUS only accepts either Status or " |
| "base::unexpected<Status>."); |
| } |
| } // namespace reporting::internal |
| |
| // Run a command that returns a Status. If the called code returns an |
| // error status, return that status up out of this method too. The macro can |
| // also apply on `base::unexpected<Status>`, which is needed when the return |
| // type is StatusOr. |
| // |
| // Examples: |
| // |
| // RETURN_IF_ERROR_STATUS(DoThing(4)); // Return type is Status |
| // |
| // // Return type is StatusOr |
| // RETURN_IF_ERROR_STATUS(base::unexpected(DoThing(4))); |
| #define RETURN_IF_ERROR_STATUS(expr) \ |
| do { \ |
| /* Using _status below to avoid capture problems if expr is "status". */ \ |
| if (auto _status = reporting::internal::ShouldReturnStatus((expr)); \ |
| _status.has_value()) { \ |
| return std::move(_status).value(); \ |
| } \ |
| } while (0) |
| |
| // Internal helper for concatenating macro values. |
| #define STATUS_MACROS_CONCAT_NAME_INNER(x, y) x##y |
| #define STATUS_MACROS_CONCAT_NAME(x, y) STATUS_MACROS_CONCAT_NAME_INNER(x, y) |
| |
| #define ASSIGN_OR_ONCE_CALLBACK_AND_RETURN_IMPL(result, lhs, callback, rexpr) \ |
| auto result = (rexpr); \ |
| if (__builtin_expect(!result.has_value(), 0)) { \ |
| std::move(callback).Run(std::move(result).error()); \ |
| return; \ |
| } \ |
| lhs = std::move(result).value(); |
| |
| // Executes an expression that returns a StatusOr, extracting its value into the |
| // variabled defined by lhs (or calls callback with error and returns). |
| // |
| // Example: |
| // base::OnceCallback<void(Status)> callback = |
| // base::BindOnce([](Status status) {...}); |
| // ASSIGN_OR_ONCE_CALLBACK_AND_RETURN(ValueType value, |
| // callback, |
| // MaybeGetValue(arg)); |
| // |
| // WARNING: ASSIGN_OR_RETURN expands into multiple statements; it cannot be used |
| // in a single statement (e.g. as the body of an if statement without {})! |
| #define ASSIGN_OR_ONCE_CALLBACK_AND_RETURN(lhs, callback, rexpr) \ |
| ASSIGN_OR_ONCE_CALLBACK_AND_RETURN_IMPL( \ |
| STATUS_MACROS_CONCAT_NAME(_status_or_value, __COUNTER__), lhs, callback, \ |
| rexpr) |
| |
| namespace reporting::internal { |
| |
| // Helper functions and classes for the macros *_OK. Overloads of the |
| // following functions to return if the given Status or StatusOr is OK. The |
| // template classes are needed here because template functions can't be |
| // partially specialized. |
| template <typename T> |
| struct StatusOKHelper { |
| static bool IsOK(const T&) { |
| static_assert(base::AlwaysFalse<T>, |
| "{CHECK,DCHECK,ASSERT,EXPECT}_OK do not accept a type other " |
| "than Status or StatusOr."); |
| } |
| }; |
| |
| template <typename T> |
| struct StatusOKHelper<StatusOr<T>> { |
| static bool IsOK(const StatusOr<T>& status_or) { |
| return status_or.has_value(); |
| } |
| }; |
| |
| template <> |
| struct StatusOKHelper<Status> { |
| static bool IsOK(const Status& status) { return status.ok(); } |
| }; |
| |
| template <typename T> |
| bool IsOK(const T& s) { |
| return StatusOKHelper<T>::IsOK(s); |
| } |
| } // namespace reporting::internal |
| |
| #define CHECK_OK(value) CHECK(internal::IsOK(value)) |
| #define DCHECK_OK(value) DCHECK(internal::IsOK(value)) |
| #define ASSERT_OK(value) ASSERT_TRUE(internal::IsOK(value)) |
| #define EXPECT_OK(value) EXPECT_TRUE(internal::IsOK(value)) |
| |
| #endif // MISSIVE_UTIL_STATUS_MACROS_H_ |