Skip to content

Commit f1d5723

Browse files
authored
*: support rich error (tikv#514)
It's a common practice to return a protobuf struct as error status instead of simple string. This PR adds complete rich error support by following C++'s implementations. It changes several APIs to make code clean and match other languages' design. For more informations about rich error model, you can also take a look at https://grpc.io/docs/guides/error/#richer-error-model. Close tikv#443. Signed-off-by: Jay Lee <[email protected]>
1 parent 64471c6 commit f1d5723

File tree

19 files changed

+509
-69
lines changed

19 files changed

+509
-69
lines changed

benchmark/src/bench.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -79,7 +79,7 @@ impl BenchmarkService for Benchmark {
7979
_: SimpleRequest,
8080
sink: ServerStreamingSink<SimpleResponse>,
8181
) {
82-
let f = sink.fail(RpcStatus::new(RpcStatusCode::UNIMPLEMENTED, None));
82+
let f = sink.fail(RpcStatus::new(RpcStatusCode::UNIMPLEMENTED));
8383
let keep_running = self.keep_running.clone();
8484
spawn!(ctx, keep_running, "reporting unimplemented method", f)
8585
}
@@ -90,7 +90,7 @@ impl BenchmarkService for Benchmark {
9090
_: RequestStream<SimpleRequest>,
9191
sink: DuplexSink<SimpleResponse>,
9292
) {
93-
let f = sink.fail(RpcStatus::new(RpcStatusCode::UNIMPLEMENTED, None));
93+
let f = sink.fail(RpcStatus::new(RpcStatusCode::UNIMPLEMENTED));
9494
let keep_running = self.keep_running.clone();
9595
spawn!(ctx, keep_running, "reporting unimplemented method", f)
9696
}

health/src/service.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -188,9 +188,9 @@ impl Health for HealthService {
188188
return;
189189
}
190190
ctx.spawn(
191-
sink.fail(RpcStatus::new(
191+
sink.fail(RpcStatus::with_message(
192192
RpcStatusCode::NOT_FOUND,
193-
Some("unknown service".to_owned()),
193+
"unknown service".to_owned(),
194194
))
195195
.map(|_| ()),
196196
)

health/tests/health_check.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@ fn assert_code(code: RpcStatusCode, client: &HealthClient, name: &str) {
3131
let mut req = HealthCheckRequest::default();
3232
req.service = name.to_string();
3333
match client.check(&req) {
34-
Err(Error::RpcFailure(s)) if s.status == code => return,
34+
Err(Error::RpcFailure(s)) if s.code() == code => return,
3535
r => panic!("{} != {:?}", code, r),
3636
}
3737
}
@@ -102,7 +102,7 @@ fn test_health_watch() {
102102
// Updating other service should not notify the stream.
103103
service.set_serving_status(TEST_SERVICE, ServingStatus::NotServing);
104104
match block_on(statuses.next()).unwrap() {
105-
Err(Error::RpcFailure(r)) if r.status == RpcStatusCode::DEADLINE_EXCEEDED => (),
105+
Err(Error::RpcFailure(r)) if r.code() == RpcStatusCode::DEADLINE_EXCEEDED => (),
106106
r => panic!("unexpected status {:?}", r),
107107
}
108108

interop/src/client.rs

+9-9
Original file line numberDiff line numberDiff line change
@@ -115,7 +115,7 @@ impl Client {
115115
futures_timer::Delay::new(Duration::from_millis(10)).await;
116116
sender.cancel();
117117
match receiver.await.unwrap_err() {
118-
grpc::Error::RpcFailure(s) => assert_eq!(s.status, RpcStatusCode::CANCELLED),
118+
grpc::Error::RpcFailure(s) => assert_eq!(s.code(), RpcStatusCode::CANCELLED),
119119
e => panic!("expected cancel, but got: {:?}", e),
120120
}
121121
println!("pass");
@@ -134,7 +134,7 @@ impl Client {
134134
assert_eq!(resp.get_payload().get_body().len(), 31415);
135135
sender.cancel();
136136
match receiver.try_next().await {
137-
Err(grpc::Error::RpcFailure(s)) => assert_eq!(s.status, RpcStatusCode::CANCELLED),
137+
Err(grpc::Error::RpcFailure(s)) => assert_eq!(s.code(), RpcStatusCode::CANCELLED),
138138
Err(e) => panic!("expected cancel, but got: {:?}", e),
139139
Ok(r) => panic!("expected error, but got: {:?}", r),
140140
}
@@ -151,7 +151,7 @@ impl Client {
151151
let _ = sender.send((req, WriteFlags::default())).await;
152152
match receiver.try_next().await {
153153
Err(grpc::Error::RpcFailure(s)) => {
154-
assert_eq!(s.status, RpcStatusCode::DEADLINE_EXCEEDED)
154+
assert_eq!(s.code(), RpcStatusCode::DEADLINE_EXCEEDED)
155155
}
156156
Err(e) => panic!("expected timeout, but got: {:?}", e),
157157
Ok(r) => panic!("expected error: {:?}", r),
@@ -170,8 +170,8 @@ impl Client {
170170
req.set_response_status(status.clone());
171171
match self.client.unary_call_async(&req)?.await.unwrap_err() {
172172
grpc::Error::RpcFailure(s) => {
173-
assert_eq!(s.status, RpcStatusCode::UNKNOWN);
174-
assert_eq!(s.details.as_ref().unwrap(), error_msg);
173+
assert_eq!(s.code(), RpcStatusCode::UNKNOWN);
174+
assert_eq!(s.message(), error_msg);
175175
}
176176
e => panic!("expected rpc failure: {:?}", e),
177177
}
@@ -181,8 +181,8 @@ impl Client {
181181
let _ = sender.send((req, WriteFlags::default())).await;
182182
match receiver.try_next().await {
183183
Err(grpc::Error::RpcFailure(s)) => {
184-
assert_eq!(s.status, RpcStatusCode::UNKNOWN);
185-
assert_eq!(s.details.as_ref().unwrap(), error_msg);
184+
assert_eq!(s.code(), RpcStatusCode::UNKNOWN);
185+
assert_eq!(s.message(), error_msg);
186186
}
187187
Err(e) => panic!("expected rpc failure: {:?}", e),
188188
Ok(r) => panic!("error expected, but got: {:?}", r),
@@ -199,7 +199,7 @@ impl Client {
199199
.await
200200
.unwrap_err()
201201
{
202-
grpc::Error::RpcFailure(s) => assert_eq!(s.status, RpcStatusCode::UNIMPLEMENTED),
202+
grpc::Error::RpcFailure(s) => assert_eq!(s.code(), RpcStatusCode::UNIMPLEMENTED),
203203
e => panic!("expected rpc failure: {:?}", e),
204204
}
205205
println!("pass");
@@ -214,7 +214,7 @@ impl Client {
214214
.await
215215
.unwrap_err()
216216
{
217-
grpc::Error::RpcFailure(s) => assert_eq!(s.status, RpcStatusCode::UNIMPLEMENTED),
217+
grpc::Error::RpcFailure(s) => assert_eq!(s.code(), RpcStatusCode::UNIMPLEMENTED),
218218
e => panic!("expected rpc failure: {:?}", e),
219219
}
220220
println!("pass");

interop/src/server.rs

+5-5
Original file line numberDiff line numberDiff line change
@@ -38,8 +38,8 @@ impl TestService for InteropTestService {
3838
) {
3939
if req.has_response_status() {
4040
let code = req.get_response_status().get_code();
41-
let msg = Some(req.take_response_status().take_message());
42-
let status = RpcStatus::new(code, msg);
41+
let msg = req.take_response_status().take_message();
42+
let status = RpcStatus::with_message(code, msg);
4343
let f = sink
4444
.fail(status)
4545
.map_err(|e| panic!("failed to send response: {:?}", e))
@@ -120,8 +120,8 @@ impl TestService for InteropTestService {
120120
while let Some(mut req) = stream.try_next().await? {
121121
if req.has_response_status() {
122122
let code = req.get_response_status().get_code();
123-
let msg = Some(req.take_response_status().take_message());
124-
let status = RpcStatus::new(code, msg);
123+
let msg = req.take_response_status().take_message();
124+
let status = RpcStatus::with_message(code, msg);
125125
sink.fail(status).await?;
126126
return Ok(());
127127
}
@@ -167,7 +167,7 @@ impl TestService for InteropTestService {
167167

168168
fn unimplemented_call(&mut self, ctx: RpcContext, _: Empty, sink: UnarySink<Empty>) {
169169
let f = sink
170-
.fail(RpcStatus::new(RpcStatusCode::UNIMPLEMENTED, None))
170+
.fail(RpcStatus::new(RpcStatusCode::UNIMPLEMENTED))
171171
.map_err(|e| error!("failed to report unimplemented method: {:?}", e))
172172
.map(|_| ());
173173
ctx.spawn(f)

proto/Cargo.toml

+2-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ build = "build.rs"
1515
[features]
1616
default = ["protobuf-codec", "use-bindgen"]
1717
protobuf-codec = ["grpcio/protobuf-codec", "protobuf-build/grpcio-protobuf-codec"]
18-
prost-codec = ["prost-derive", "bytes", "lazy_static", "grpcio/prost-codec", "prost", "protobuf-build/grpcio-prost-codec"]
18+
prost-codec = ["prost-derive", "prost-types", "bytes", "lazy_static", "grpcio/prost-codec", "prost", "protobuf-build/grpcio-prost-codec"]
1919
use-bindgen = ["grpcio/use-bindgen"]
2020

2121
[dependencies]
@@ -24,6 +24,7 @@ grpcio = { path = "..", features = ["secure"], version = "0.8.0", default-featur
2424
bytes = { version = "1.0", optional = true }
2525
prost = { version = "0.7", optional = true }
2626
prost-derive = { version = "0.7", optional = true }
27+
prost-types = { version = "0.7", optional = true }
2728
protobuf = "2"
2829
lazy_static = { version = "1.3", optional = true }
2930

proto/build.rs

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ fn main() {
88
("grpc/testing", "testing"),
99
("grpc/health/v1/", "health"),
1010
("grpc/example", "example"),
11+
("google/rpc", "rpc"),
1112
];
1213
for (dir, package) in modules {
1314
let out_dir = format!("{}/{}", out_dir, package);
@@ -26,6 +27,7 @@ fn main() {
2627
.includes(&["proto".to_owned()])
2728
.files(&files)
2829
.out_dir(&out_dir)
30+
.black_list(&["protobuf"])
2931
.generate();
3032
}
3133
}

proto/proto/google/protobuf/any.proto

+158
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,158 @@
1+
// Protocol Buffers - Google's data interchange format
2+
// Copyright 2008 Google Inc. All rights reserved.
3+
// https://developers.google.com/protocol-buffers/
4+
//
5+
// Redistribution and use in source and binary forms, with or without
6+
// modification, are permitted provided that the following conditions are
7+
// met:
8+
//
9+
// * Redistributions of source code must retain the above copyright
10+
// notice, this list of conditions and the following disclaimer.
11+
// * Redistributions in binary form must reproduce the above
12+
// copyright notice, this list of conditions and the following disclaimer
13+
// in the documentation and/or other materials provided with the
14+
// distribution.
15+
// * Neither the name of Google Inc. nor the names of its
16+
// contributors may be used to endorse or promote products derived from
17+
// this software without specific prior written permission.
18+
//
19+
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20+
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21+
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22+
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23+
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24+
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25+
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26+
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27+
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28+
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29+
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30+
31+
syntax = "proto3";
32+
33+
package google.protobuf;
34+
35+
option csharp_namespace = "Google.Protobuf.WellKnownTypes";
36+
option go_package = "google.golang.org/protobuf/types/known/anypb";
37+
option java_package = "com.google.protobuf";
38+
option java_outer_classname = "AnyProto";
39+
option java_multiple_files = true;
40+
option objc_class_prefix = "GPB";
41+
42+
// `Any` contains an arbitrary serialized protocol buffer message along with a
43+
// URL that describes the type of the serialized message.
44+
//
45+
// Protobuf library provides support to pack/unpack Any values in the form
46+
// of utility functions or additional generated methods of the Any type.
47+
//
48+
// Example 1: Pack and unpack a message in C++.
49+
//
50+
// Foo foo = ...;
51+
// Any any;
52+
// any.PackFrom(foo);
53+
// ...
54+
// if (any.UnpackTo(&foo)) {
55+
// ...
56+
// }
57+
//
58+
// Example 2: Pack and unpack a message in Java.
59+
//
60+
// Foo foo = ...;
61+
// Any any = Any.pack(foo);
62+
// ...
63+
// if (any.is(Foo.class)) {
64+
// foo = any.unpack(Foo.class);
65+
// }
66+
//
67+
// Example 3: Pack and unpack a message in Python.
68+
//
69+
// foo = Foo(...)
70+
// any = Any()
71+
// any.Pack(foo)
72+
// ...
73+
// if any.Is(Foo.DESCRIPTOR):
74+
// any.Unpack(foo)
75+
// ...
76+
//
77+
// Example 4: Pack and unpack a message in Go
78+
//
79+
// foo := &pb.Foo{...}
80+
// any, err := anypb.New(foo)
81+
// if err != nil {
82+
// ...
83+
// }
84+
// ...
85+
// foo := &pb.Foo{}
86+
// if err := any.UnmarshalTo(foo); err != nil {
87+
// ...
88+
// }
89+
//
90+
// The pack methods provided by protobuf library will by default use
91+
// 'type.googleapis.com/full.type.name' as the type URL and the unpack
92+
// methods only use the fully qualified type name after the last '/'
93+
// in the type URL, for example "foo.bar.com/x/y.z" will yield type
94+
// name "y.z".
95+
//
96+
//
97+
// JSON
98+
// ====
99+
// The JSON representation of an `Any` value uses the regular
100+
// representation of the deserialized, embedded message, with an
101+
// additional field `@type` which contains the type URL. Example:
102+
//
103+
// package google.profile;
104+
// message Person {
105+
// string first_name = 1;
106+
// string last_name = 2;
107+
// }
108+
//
109+
// {
110+
// "@type": "type.googleapis.com/google.profile.Person",
111+
// "firstName": <string>,
112+
// "lastName": <string>
113+
// }
114+
//
115+
// If the embedded message type is well-known and has a custom JSON
116+
// representation, that representation will be embedded adding a field
117+
// `value` which holds the custom JSON in addition to the `@type`
118+
// field. Example (for message [google.protobuf.Duration][]):
119+
//
120+
// {
121+
// "@type": "type.googleapis.com/google.protobuf.Duration",
122+
// "value": "1.212s"
123+
// }
124+
//
125+
message Any {
126+
// A URL/resource name that uniquely identifies the type of the serialized
127+
// protocol buffer message. This string must contain at least
128+
// one "/" character. The last segment of the URL's path must represent
129+
// the fully qualified name of the type (as in
130+
// `path/google.protobuf.Duration`). The name should be in a canonical form
131+
// (e.g., leading "." is not accepted).
132+
//
133+
// In practice, teams usually precompile into the binary all types that they
134+
// expect it to use in the context of Any. However, for URLs which use the
135+
// scheme `http`, `https`, or no scheme, one can optionally set up a type
136+
// server that maps type URLs to message definitions as follows:
137+
//
138+
// * If no scheme is provided, `https` is assumed.
139+
// * An HTTP GET on the URL must yield a [google.protobuf.Type][]
140+
// value in binary format, or produce an error.
141+
// * Applications are allowed to cache lookup results based on the
142+
// URL, or have them precompiled into a binary to avoid any
143+
// lookup. Therefore, binary compatibility needs to be preserved
144+
// on changes to types. (Use versioned type names to manage
145+
// breaking changes.)
146+
//
147+
// Note: this functionality is not currently available in the official
148+
// protobuf release, and it is not used for type URLs beginning with
149+
// type.googleapis.com.
150+
//
151+
// Schemes other than `http`, `https` (or the empty scheme) might be
152+
// used with implementation specific semantics.
153+
//
154+
string type_url = 1;
155+
156+
// Must be a valid serialized protocol buffer of the above specified type.
157+
bytes value = 2;
158+
}

0 commit comments

Comments
 (0)