aboutsummaryrefslogtreecommitdiff
path: root/src/core/lib/promise/detail/status.h
blob: bfc649e6c48609d91155a8d3f3189df673788672 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
// Copyright 2021 gRPC authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

#ifndef GRPC_SRC_CORE_LIB_PROMISE_DETAIL_STATUS_H
#define GRPC_SRC_CORE_LIB_PROMISE_DETAIL_STATUS_H

#include <grpc/support/port_platform.h>

#include <utility>

#include "absl/status/status.h"
#include "absl/status/statusor.h"

// Helpers for dealing with absl::Status/StatusOr generically

namespace grpc_core {
namespace promise_detail {

// Convert with a move the input status to an absl::Status.
template <typename T>
absl::Status IntoStatus(absl::StatusOr<T>* status) {
  return std::move(status->status());
}

// Convert with a move the input status to an absl::Status.
inline absl::Status IntoStatus(absl::Status* status) {
  return std::move(*status);
}

}  // namespace promise_detail

// Return true if the status represented by the argument is ok, false if not.
// By implementing this function for other, non-absl::Status types, those types
// can participate in TrySeq as result types that affect control flow.
inline bool IsStatusOk(const absl::Status& status) { return status.ok(); }

template <typename T>
inline bool IsStatusOk(const absl::StatusOr<T>& status) {
  return status.ok();
}

template <typename To, typename From, typename SfinaeVoid = void>
struct StatusCastImpl;

template <typename To>
struct StatusCastImpl<To, To> {
  static To Cast(To&& t) { return std::move(t); }
};

template <typename To>
struct StatusCastImpl<To, const To&> {
  static To Cast(const To& t) { return t; }
};

template <typename T>
struct StatusCastImpl<absl::Status, absl::StatusOr<T>> {
  static absl::Status Cast(absl::StatusOr<T>&& t) {
    return std::move(t.status());
  }
};

template <typename T>
struct StatusCastImpl<absl::Status, absl::StatusOr<T>&> {
  static absl::Status Cast(const absl::StatusOr<T>& t) { return t.status(); }
};

template <typename T>
struct StatusCastImpl<absl::Status, const absl::StatusOr<T>&> {
  static absl::Status Cast(const absl::StatusOr<T>& t) { return t.status(); }
};

// StatusCast<> allows casting from one status-bearing type to another,
// regardless of whether the status indicates success or failure.
// This means that we can go from StatusOr to Status safely, but not in the
// opposite direction.
// For cases where the status is guaranteed to be a failure (and hence not
// needing to preserve values) see FailureStatusCast<> below.
template <typename To, typename From>
To StatusCast(From&& from) {
  return StatusCastImpl<To, From>::Cast(std::forward<From>(from));
}

template <typename To, typename From, typename SfinaeVoid = void>
struct FailureStatusCastImpl : public StatusCastImpl<To, From> {};

template <typename T>
struct FailureStatusCastImpl<absl::StatusOr<T>, absl::Status> {
  static absl::StatusOr<T> Cast(absl::Status&& t) { return std::move(t); }
};

template <typename T>
struct FailureStatusCastImpl<absl::StatusOr<T>, const absl::Status&> {
  static absl::StatusOr<T> Cast(const absl::Status& t) { return t; }
};

template <typename To, typename From>
To FailureStatusCast(From&& from) {
  GPR_DEBUG_ASSERT(!IsStatusOk(from));
  return FailureStatusCastImpl<To, From>::Cast(std::forward<From>(from));
}

}  // namespace grpc_core

#endif  // GRPC_SRC_CORE_LIB_PROMISE_DETAIL_STATUS_H