This tutorial provides a detailed guide on how to run a gRPC service and access it in the browser.
The first step when creating a gRPC service is to define the service methods
and their request and response message types using protocol buffers. In this
example, we define our EchoService
in a file called
echo.proto
. For more information about protocol buffers and
proto3 syntax, please see the protobuf documentation.
message EchoRequest {
string message = 1;
}
message EchoResponse {
string message = 1;
}
service EchoService {
rpc Echo(EchoRequest) returns (EchoResponse);
}
Next, we implement our EchoService interface using Node in the backend gRPC
EchoServer
. This will handle requests from clients. See the file
node-server/server.js
for details.
You can implement the server in any language supported by gRPC. Please see the gRPC website for more details.
function doEcho(call, callback) {
callback(null, {message: call.request.message});
}
In this example, we will use the Envoy proxy to forward the gRPC browser request to the backend server. You can see the complete config file in envoy.yaml
To forward the gRPC requests to the backend server, we need a block like this:
static_resources:
listeners:
- name: listener_0
address:
socket_address: { address: 0.0.0.0, port_value: 8080 }
filter_chains:
- filters:
- name: envoy.filters.network.http_connection_manager
typed_config:
"@type": type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager
codec_type: auto
stat_prefix: ingress_http
route_config:
name: local_route
virtual_hosts:
- name: local_service
domains: ["*"]
routes:
- match: { prefix: "/" }
route:
cluster: echo_service
timeout: 0s
http_filters:
- name: envoy.filters.http.grpc_web
typed_config:
"@type": type.googleapis.com/envoy.extensions.filters.http.grpc_web.v3.GrpcWeb
- name: envoy.filters.http.router
typed_config:
"@type": type.googleapis.com/envoy.extensions.filters.http.router.v3.Router
clusters:
- name: echo_service
connect_timeout: 0.25s
type: logical_dns
# HTTP/2 support
typed_extension_protocol_options:
envoy.extensions.upstreams.http.v3.HttpProtocolOptions:
"@type": type.googleapis.com/envoy.extensions.upstreams.http.v3.HttpProtocolOptions
explicit_http_config:
http2_protocol_options: {}
lb_policy: round_robin
load_assignment:
cluster_name: cluster_0
endpoints:
- lb_endpoints:
- endpoint:
address:
socket_address:
address: node-server
port_value: 9090
You may also need to add some CORS setup to make sure the browser can request cross-origin content.
In this simple example, the browser makes gRPC requests to port :8080
. Envoy
forwards the request to the backend gRPC server listening on port :9090
.
To generate the protobuf message classes from our echo.proto
, run the
following command:
$ protoc -I=$DIR echo.proto \
--js_out=import_style=commonjs:$OUT_DIR
The import_style
option passed to the --js_out
flag makes sure the
generated files will have CommonJS style require()
statements.
To generate the gRPC-Web service client stub, first you need the gRPC-Web
protoc plugin. To compile the plugin protoc-gen-grpc-web
, you need to run
this from the repo's root directory:
$ cd grpc-web
$ sudo make install-plugin
To generate the service client stub file, run this command:
$ protoc -I=$DIR echo.proto \
--grpc-web_out=import_style=commonjs,mode=grpcwebtext:$OUT_DIR
In the --grpc-web_out
param above:
mode
can begrpcwebtext
(default) orgrpcweb
import_style
can beclosure
(default) orcommonjs
Our command generates the client stub, by default, to the file
echo_grpc_web_pb.js
.
Now you are ready to write some JS client code. Put this in a client.js
file.
const {EchoRequest, EchoResponse} = require('./echo_pb.js');
const {EchoServiceClient} = require('./echo_grpc_web_pb.js');
var echoService = new EchoServiceClient('http://localhost:8080');
var request = new EchoRequest();
request.setMessage('Hello World!');
echoService.echo(request, {}, function(err, response) {
// ...
});
You will need a package.json
file
{
"name": "grpc-web-commonjs-example",
"dependencies": {
"google-protobuf": "~3.21.4",
"grpc-web": "~1.5.0"
},
"devDependencies": {
"webpack": "~5.82.1",
"webpack-cli": "~5.1.1"
}
}
Finally, putting all these together, we can compile all the relevant JS files into one single JS library that can be used in the browser.
$ npm install
$ npx webpack client.js
Now embed dist/main.js
into your project and see it in action!