DEV Community

Cover image for Build a gRPC API using Go and gRPC-gateway
alisdairbr for Koyeb

Posted on • Originally published at koyeb.com

Build a gRPC API using Go and gRPC-gateway

gRPC, developed by Google, is a modern high-performance RPC (Remote Procedure Calls) framework widely used in today's microservices-oriented landscape. gRPC uses protobufs as its underlying message interchange format and leverages HTTP/2, enabling features such as multiplexing and bi-directional streaming. While gRPC is compatible with various programming languages, Go is particularly prevalent and recognized as the most commonly used and ergonomic option.

In this guide, we'll begin by exploring the fundamentals of gRPC, understanding its purpose and use cases. Following that, we'll look at how Protocol Buffers work and how to write message definitions and generate code for those definitions. Next, we'll dive into creating gRPC service definitions and we'll proceed to writing a simple microservice API for an order service within the context of an e-commerce platform.

Simply creating a microservice isn't enough; ideally, we also want to consume that service. For that, we'll create a REST-based API gateway service created using gRPC-gateway that can invoke methods on the newly implemented gRPC service and return responses to users in JSON format.

In the end, we'll deploy both services to Koyeb and you'll get to see how easy it is to deploy APIs and leverage service-to-service communication in Koyeb!

You can deploy and preview the applications from this guide by clicking the Deploy to Koyeb buttons below. View the application repository to view the project files and follow along with the guide.

To deploy the gRPC API, use the following button:

Deploy to Koyeb

Afterwards, to deploy the HTTP gateway, click this button:

Deploy to Koyeb

Requirements:

The requirements for the projects are the following:

RPC in a nutshell

Before diving into the tutorial, let's quickly discuss what RPC and an IDL (Interface Definition Language) are before looking into gRPC specific details.

In conventional REST-based architecture, an HTTP server registers endpoints to determine which handlers to invoke based on the URL path, HTTP verb (GET, POST, PUT), and path parameters. In contrast, RPC offers a high-level abstraction enabling clients to invoke remote methods on an HTTP server as if they were local method calls.

RPCs commonly rely on an IDL, a specification outlining the structure and communication protocols. In the RPC context, payload formats and service definitions are defined using serializable languages like Protobuf, Apache Thrift, Apache Avro, and others. These definitions are then used to generate corresponding implementations for a specific general-purpose programming language, such as Go, Java, Python, etc. These implementations can then be integrated into an RPC framework like gRPC, enabling us to create a web server and a corresponding client capable of communicating with the created web server.

The below flowchart provides a general idea of what an RPC framework does:

gRPC flowchart

Even though this may seem like magic, under the hood, the communication happens via HTTP, and its abstracted away from the user as below:

gRPC message processing

gRPC as a framework

gRPC is an RPC framework as described above, and it's one of the most widely-used RPC frameworks at the moment. Some essential information regarding gRPC includes:

  • It uses Protocol Buffers as the IDL for writing messages and service definitions.
  • Underlying communication is done over HTTP/2, which supports multiplexing multiple requests and responses over a single connection. This reduces latency compared to multiple HTTP/1.1 connections.
  • It provides a standardized way of handling errors with detailed status codes and messages, making it easier for developers to understand and handle failures.
  • It supports a wide variety of languages such as Go, Java, Node, C++ and more.

Steps

Now that we looked at a bit of theory, let's implement a gPRC server and an accompanying gateway server. We'll do this through the following steps:

  1. Install protobuf compiler
  2. Initialize the Go project
  3. Message definitions and code generation
  4. Service definitions and code generation
  5. Service implementation
  6. Set up the API Gateway
  7. Finish the rest of the app and test using postman
  8. Docker setup
  9. Push your project to GitHub
  10. Deploy the services to Koyeb

Install protobuf compiler

To compile implementations for the message and definition services that we write in .proto files, we need to first have the Protocol Buffer compiler, protoc, installed in our system.

You can install it with a package manager under Linux or macOS using the following commands.

In apt-based Linux distributions like Debian and Ubuntu, you can install the compiler by typing:

apt install -y protobuf-compiler
protoc --version  # Ensure compiler version is 3+
Enter fullscreen mode Exit fullscreen mode

On macOS, using Homebrew, you can install by typing:

brew install protobuf
protoc --version  # Ensure compiler version is 3+
Enter fullscreen mode Exit fullscreen mode

If you'd like to build the protocol compiler from sources, or access older versions of the pre-compiled binaries, visit the Protocol Buffers downloads page.

Initialize the Go project

Next, let's set up the initial file structure for our Go project. We'll be using Go modules in our projects, so you can initialize a new Go project using the following commands:

