forked from infobloxopen/atlas-app-toolkit
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstatus.go
207 lines (188 loc) · 6.08 KB
/
status.go
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
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
package gateway
import (
"context"
"net/http"
"google.golang.org/genproto/googleapis/rpc/code"
"google.golang.org/grpc"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/grpclog"
"google.golang.org/grpc/metadata"
"google.golang.org/grpc/status"
"github.com/grpc-ecosystem/grpc-gateway/v2/runtime"
)
// OldStatesCreatedOnUpdate if true will return http.StatusCreated from HTTPStatusFromCode
// function if gRPC code is equal to Updated. This variable should only be set in an init()
// function by code that vendors this library.
var OldStatusCreatedOnUpdate = false
const (
// These custom codes defined here to conform REST API Syntax
// It is supposed that you do not send them over the wire as part of gRPC Status,
// because they will be treated as Unknown by gRPC library.
// You should use them to send successfull status of your RPC method
// using SetStatus function from this package.
Created codes.Code = 10000 + iota // 10000 is an offset from standard codes
Updated
Deleted
LongRunning
PartialContent
)
// SetStatus sets gRPC status as gRPC metadata
// Status.Code will be set with metadata key `grpcgateway-status-code` and
// with value as string name of the code.
// Status.Message will be set with metadata key `grpcgateway-status-message`
// and with corresponding value.
func SetStatus(ctx context.Context, st *status.Status) error {
if st == nil {
return nil
}
md := metadata.Pairs(
runtime.MetadataPrefix+"status-code", CodeName(st.Code()),
)
return grpc.SetHeader(ctx, md)
}
// SetCreated is a shortcut for SetStatus(ctx, status.New(Created, msg))
func SetCreated(ctx context.Context, msg string) error {
WithSuccess(ctx, NewWithFields(msg))
return SetStatus(ctx, status.New(Created, msg))
}
// SetUpdated is a shortcut for SetStatus(ctx, status.New(Updated, msg))
func SetUpdated(ctx context.Context, msg string) error {
return SetStatus(ctx, status.New(Updated, msg))
}
// SetDeleted is a shortcut for SetStatus(ctx, status.New(Deleted, msg))
func SetDeleted(ctx context.Context, msg string) error {
return SetStatus(ctx, status.New(Deleted, msg))
}
// SetRunning is a shortcut for SetStatus(ctx, status.New(LongRunning, url))
func SetRunning(ctx context.Context, message, resource string) error {
grpc.SetHeader(ctx, metadata.Pairs("Location", resource))
return SetStatus(ctx, status.New(LongRunning, message))
}
// Status returns REST representation of gRPC status.
// If status.Status is not nil it will be converted in accordance with REST
// API Syntax otherwise context will be used to extract
// `grpcgateway-status-code` from gRPC metadata.
// If `grpcgateway-status-code` is not set it is assumed that it is OK.
func HTTPStatus(ctx context.Context, st *status.Status) (int, string) {
if st != nil {
httpStatus := HTTPStatusFromCode(st.Code())
return httpStatus, CodeName(st.Code())
}
statusName := CodeName(codes.OK)
if sc, ok := Header(ctx, "status-code"); ok {
statusName = sc
}
httpCode := HTTPStatusFromCode(Code(statusName))
return httpCode, statusName
}
// CodeName returns stringname of gRPC code, function handles as standard
// codes from "google.golang.org/grpc/codes" as well as custom ones defined
// in this package.
// The codes.Unimplemented is named "NOT_IMPLEMENTED" in accordance with
// REST API Syntax Specification.
func CodeName(c codes.Code) string {
switch c {
case codes.Unimplemented:
return "NOT_IMPLEMENTED"
case Created:
return "CREATED"
case Updated:
return "UPDATED"
case Deleted:
return "DELETED"
case LongRunning:
return "LONG_RUNNING_OP"
case PartialContent:
return "PARTIAL_CONTENT"
default:
var cname string
if cn, ok := code.Code_name[int32(c)]; !ok {
cname = code.Code_UNKNOWN.String()
} else {
cname = cn
}
return cname
}
}
// Code returns an instance of gRPC code by its string name.
// The `cname` must be in upper case and one of the code names
// defined in REST API Syntax.
// If code name is invalid or unknow the codes.Unknown will be returned.
func Code(cname string) codes.Code {
switch cname {
case "NOT_IMPLEMENTED":
return codes.Unimplemented
case "CREATED":
return Created
case "UPDATED":
return Updated
case "DELETED":
return Deleted
case "LONG_RUNNING_OP":
return LongRunning
case "PARTIAL_CONTENT":
return PartialContent
default:
var c codes.Code
if cc, ok := code.Code_value[cname]; !ok {
c = codes.Unknown
} else {
c = codes.Code(cc)
}
return c
}
}
// HTTPStatusFromCode converts a gRPC error code into the corresponding HTTP response status.
func HTTPStatusFromCode(code codes.Code) int {
switch code {
case Created:
return http.StatusCreated
case Updated:
if OldStatusCreatedOnUpdate {
return http.StatusCreated
}
return http.StatusOK
case Deleted:
return http.StatusNoContent
case LongRunning:
return http.StatusAccepted
case PartialContent:
return http.StatusPartialContent
case codes.OK:
return http.StatusOK
case codes.Canceled:
return 499 // (gRPC-Gateway - http.StatusRequestTimeout = 408)
case codes.Unknown:
return http.StatusInternalServerError
case codes.InvalidArgument:
return http.StatusBadRequest
case codes.DeadlineExceeded:
return http.StatusGatewayTimeout // = 504 (gRPC-Gateway - http.StatusRequestTimeout = 408)
case codes.NotFound:
return http.StatusNotFound
case codes.AlreadyExists:
return http.StatusConflict
case codes.PermissionDenied:
return http.StatusForbidden
case codes.Unauthenticated:
return http.StatusUnauthorized
case codes.ResourceExhausted:
return http.StatusTooManyRequests // = 429 (gRPC-Gateway - http.StatusForbidden = 403)
case codes.FailedPrecondition:
return http.StatusBadRequest // = 400 (gRPC-Gateway - http.StatusPreconditionFailed = 412)
case codes.Aborted:
return http.StatusConflict
case codes.OutOfRange:
return http.StatusBadRequest
case codes.Unimplemented:
return http.StatusNotImplemented
case codes.Internal:
return http.StatusInternalServerError
case codes.Unavailable:
return http.StatusServiceUnavailable
case codes.DataLoss:
return http.StatusInternalServerError
}
grpclog.Infof("Unknown gRPC error code: %v", code)
return http.StatusInternalServerError
}