mirror of https://github.com/grpc/grpc-node.git
Merge pull request #2585 from murgatroid99/example_error_handling
Add error handling example
This commit is contained in:
commit
f7d9baaf25
|
@ -0,0 +1,23 @@
|
||||||
|
# Error Handling
|
||||||
|
|
||||||
|
This example demonstrates basic RPC error handling in gRPC for unary and
|
||||||
|
streaming response cardinalities.
|
||||||
|
|
||||||
|
## Start the server
|
||||||
|
|
||||||
|
Run the server, whcih returns an error if the RPC request's `name` field is
|
||||||
|
empty.
|
||||||
|
|
||||||
|
```
|
||||||
|
node server.js
|
||||||
|
```
|
||||||
|
|
||||||
|
## Run the client
|
||||||
|
|
||||||
|
Then run the client in another terminal, which makes two requests for each of
|
||||||
|
unary and streaming responses: one with an empty Name field and one with it
|
||||||
|
populated with the current username provided by os/user.
|
||||||
|
|
||||||
|
```
|
||||||
|
node client.js
|
||||||
|
```
|
|
@ -0,0 +1,89 @@
|
||||||
|
/*
|
||||||
|
*
|
||||||
|
* Copyright 2023 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.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
const grpc = require('@grpc/grpc-js');
|
||||||
|
const protoLoader = require('@grpc/proto-loader');
|
||||||
|
const parseArgs = require('minimist');
|
||||||
|
const os = require('os');
|
||||||
|
|
||||||
|
const PROTO_PATH = __dirname + '/../protos/helloworld.proto';
|
||||||
|
|
||||||
|
const packageDefinition = protoLoader.loadSync(
|
||||||
|
PROTO_PATH,
|
||||||
|
{keepCase: true,
|
||||||
|
longs: String,
|
||||||
|
enums: String,
|
||||||
|
defaults: true,
|
||||||
|
oneofs: true
|
||||||
|
});
|
||||||
|
const helloProto = grpc.loadPackageDefinition(packageDefinition).helloworld;
|
||||||
|
|
||||||
|
function unaryCall(client, requestId, name, expectedCode) {
|
||||||
|
console.log(`[${requestId}] Calling SayHello with name:"${name}"`);
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
client.sayHello({name: name}, (error, value) => {
|
||||||
|
if (error) {
|
||||||
|
if (error.code === expectedCode) {
|
||||||
|
console.log(`[${requestId}] Received error ${error.message}`);
|
||||||
|
} else {
|
||||||
|
console.log(`[${requestId}] Received unexpected error ${error.message}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (value) {
|
||||||
|
console.log(`[${requestId}] Received response ${value.message}`);
|
||||||
|
}
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function streamingCall(client, requestId, name, expectedCode) {
|
||||||
|
console.log(`[${requestId}] Calling SayHelloStreamReply with name:"${name}"`);
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const call = client.sayHelloStreamReply({name: name});
|
||||||
|
call.on('data', value => {
|
||||||
|
console.log(`[${requestId}] Received response ${value.message}`);
|
||||||
|
});
|
||||||
|
call.on('status', status => {
|
||||||
|
console.log(`[${requestId}] Received status with code=${grpc.status[status.code]} details=${status.details}`);
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
call.on('error', error => {
|
||||||
|
if (error.code === expectedCode) {
|
||||||
|
console.log(`[${requestId}] Received expected error ${error.message}`);
|
||||||
|
} else {
|
||||||
|
console.log(`[${requestId}] Received unexpected error ${error.message}`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function main() {
|
||||||
|
let argv = parseArgs(process.argv.slice(2), {
|
||||||
|
string: 'target',
|
||||||
|
default: {target: 'localhost:50052'}
|
||||||
|
});
|
||||||
|
const client = new helloProto.Greeter(argv.target, grpc.credentials.createInsecure());
|
||||||
|
const name = os.userInfo().username ?? 'unknown';
|
||||||
|
await unaryCall(client, 1, '', grpc.status.INVALID_ARGUMENT);
|
||||||
|
await unaryCall(client, 2, name, grpc.status.OK);
|
||||||
|
await streamingCall(client, 3, '', grpc.status.INVALID_ARGUMENT);
|
||||||
|
await streamingCall(client, 4, name, grpc.status.OK);
|
||||||
|
}
|
||||||
|
|
||||||
|
main();
|
|
@ -0,0 +1,68 @@
|
||||||
|
/*
|
||||||
|
*
|
||||||
|
* Copyright 2023 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.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
var PROTO_PATH = __dirname + '/../protos/helloworld.proto';
|
||||||
|
|
||||||
|
var grpc = require('@grpc/grpc-js');
|
||||||
|
var protoLoader = require('@grpc/proto-loader');
|
||||||
|
var packageDefinition = protoLoader.loadSync(
|
||||||
|
PROTO_PATH,
|
||||||
|
{keepCase: true,
|
||||||
|
longs: String,
|
||||||
|
enums: String,
|
||||||
|
defaults: true,
|
||||||
|
oneofs: true
|
||||||
|
});
|
||||||
|
var hello_proto = grpc.loadPackageDefinition(packageDefinition).helloworld;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Implements the SayHello RPC method.
|
||||||
|
*/
|
||||||
|
function sayHello(call, callback) {
|
||||||
|
if (call.request.name === '') {
|
||||||
|
callback({code: grpc.status.INVALID_ARGUMENT, details: 'request missing required field: name'});
|
||||||
|
}
|
||||||
|
callback(null, {message: 'Hello ' + call.request.name});
|
||||||
|
}
|
||||||
|
|
||||||
|
const REPLY_COUNT = 5;
|
||||||
|
|
||||||
|
function sayHelloStreamReply(call) {
|
||||||
|
if (call.request.name === '') {
|
||||||
|
call.emit('error', {code: grpc.status.INVALID_ARGUMENT, details: 'request missing required field: name'});
|
||||||
|
} else {
|
||||||
|
for (let i = 0; i < REPLY_COUNT; i++) {
|
||||||
|
call.write({message: 'Hello ' + call.request.name});
|
||||||
|
}
|
||||||
|
call.end();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Starts an RPC server that receives requests for the Greeter service at the
|
||||||
|
* sample server port
|
||||||
|
*/
|
||||||
|
function main() {
|
||||||
|
var server = new grpc.Server();
|
||||||
|
server.addService(hello_proto.Greeter.service, {sayHello: sayHello, sayHelloStreamReply: sayHelloStreamReply});
|
||||||
|
server.bindAsync('0.0.0.0:50052', grpc.ServerCredentials.createInsecure(), () => {
|
||||||
|
server.start();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
main();
|
Loading…
Reference in New Issue