mkdir example-go-grpc-gateway
cd example-go-grpc-gateway
go mod init github.com/koyeb/example-go-grpc-gateway # Substitute the repo name as you wish
Enter fullscreen mode Exit fullscreen mode

Note: Throughout this guide, you will see the above repository referenced in files and commands. Be sure to substitute your own repository name so that Go can successfully find and build your project.

You should now have a file called go.mod in the example-go-grpc-gateway directory. Check the Go version defined within. If it has three version components, remove the final component to expand the minimum version for greater compatibility:

// go.mod

. . .
// go 1.21.4
go 1.21
. . .
Enter fullscreen mode Exit fullscreen mode

We need to create a directory called proto to keep our protobuf file definitions and a another directory called protogen to keep our compiled files. It's good practice to have a dedicated sub directory for each language that you'd like to compile the proto files to, so we'll create a golang subdirectory within the protogen directory:

mkdir -p proto/{orders,product}
mkdir -p protogen/golang
Enter fullscreen mode Exit fullscreen mode

Your directory structure should look like this:

example-go-grpc-gateway
├── go.mod
├── proto
│   ├── orders
│   └── product
└── protogen
    └── golang
Enter fullscreen mode Exit fullscreen mode

Message definitions and code generation

Now, let's write our first message definition and generate the code for it! Create a file called order.proto in the proto/orders directory with the following contents:

// ./proto/orders/order.proto

syntax = "proto3";

option go_package = "github.com/koyeb/example-go-grpc-gateway/protogen/golang/orders";

import "product/product.proto";
import "google/api/date.proto";

message Order {
    uint64 order_id = 1 [ json_name = "order_id" ];
    uint64 customer_id = 2 [ json_name = "customer_id" ];
    bool is_active = 3 [ json_name = "is_active" ];
    repeated Product products = 4;
    google.type.Date order_date = 5 [ json_name = "order_date" ];
}
Enter fullscreen mode Exit fullscreen mode

Next, create a product.proto file in the proto/product directory and add the below message definitions:

// ./proto/product/product.proto

syntax = "proto3";

option go_package = "github.com/koyeb/example-go-grpc-gateway/protogen/golang/product";

enum ProductType {
    UNKNOWN = 0;
    FOOD = 1;
    DRINK = 2;
}

message Product {
    uint64 product_id = 1 [ json_name = "product_id" ];
    string product_name = 2 [ json_name = "product_name" ];
    ProductType product_type = 5 [ json_name = "product_type" ];
}
Enter fullscreen mode Exit fullscreen mode

A few observations can be made based on the definitions given above:

  • syntax refers to the set of rules that define the structure and format for describing protocol buffer message types and services.
  • The go_package option is used to specify the Go import path for the generated Go language bindings. Hence, the compiled code for order.proto will be a file with the path protogen/golang/orders/order.pb.go.
  • A message is a structured unit that represents data. The compiled Go code will be an equivalent struct type.
  • We specify message fields in the message definition by indicating the data type, field name, and a unique field number assigned to each field. This field number serves as a distinctive identifier, facilitating the processes of serialization and de-serialization. Each data type corresponds to an equivalent Go type. For instance, a uint64 in Protobuf corresponds to uint64 in Go.
  • Field names in JSON can optionally be specified, ensuring that the serialized messages align with the defined field names. For instance, while we employ camel case for our names, gRPC defaults to pascal case.
  • We can modularize the definitions by defining them in separate files and importing them as needed. We have created a Product definition and have imported it in Order.
  • Protobuf supports complex types such as arrays defined by the repeated keyword, Enums, Unions, and many more.
  • Google also provides a number of custom types that are not supported by protobuf out of the box, as seen in the order_date field.

To compile this code we need to copy the Date definition file and add it to our project. You can create a folder called google/api under the proto folder and copy the code under the filename date.proto.

mkdir -p proto/google/api
curl -L https://raw.githubusercontent.com/googleapis/googleapis/master/google/type/date.proto -o proto/google/api/date.proto
Enter fullscreen mode Exit fullscreen mode

Now our folder structure looks like this:

example-go-grpc-gateway
├── go.mod
├── proto
│   ├── google
│   │   └── api
│   │       └── date.proto
│   ├── orders
│   │   ├── order.proto
│   │   └── product.proto
│   └── product
└── protogen
    └── golang
Enter fullscreen mode Exit fullscreen mode

Now that we have our definitions, let's compile the code. Before doing so, we need to to install a binary to help the protobuf compiler generate Go-specific code. You can install it in your GOPATH using the following command:

go install google.golang.org/protobuf/cmd/protoc-gen-go@v1.28
Enter fullscreen mode Exit fullscreen mode

Now, create a Makefile and add the below line to compile the proto files.

# Makefile

