Skip to content

Add HTTP Bindings

Define HTTP Bindings

To define the HTTP bindings for our gRPC service, we need to map gRPC methods to HTTP endpoints.

This can be achieved either via gRPC API Gateway proto extensions or through a configuration file.

Configuration loading can be highly customized. Refer to Configuration for more details. By default, for any proto file file.proto, the files file_gateway.yaml, file_gateway.yml, and file_gateway.json will be tried in that order. If any file is available, it will be used and the search will be stopped.

Choose the method that works best for you and your project. This document provides guidelines for both methods.

Info

You can use the generate_unbound_methods option to automatically define HTTP bindings for gRPC methods that do not already have a defined binding. The path will be in the form of /proto.package.Service/Method and the HTTP method will be POST.

Let's create two HTTP endpoints for the Echo method, one using GET and one using POST, using your preferred method: either configuration files or proto annotations.

1. Using Configuration Files

Create a new file with the pattern <proto-filename>_gateway.yaml. Since we used echo_service.proto in this demo, we will create echo_service_gateway.yaml with the following content:

echo_service_gateway.yaml
1
2
3
4
5
6
7
gateway:
  endpoints:
    - selector: '~.EchoService.Echo' # (1)!
      get: '/echo/{text}' # (2)!
      additional_bindings: # (3)!
        - post: '/echo'
          body: '*' # (4)!
  1. selector is the dot-separated path to the service method. echo.EchoService.Echo is the full path. ~ gets substituted with the proto package from the related proto file and can be used to shorten the selector.
  2. This line specifies that the HTTP method is GET and the route is /echo/{text}, where text is a path to a field in the request proto message.
  3. Used to specify additional HTTP endpoint bindings.
  4. body (default: null) specifies which fields in the proto request message should be read from the HTTP body. In this case, * indicates that all fields in the proto message should be read from the HTTP body.

Tip

If you choose to work with configuration files, consider installing a YAML or JSON extension for your editor. Files named according to the pattern *_gateway.[yml|yaml|json] utilize the API Gateway's configuration schema, providing auto-completion and in-editor documentation features.

2. Using Proto Extensions

To use proto extensions, first download and import the gRPC API Gateway annotations.

Create a file named buf.yaml with the following content:

buf.yaml
1
2
3
version: v1
deps:
  - "buf.build/meshapi/grpc-api-gateway"

Download the dependencies using:

$ buf mod update

Download the proto files from the api/meshapi directory in the gRPC API Gateway git repository to a local directory named meshapi.

Modify your existing proto file with the following additions:

echo_service.proto
syntax = "proto3";

package echo;

import "meshapi/gateway/annotations.proto"; //(1)!

option go_package = "demo/echo";

message EchoRequest {
  string text = 1;
  bool capitalize = 2;
}

message EchoResponse {
    string text = 1;
}

service EchoService {
    // Echo returns the received text and makes it louder too!
    rpc Echo(EchoRequest) returns (EchoResponse) {
        option (meshapi.gateway.http) = {
            get: '/echo/{text}' //(2)!
            additional_bindings: [ //(3)!
              {
                post: '/echo',
                body: '*' //(4)!
              }
            ]
        };
    };
}
  1. This line imports the gRPC API Gateway proto annotations.
  2. This line specifies that the HTTP method is GET and the route is /echo/{text}, where text is a path to a field in the request proto message.
  3. Used to specify additional HTTP endpoint bindings.
  4. body (default: null) specifies which fields in the proto request message should be read from the HTTP body. In this case, * indicates that all fields in the proto message should be read from the HTTP body.

Add HTTP Server

Now that we have defined HTTP bindings, we need to regenerate the gateway code.

$ buf generate
$ protoc \
    --go_out=gen \
    --go-grpc_out=gen \
    --grpc-api-gateway_out=gen \
    --openapiv3_out=gen \
    echo_service.proto

Using either method, you should now see a new file named echo_service.pb.rgw.go.

Next, get the meshapi/grpc-api-gateway module as it contains necessary types for our HTTP server:

$ go get github.com/meshapi/grpc-api-gateway

Finally, update main.go to add the HTTP server:

main.go
package main

import (
    "context"
    "demo/gen/demo/echo"
    "log"
    "net"
    "strings"

    "github.com/meshapi/grpc-api-gateway/gateway"
    "google.golang.org/grpc"
    "google.golang.org/grpc/credentials/insecure"
)

// ... removed service implementation for brevity.

func main() {
    listener, err := net.Listen("tcp", ":40000")
    if err != nil {
        log.Fatalf("failed to listen: %s", err)
    }

    gateway := gateway.NewServeMux()

    connection, err := grpc.NewClient( //(1)!
        ":40000",
        grpc.WithTransportCredentials(insecure.NewCredentials()))
    if err != nil {
        log.Fatalf("could not dial gRPC server: %v", err)
    }

    server := grpc.NewServer()
    echo.RegisterEchoServiceServer(server, Service{})
    echo.RegisterEchoServiceHandler(context.Background(), gateway, connection) //(2)!

    go func() {
        log.Fatalln(http.ListenAndServe("0.0.0.0:4000", gateway))
    }()

    if err := server.Serve(listener); err != nil {
        log.Fatalf("gRPC server failed: %v", err)
    }
}
  1. We create a gRPC connection to our own gRPC server. gRPC API Gateway uses this connection to communicate with our gRPC services.
  2. RegisterEchoServiceHandler is a generated function that registers the EchoService to the gateway mux.

See it in action

Time to run the code and see it work:

$ go run .

You should be able to send an HTTP request and get a response back:

$ curl http://localhost:4000/echo/greetings

You should get the following response back:

{ "text": "greetings" }