/grpc-fallback-go

A lightweight proxy that connects gRPC to HTTP/1.1

Primary LanguageGoApache License 2.0Apache-2.0

grpc-fallback-go

grpc-fallback-go contains a gRPC Fallback-compliant proxy server and HTTP client, both implemented in Go.

Description

grpc-fallback-go consists of a gRPC Fallback proxy server package, a proxy binary, and a HTTP client.

Note: The gRPC Fallback protocol will be referred to as just "fallback" for the remainder of this document.

The proxy server is designed to front an arbitrary gRPC service. It accepts fallback requests, forwards a corresponding gRPC request to the gRPC backend, and translates the response to a HTTP response before returning it to the caller.

The proxy server can be used from a Go program, as a binary, or as a Docker image. Usage examples can be found in the Server Usage Example section below.

The Go client is basic and is meant for testing and simple use cases. Usage examples can be found in the Client Usage Examples section below.

Installation

Check the Releases tab for pre-built binaries in a variety of architectures.

Via Go Tooling

Proxy server binary:

> go get github.com/googleapis/grpc-fallback-go/cmd/fallback-proxy

Proxy server package:

> go get github.com/googleapis/grpc-fallback-go/server

HTTP Client

> go get github.com/googleapis/grpc-fallback-go/client

Server Usage Example

CLI Usage Example

> fallback-proxy -address "localhost:7469"
2019/06/13 18:35:01 Fallback server listening on port: :1337

In-process w/gRPC Backend Usage Example

// setup listener & server
lis, _ := net.Listen("tcp", port)
s := grpc.NewServer(opts...)

// Register Services to the server.
// ...

// Create a new grpc-fallback server on port 1337
// for gRPC server listening on "port"
fb := fallback.NewServer(":1337", "localhost"+port)
fb.StartBackground()
defer fb.Shutdown()

// Start gRPC server.
s.Serve(lis)

Docker Usage Example

> docker build -t gcr.io/gapic-images/fallback-proxy .
> docker run \
  --net=host \
  gcr.io/gapic-images/fallback-proxy \
  -address=localhost:7469 
2019/06/13 18:35:01 Fallback server listening on port: :1337

Client Usage Examples

Basic Client Usage Example

package main

import (
	"fmt"

	showpb "github.com/googleapis/gapic-showcase/server/genproto"
	"github.com/googleapis/grpc-fallback-go/client"
)

func main() {
	req := &showpb.EchoRequest{Response: &showpb.EchoRequest_Content{"testing"}}
	res := &showpb.EchoResponse{}

	err := client.Do("http://localhost:1337", "google.showcase.v1beta1.Echo", "Echo", req, res, nil)
	if err != nil {
		panic(err)
	}

	fmt.Println(res)
}

Authenticated Client Usage Example

package main

import (
	"context"
	"fmt"
	"net/http"

	"github.com/googleapis/grpc-fallback-go/client"

	"golang.org/x/oauth2/google"
	lpb "google.golang.org/genproto/googleapis/cloud/language/v1"
)

func main() {
	hdr := make(http.Header)
	ts, _ := google.DefaultTokenSource(context.Background(),
		"https://www.googleapis.com/auth/cloud-platform")
	t, _ := ts.Token()
	hdr.Add("Authorization", "Bearer "+t.AccessToken)

	req := &lpb.AnalyzeSentimentRequest{
		Document: &lpb.Document{
			Type:   lpb.Document_PLAIN_TEXT,
			Source: &lpb.Document_Content{"hello, world"},
		},
	}

	{
		// Here we use fallback-proxy to call a production Google service.
		// This expects that fallback-proxy has been started with the following command:
		//   fallback-proxy -address language.googleapis.com:443
		fmt.Println("\nCalling sentiment analysis using fallback-proxy...")
		res := &lpb.AnalyzeSentimentResponse{}
		err := client.Do("http://localhost:1337",
			"google.cloud.language.v1.LanguageService",
			"AnalyzeSentiment", req, res, hdr)
		if err != nil {
			fmt.Println(err)
		} else {
			fmt.Println(res)
		}
	}

	{
		// The service we just called is also directly available using grpc-fallback.
		// Here, for comparison, we call it directly.
		fmt.Println("\nCalling sentiment analysis directly with grpc-fallback...")
		res := &lpb.AnalyzeSentimentResponse{}
		err := client.Do("https://language.googleapis.com:443",
			"google.cloud.language.v1.LanguageService",
			"AnalyzeSentiment", req, res, hdr)
		if err != nil {
			fmt.Println(err)
		} else {
			fmt.Println(res)
		}
	}
}

Testing

If developing the grpc-fallback-go project, run tests via make test.

Releasing

To make a release, create a new tag with the form vX.Y.Z and push it using git push --tags. GitHub Actions will create the release and the appropriate assets.

Contributing

If you are planning to contribute to grpc-fallback-go, please read the CONTRIBUTING.md located in the root directory, as well as the Testing and Releasing sections.

Disclaimer

This is not an official Google product.