protoc:
    cd proto && protoc --go_out=../protogen/golang --go_opt=paths=source_relative \
    ./**/*.proto
Enter fullscreen mode Exit fullscreen mode

With this command, we've defined the output directory for code generation using the --go_out flag. Additionally, we include the --go_opt option to specify that Go package paths should align with the directory structure relative to the source directory. The ./**/*.proto glob expands the current directory and includes all proto files for the compilation process.

Run the command by typing:

make protoc
Enter fullscreen mode Exit fullscreen mode

It should generate the appropriate code in the protogen/golang directory. If you look at the generated code however, you may notice red squiggly lines in your IDE, indicating that your project lacks some of the expected dependencies. To address this, import the following packages.

go get google.golang.org/protobuf # Go implementation for protocol buffers
go get google.golang.org/genproto # Contains the generated Go packages for common protocol buffer types
Enter fullscreen mode Exit fullscreen mode

Let's write some code to see the generated Order struct in action. Create a temporary main.go file in the root directory with the following code:

// main.go

package main

import (
    "fmt"
    "log"

    "google.golang.org/genproto/googleapis/type/date"
    "google.golang.org/protobuf/encoding/protojson"

    "github.com/koyeb/example-go-grpc-gateway/protogen/golang/orders"
    "github.com/koyeb/example-go-grpc-gateway/protogen/golang/product"
)

func main() {
    orderItem := orders.Order{
        OrderId:    10,
        CustomerId: 11,
        IsActive:   true,
        OrderDate:  &date.Date{Year: 2021, Month: 1, Day: 1},
        Products: []*product.Product{
            {ProductId: 1, ProductName: "CocaCola", ProductType: product.ProductType_DRINK},
        },
    }

    bytes, err := protojson.Marshal(&orderItem)
    if err != nil {
        log.Fatal("deserialization error:", err)
    }

    fmt.Println(string(bytes))
}
Enter fullscreen mode Exit fullscreen mode

The created order-item will be serialized to JSON using the protojson package.

You can run the code by typing:

go run main.go
Enter fullscreen mode Exit fullscreen mode

It will produce the following JSON output (expanded here for readability):

{
  "order_id": "10",
  "customer_id": "11",
  "is_active": true,
  "products": [{ "product_id": "1", "product_name": "CocaCola", "product_type": "DRINK" }],
  "order_date": { "year": 2021, "month": 1, "day": 1 }
}
Enter fullscreen mode Exit fullscreen mode

Note that, gRPC will typically serialize the messages in binary format, which is faster and takes less space compared to a text format like JSON.

As this was only for testing, you can remove the main.go file when you are finished:

rm main.go
Enter fullscreen mode Exit fullscreen mode

Service definitions and code generation

We've looked at how to create message/payload definitions using protobuf. Now, let's add the service (endpoints in REST) definitions to register in our gRPC server.

Open the order.proto file in the proto/orders directory again and add the following definitions to the end of the file:

// ./proto/orders/order.proto

. . .

// A generic empty message that you can re-use to avoid defining duplicated
// empty messages in your APIs
message Empty {}

message PayloadWithSingleOrder {
  Order order = 1;
}

service Orders {
  rpc AddOrder(PayloadWithSingleOrder) returns (Empty) {}
}
Enter fullscreen mode Exit fullscreen mode

Here, we've added a service definition to add a new order. It takes a payload with single order as an argument and returns an empty body.

To compile this service definition, it is important to have a gRPC-specific binary installed. You can be install it with the following command:

go install google.golang.org/grpc/cmd/protoc-gen-go-grpc@v1.2
Enter fullscreen mode Exit fullscreen mode

Let's modify the protoc command in our Makefile to generate gRPC code as well.

# Makefile

protoc:
    cd proto && protoc --go_out=../protogen/golang --go_opt=paths=source_relative \
    --go-grpc_out=../protogen/golang --go-grpc_opt=paths=source_relative \
    ./**/*.proto
Enter fullscreen mode Exit fullscreen mode

We have added two new arguments with --go-grpc_out and --go-grpc_opt.

Run protoc target again:

make protoc
Enter fullscreen mode Exit fullscreen mode

The output should now include a file with the path protogen/golang/orders/order_grpc.pb.go.

To make the generated code work in our system we need to install the following gRPC dependency:

go get google.golang.org/grpc
Enter fullscreen mode Exit fullscreen mode

Service implementation

If you look at the generated gRPC code in the protogen/golang/orders/order_grpc.pb.go file, you'll see the below interface defined.

// ./protogen/golang/orders/order_grpc.pb.go

. . .
// OrdersClient is the client API for Orders service.
//
// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream.
type OrdersClient interface {
    AddOrder(ctx context.Context, in *PayloadWithSingleOrder, opts ...grpc.CallOption) (*Empty, error)
}
. . .
Enter fullscreen mode Exit fullscreen mode

Our goal in this section is to create a structure which implements this interface and wire it up with a new gRPC server. We'll use the necessary file structure:

example-go-grpc-gateway
├── cmd
│   └── server
│       └── main.go
├── go.mod
├── go.sum
├── internal
│   ├── db.go
│   └── orderservice.go
├── Makefile
├── proto
│   ├── google
│   │   └── api
│   │       └── date.proto
│   ├── orders
│   │   └── order.proto
│   └── product
│       └── product.proto
└── protogen
    └── . . .
Enter fullscreen mode Exit fullscreen mode

To create the missing directories and files, type:

mkdir -p cmd/server internal
touch cmd/server/main.go internal/{db,orderservice}.go
Enter fullscreen mode Exit fullscreen mode

Next, open the internal/orderservice.go file and paste in the following contents:

// ./internal/orderservice.go

package internal

import (
    "context"
    "log"

    "github.com/koyeb/example-go-grpc-gateway/protogen/golang/orders"
)

// OrderService should implement the OrdersServer interface generated from grpc.
//
// UnimplementedOrdersServer must be embedded to have forwarded compatible implementations.
type OrderService struct {
    db *DB
    orders.UnimplementedOrdersServer
}

// NewOrderService creates a new OrderService
func NewOrderService(db *DB) OrderService {
    return OrderService{db: db}
}

// AddOrder implements the AddOrder method of the grpc OrdersServer interface to add a new order
func (o *OrderService) AddOrder(_ context.Context, req *orders.PayloadWithSingleOrder) (*orders.Empty, error) {
    log.Printf("Received an add-order request")

    err := o.db.AddOrder(req.GetOrder())

    return &orders.Empty{}, err
}
Enter fullscreen mode Exit fullscreen mode

The above code creates a struct called OrderService to implement the gRPC interface and we have added the same method signature as given in the interface definition for the AddOrder method. This method accepts an order from the request, stores it in a database, and returns an empty message along with any associated errors.

We can create a mock version of an in-memory database using an array to illustrate that we can utilize databases and other services exactly the same way as we would in a REST environment.

Place the following in the internal/db.go file:

// ./internal/db.go
package internal

import (
    "fmt"

    "github.com/koyeb/example-go-grpc-gateway/protogen/golang/orders"
)

type DB struct {
    collection []*orders.Order
}

// NewDB creates a new array to mimic the behaviour of a in-memory database
func NewDB() *DB {
    return &DB{
        collection: make([]*orders.Order, 0),
    }
}

// AddOrder adds a new order to the DB collection. Returns an error on duplicate ids
func (d *DB) AddOrder(order *orders.Order) error {
    for _, o := range d.collection {
        if o.OrderId == order.OrderId {
            return fmt.Errorf("duplicate order id: %d", order.GetOrderId())
        }
    }
    d.collection = append(d.collection, order)
    return nil
}
Enter fullscreen mode Exit fullscreen mode

Let's create the gRPC server and see if we can register the OrderService that we have created above.

Add the following to the cmd/server/main.go file:

// ./cmd/server/main.go

package main

import (
    "log"
    "net"

    "github.com/koyeb/example-go-grpc-gateway/internal"
    "github.com/koyeb/example-go-grpc-gateway/protogen/golang/orders"
    "google.golang.org/grpc"
)

func main() {
    const addr = "0.0.0.0:50051"

    // create a TCP listener on the specified port
    listener, err := net.Listen("tcp", addr)
    if err != nil {
        log.Fatalf("failed to listen: %v", err)
    }

    // create a gRPC server instance
    server := grpc.NewServer()

    // create a order service instance with a reference to the db
    db := internal.NewDB()
    orderService := internal.NewOrderService(db)

    // register the order service with the grpc server
    orders.RegisterOrdersServer(server, &orderService)

    // start listening to requests
    log.Printf("server listening at %v", listener.Addr())
    if err = server.Serve(listener); err != nil {
        log.Fatalf("failed to serve: %v", err)
    }
}
Enter fullscreen mode Exit fullscreen mode

The code above starts a gRPC server listening on port 50051 using the mock database we created. You can run it by typing:

go run cmd/server/main.go
Enter fullscreen mode Exit fullscreen mode

After compilation, the server will start. This means that you have successfully created a service definition, generated the corresponding code, implemented a service based on those definitions, registered the service, and initialized a gRPC server!

Though the server is running, you can confirm that the server doesn't respond to HTTP requests by making a request with curl:

curl 127.0.0.1:50051
Enter fullscreen mode Exit fullscreen mode

You should receive a message similar to this:

curl: (1) Received HTTP/0.9 when not allowed
Enter fullscreen mode Exit fullscreen mode

Unfortunately, it's not easy to test a gRPC server like a REST server by using tools like browsers, Postman, or curl.

While there are tools available for testing gRPC servers, we'll instead create an API gateway server to demonstrate how we can invoke methods in a manner similar to the REST paradigm.

Set up the API Gateway

Reasons for using an API Gateway can range from maintaining backward-compatibility, supporting languages or clients that are not well-supported by gRPC, or simply maintaining the aesthetics and tooling associated with a RESTful JSON architecture.

The diagram below shows one way that REST and gRPC can co-exist in the same system:

client request using gRPC gateway

Fortunately for our purposes, Google has a library called grpc-gateway that we can use to simplify the process of setting up a reverse proxy. It will act as a HTTP+JSON interface to the gRPC service. All that we need is a small amount of configuration to attach HTTP semantics to the service and it will be able to generate the necessary code.

To help generate the gateway code, we require two additional binaries:

go get github.com/grpc-ecosystem/grpc-gateway/v2/protoc-gen-grpc-gateway
go get github.com/grpc-ecosystem/grpc-gateway/v2/protoc-gen-openapiv2
go install github.com/grpc-ecosystem/grpc-gateway/v2/protoc-gen-grpc-gateway
go install github.com/grpc-ecosystem/grpc-gateway/v2/protoc-gen-openapiv2
Enter fullscreen mode Exit fullscreen mode

As mentioned above, we need to make a few small adjustments to our service definition to make this work. But before we do that, we need to add two new files into our proto/google/api folder, namely annotations.proto and http.proto:

curl -L https://raw.githubusercontent.com/googleapis/googleapis/master/google/api/annotations.proto -o proto/google/api/annotations.proto
curl -L https://raw.githubusercontent.com/googleapis/googleapis/master/google/api/http.proto -o proto/google/api/http.proto
Enter fullscreen mode Exit fullscreen mode

The proto directory should now look like this:

example-go-grpc-gateway/proto
├── google
│   └── api
│       ├── annotations.proto
│       ├── date.proto
│       └── http.proto
├── orders
│   └── order.proto
└── product
    └── product.proto
Enter fullscreen mode Exit fullscreen mode

Next modify the proto/orders/orders.proto file to add the gateway server changes. The new contents look like this:

// ./proto/orders/orders.proto

syntax = "proto3";

option go_package = "github.com/koyeb/example-go-grpc-gateway/protogen/golang/orders";

import "product/product.proto";
import "google/api/annotations.proto";
import "google/api/date.proto";

message Order {
    ...
}

// A generic empty message that you can re-use to avoid defining duplicated
// empty messages in your APIs
message Empty {}

message PayloadWithSingleOrder {
  Order order = 1;
}

service Orders {
  rpc AddOrder(PayloadWithSingleOrder) returns (Empty) {
    option (google.api.http) = {
      post: "/v0/orders",
      body: "*"
    };
  }
}
Enter fullscreen mode Exit fullscreen mode

Notice how we've designated AddOrder as a POST endpoint with the path as /v0/orders and body specified as "*". This indicates that the entire request body will be utilized as input for the AddOrder invocation.

Next, let's modify our Makefile and add the new gRPC gateway options to our existing protoc command.

# Makefile

protoc:
    cd proto && protoc --go_out=../protogen/golang --go_opt=paths=source_relative \
    --go-grpc_out=../protogen/golang --go-grpc_opt=paths=source_relative \
    --grpc-gateway_out=../protogen/golang --grpc-gateway_opt paths=source_relative \
    --grpc-gateway_opt generate_unbound_methods=true \
    ./**/*.proto
Enter fullscreen mode Exit fullscreen mode

Use the Makefile to generate the necessary code again by typing:

make protoc
Enter fullscreen mode Exit fullscreen mode

A new file will be created with the path protogen/golang/orders/order.pb.gw.go. If you take a peek at the generated code, you'll see a function called RegisterOrdersHandlerServer, with a function body that resembles a typical REST handler register that we'd write in Go.

Now that we have successfully generated the handler code, let's create the API gateway server. Create a cmd/client directory and then create a new file with the path cmd/client/main.go:

mkdir cmd/client
touch cmd/client/main.go
Enter fullscreen mode Exit fullscreen mode

Note that we've named this directory as client because it essentially serves as a client to invoke gRPC methods on the order server.

Add the following code to the cmd/client/main.go file:

// ./cmd/client/main.go

package main

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

    "github.com/grpc-ecosystem/grpc-gateway/v2/runtime"
    "github.com/koyeb/example-go-grpc-gateway/protogen/golang/orders"
    "google.golang.org/grpc"
    "google.golang.org/grpc/credentials/insecure"
)

func main() {
    // Set up a connection to the order server.
    orderServiceAddr := "localhost:50051"
    conn, err := grpc.Dial(orderServiceAddr, grpc.WithTransportCredentials(insecure.NewCredentials()))
    if err != nil {
        log.Fatalf("could not connect to order service: %v", err)
    }
    defer conn.Close()

    // Register gRPC server endpoint
    // Note: Make sure the gRPC server is running properly and accessible
    mux := runtime.NewServeMux()
    if err = orders.RegisterOrdersHandler(context.Background(), mux, conn); err != nil {
        log.Fatalf("failed to register the order server: %v", err)
    }

    // start listening to requests from the gateway server
    addr := "0.0.0.0:8080"
    fmt.Println("API gateway server is running on " + addr)
    if err = http.ListenAndServe(addr, mux); err != nil {
        log.Fatal("gateway server closed abruptly: ", err)
    }
}
Enter fullscreen mode Exit fullscreen mode

We initiated a connection to the gRPC server running on localhost:50051 and established a new HTTP server running on 0.0.0.0:8080. This server is configured to receive requests and execute the relevant gRPC methods for the orders service.

We can test this by creating a payload file called data.json with the following content:

{
  "order": {
    "order_id": "14",
    "customer_id": "11",
    "is_active": true,
    "products": [
      {
        "product_id": "1",
        "product_name": "CocaCola",
        "product_type": "DRINK"
      }
    ],
    "order_date": {
      "year": 2023,
      "month": 11,
      "day": 26
    }
  }
}
Enter fullscreen mode Exit fullscreen mode

Start up both services by executing the following in two separate terminal windows:

go run cmd/server/main.go
go run cmd/client/main.go
Enter fullscreen mode Exit fullscreen mode

Now, in a third terminal, send the payload to the server by typing:

curl -d "@data.json" -X POST -i http://localhost:8080/v0/orders
Enter fullscreen mode Exit fullscreen mode

You should receive a 200 status message indicating that the payload was accepted:

HTTP/1.1 200 OK
Content-Type: application/json
Grpc-Metadata-Content-Type: application/grpc
Date: Fri, 05 Jan 2024 12:03:49 GMT
Content-Length: 2

{}
Enter fullscreen mode Exit fullscreen mode

The gateway server will log the request as well:

API gateway server is running on 0.0.0.0:8080
2024/01/11 13:15:29 Received an add-order request
Enter fullscreen mode Exit fullscreen mode

The entire flow is operational. Now, its time to add a few more CRUD operations and run a postman test suite to see if we can get all the Postman tests to pass.

Finish the rest of the app and test using postman

Let's add a few more CRUD methods to our Orders service to get a complete picture.

We'll start by modifying our proto/orders/order.proto file with few added definitions.

// ./proto/orders/order.proto
. . .

message PayloadWithOrderID {
  uint64 order_id = 1;
}

. . .

service Orders {
  . . .

  rpc GetOrder(PayloadWithOrderID) returns (PayloadWithSingleOrder) {
    option (google.api.http) = {
      get: "/v0/orders/{order_id}",
    };
  }

  rpc UpdateOrder(PayloadWithSingleOrder) returns (Empty) {
    option (google.api.http) = {
      put: "/v0/orders",
      body: "*"
    };
  }

  rpc RemoveOrder(PayloadWithOrderID) returns (Empty) {
    option (google.api.http) = {
      delete: "/v0/orders/{order_id}",
    };
  }
}
Enter fullscreen mode Exit fullscreen mode

Notice how we have added GetOrder endpoint with the path /v0/orders/{order_id} which includes a path parameter.

Next we'll update our in-memory db to add few more methods. Open the internal/db.go file and add the following functions to the end of the file:

// ./internal/db.go
. . .

// GetOrderByID returns an order by the order_id
func (d *DB) GetOrderByID(orderID uint64) *orders.Order {
    for _, o := range d.collection {
        if o.OrderId == orderID {
            return o
        }
    }
    return nil
}

// GetOrdersByIDs returns all orders pertaining to the given order ids
func (d *DB) GetOrdersByIDs(orderIDs []uint64) []*orders.Order {
    filtered := make([]*orders.Order, 0)

    for _, idx := range orderIDs {
        for _, order := range d.collection {
            if order.OrderId == idx {
                filtered = append(filtered, order)
                break
            }
        }
    }

    return filtered
}

// UpdateOrder updates an order in place
func (d *DB) UpdateOrder(order *orders.Order) {
    for i, o := range d.collection {
        if o.OrderId == order.OrderId {
            d.collection[i] = order
            return
        }
    }
}

// RemoveOrder removes an order from the orders collection
func (d *DB) RemoveOrder(orderID uint64) {
    filtered := make([]*orders.Order, 0, len(d.collection)-1)
    for i := range d.collection {
        if d.collection[i].OrderId != orderID {
            filtered = append(filtered, d.collection[i])
        }
    }
    d.collection = filtered
}
Enter fullscreen mode Exit fullscreen mode

Finally, we'll add the implementations for the newly added RPC methods in our internal/orderservice.go file. Replace the file contents with the following code:

// ./internal/orderservice.go

package internal

import (
    "context"
    "fmt"
    "log"

    "github.com/koyeb/example-go-grpc-gateway/protogen/golang/orders"
)

// OrderService should implement the OrdersServer interface generated from grpc.
//
// UnimplementedOrdersServer must be embedded to have forwarded compatible implementations.
type OrderService struct {
    db *DB
    orders.UnimplementedOrdersServer
}

// NewOrderService creates a new OrderService
func NewOrderService(db *DB) OrderService {
    return OrderService{db: db}
}

// AddOrder implements the AddOrder method of the grpc OrdersServer interface to add a new order
func (o *OrderService) AddOrder(_ context.Context, req *orders.PayloadWithSingleOrder) (*orders.Empty, error) {
    log.Printf("Received an add order request")

    err := o.db.AddOrder(req.GetOrder())

    return &orders.Empty{}, err
}

// GetOrder implements the GetOrder method of the grpc OrdersServer interface to fetch an order for a given orderID
func (o *OrderService) GetOrder(_ context.Context, req *orders.PayloadWithOrderID) (*orders.PayloadWithSingleOrder, error) {
    log.Printf("Received get order request")

    order := o.db.GetOrderByID(req.GetOrderId())
    if order == nil {
        return nil, fmt.Errorf("order not found for orderID: %d", req.GetOrderId())
    }

    return &orders.PayloadWithSingleOrder{Order: order}, nil
}

// UpdateOrder implements the UpdateOrder method of the grpc OrdersServer interface to update an order
func (o *OrderService) UpdateOrder(_ context.Context, req *orders.PayloadWithSingleOrder) (*orders.Empty, error) {
    log.Printf("Received an update order request")

    o.db.UpdateOrder(req.GetOrder())

    return &orders.Empty{}, nil
}

// RemoveOrder implements the RemoveOrder method of the grpc OrdersServer interface to remove an order
func (o *OrderService) RemoveOrder(_ context.Context, req *orders.PayloadWithOrderID) (*orders.Empty, error) {
    log.Printf("Received a remove order request")

    o.db.RemoveOrder(req.GetOrderId())

    return &orders.Empty{}, nil
}
Enter fullscreen mode Exit fullscreen mode

Rerun the Makefile to generate the new files:

make protoc
Enter fullscreen mode Exit fullscreen mode

The great thing about our gateway service is that all of these new endpoints work seamlessly without needing to add additional code.

In the repository for this project, you can find a test suite for Postman that you can optionally use to test the whole flow in an end-to-end fashion.

If you download this Postman collection and import it, you should be able to see all our tests passing with flying colors. Just set the gateway-service-url variable to http://localhost:8080 when you run the tests:

gRPC Postman testing

Docker setup

At the moment, we have to run and shut down the two services in two different terminal windows. Let's improve the developer experience by incorporating Docker and docker-compose in our setup. This will also prove highly beneficial in the upcoming deployment of services through Koyeb.

Add the below Dockerfile to the root of your project:

FROM golang:1.19-alpine AS builder
ARG ORDER_SERVICE_ADDRESS
WORKDIR /app
COPY . .
RUN go mod download
RUN go build -o ./orders-service ./cmd/server/main.go
RUN go build -ldflags  "-X main.orderServiceAddr=$ORDER_SERVICE_ADDRESS" -o ./gateway-service ./cmd/client/main.go

FROM alpine:latest AS orders-service
WORKDIR /app
COPY --from=builder /app/orders-service .
EXPOSE 50051
ENTRYPOINT ["./orders-service"]

FROM alpine:latest AS gateway-service
WORKDIR /app
COPY --from=builder /app/gateway-service .
EXPOSE 8080
ENTRYPOINT ["./gateway-service"]
Enter fullscreen mode Exit fullscreen mode

The file above builds the two servers as two Docker targets using multi-stage builds.

It also includes a ORDER_SERVICE_ADDRESS build argument that is passed to go build as an ldflag (linker flag). These linker flags expose variables during the compilation process. Let's modify our gateway service code to accommodate this ORDER_SERVICE_ADDRESS variable.

Modify the cmd/client/main.go file so that the orderServiceAddr is not hard-coded:

// ./cmd/client/main.go

package main

import (
    . . .
)

var orderServiceAddr string

func main() {
    // Set up a connection to the order server.
    fmt.Println("Connecting to order service via", orderServiceAddr)
    conn, err := grpc.Dial(orderServiceAddr, grpc.WithTransportCredentials(insecure.NewCredentials()))
    . . .
Enter fullscreen mode Exit fullscreen mode

We can also add a docker-compose file to serve both services locally. Create a docker-compose.yaml file in the project root directory with the following configuration:

# docker-compose.yaml
version: '3.8'
services:
  orders-service:
    image: orders-service
    build:
      context: .
      dockerfile: Dockerfile
      target: orders-service
    ports:
      - '50051/tcp'
    cpus: 0.125
    mem_limit: 128m
  gateway-service:
    image: gateway-service
    build:
      context: .
      dockerfile: Dockerfile
      target: gateway-service
      args:
        ORDER_SERVICE_ADDRESS: orders-service:50051
    ports:
      - '8080:8080'
    cpus: 0.125
    mem_limit: 128m
Enter fullscreen mode Exit fullscreen mode

We've specified build targets in each service that we define and set the ORDER_SERVICE_ADDRESS variable to order-service (the service name of the order service) followed by the port number. This works because Compose enables services to discover and communicate with each other using service names as hostnames.

You can start both services by typing:

docker-compose up -d
Enter fullscreen mode Exit fullscreen mode

This will build the images and start the services. You can run the postman test suite just as before without any additional changes.

Push your project to GitHub

If you haven't done so already, create a new repository for your project on GitHub. Now we can initialize a new Git repository for the project, commit our changes, and push them to the new GitHub repo:

cd example-go-grpc-gateway
git init
git add :/
git commit -m "Initial commit"
git remote add origin git@github.com:<YOUR_GITHUB_USERNAME>/<YOUR_REPOSITORY_NAME>.git
git branch -M main
git push -u origin main
Enter fullscreen mode Exit fullscreen mode

Your project files should now be synced up to GitHub, ready to deploy.

Deploy the services to Koyeb

Next, let's proceed to the most exiting part of our tutorial: deploying the newly crafted services and seeing them in action.

Deploy the orders service

To deploy the orders-service, open the Koyeb control panel and complete the following steps:

  1. Click Create App to begin configuration.
  2. Select GitHub as the deployment method.
  3. Select your project from the GitHub repository list and choose the appropriate branch.
  4. Set the name of the service to orders-service.
  5. Select Dockerfile in the "Choose your builder" section.
  6. Expand the Build and deployment settings section. Inside, click the override toggle associated with the Target option and enter orders-service in the field.
  7. Expand the Advanced and Exposing your service sections and set the port to 50051. De-select the Public option to mark the service as internal.
  8. Set the App name to orders-service.
  9. Click Deploy to begin the deployment process.

Once the orders service is up and running, copy its service URL, which we'll use next when we deploy the gateway service. Koyeb provides built-in service discovery, streamlining the process of connecting to other internal services within your application without requiring additional configuration.

Deploy the gRPC gateway service

Let's deploy our Gateway service now. Return to the Koyeb control panel and complete the following steps:

  1. Click Create App to begin configuration.
  2. Select GitHub as the deployment method.
  3. Select your project from the GitHub repository list and choose the appropriate branch.
  4. Set the name of the service to gateway-service.
  5. Select Dockerfile in the "Choose your builder" section.
  6. Expand the Build and deployment settings section. Inside, click the override toggle associated with the Target option and enter gateway-service in the field. Click the override toggle associated with the Args option and set the value to ["ORDER_SERVICE_ADDRESS"].
  7. Expand the Advanced and Environment variable sections and add a new environment variable called ORDER_SERVICE_ADDRESS with the private address where your order service can be reached. It should follow this format: <SERVICE_NAME>.<APP_NAME>.koyeb:50051.
  8. Expand the Exposing your service sections and set the port to 8080.
  9. Set the App name to gateway-service. This determines where the application will be deployed to. For example, https://gateway-service-<YOUR_USERNAME>.koyeb.app.
  10. Click Deploy to begin the deployment process.

Once the gateway service is also fully deployed, you can run the Postman test suite again after changing the gateway-service-url variable to the public domain of the gateway app. The tests should succeed again!

Conclusions

In this tutorial, we've explored the process of creating gRPC services in Go and developing a corresponding gateway service. Subsequently, we deployed these services in Koyeb, illustrating how to establish communication between public and private services.

While we've primarily focused on unary operations, the simplest type of RPC involving a single request and response, gRPC offers a rich set of features including server/client/bidirectional streaming, configuring deadlines and timeouts, handling gRPC channels, and more. Take this guide as a starting point to delve into the broader spectrum of gRPC's capabilities.

Top comments (0)