Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add documentation for gRPC handlers using gofr-cli #1379

Closed
Closed
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
226 changes: 125 additions & 101 deletions docs/advanced-guide/grpc/page.md
Original file line number Diff line number Diff line change
@@ -1,125 +1,149 @@
# gRPC
We have already seen how GoFr can help ease the development of HTTP servers, but there are
cases where performance is primarily required sacrificing flexibility. In these types of
scenarios gRPC protocol comes into picture. {% new-tab-link title="gRPC" href="https://grpc.io/docs/what-is-grpc/introduction/" /%} is an open-source RPC(Remote Procedure Call)
framework initially developed by Google.
# gRPC with Gofr

We have already seen how GoFr can help ease the development of HTTP servers, but there are cases where performance is primarily required sacrificing flexibility. In these types of scenarios gRPC protocol comes into picture. {% new-tab-link title="gRPC" href="https://grpc.io/docs/what-is-grpc/introduction/" /%} is an open-source RPC(Remote Procedure Call) framework initially developed by Google.

Gofr empowers you to create gRPC handlers efficiently while taking advantage of context support for effective dependency and tracing management within your handlers.

## Prerequisites
- Install the `protoc` protocol buffer compilation
- Linux, using `apt` or `apt-get`
```shell
$ apt install -y protobuf-compiler
$ protoc --version # Ensure compiler version is 3+
```
- macOS, using Homebrew
```shell
$ brew install protobuf
$ protoc --version # Ensure compiler version is 3+
```
- Install **Go Plugins** for protocol compiler:
1. Install protocol compiler plugins for Go
```shell
$ go install google.golang.org/protobuf/cmd/protoc-gen-go@v1.28
$ go install google.golang.org/grpc/cmd/protoc-gen-go-grpc@v1.2
```
2. Update `PATH` for `protoc` compiler to find the plugins:
```shell
$ export PATH="$PATH:$(go env GOPATH)/bin"
```

## Creating protocol buffers
For a detailed guide, please take a look at the {% new-tab-link title="Tutorial" href="https://grpc.io/docs/languages/go/basics/" /%} at official gRPC docs.

We need to create a `customer.proto` file to define our service and the RPC methods that the service provides.
```protobuf
// Indicates the protocol buffer version that is being used
syntax = "proto3";
// Indicates the go package where the generated file will be produced
option go_package = "";

service CustomerService {
// ...
}
```
Inside the service one can define all the `rpc` methods, specifying the request and responses types.
```protobuf
service CustomerService {
// GetCustomer is a rpc method to get customer data using specific filters
rpc GetCustomer(CustomerFilter) returns(CustomerData) {}
}

**1. Protocol Buffer Compiler (`protoc`) Installation:**

- **Linux (using `apt` or `apt-get`):**
```bash
sudo apt install -y protobuf-compiler
protoc --version # Ensure compiler version is 3+
```

- **macOS (using Homebrew):**

```bash
brew install protobuf
protoc --version # Ensure compiler version is 3+
```
The `CustomerFilter` and `CustomerData` are two types of messages that will be exchanged between server
and client. Users must define those for protocol buffer to serialize them when making a remote procedure call.
```protobuf
syntax = "proto3";

**2. Go Plugins for Protocol Compiler:**

a. Install protocol compiler plugins for Go:

```bash
go install google.golang.org/protobuf/cmd/protoc-gen-go@v1.28
go install google.golang.org/grpc/cmd/protoc-gen-go-grpc@v1.2
```

b. Update `PATH` for `protoc` to locate the plugins:

```bash
export PATH="$PATH:$(go env GOPATH)/bin"
```

## Creating Protocol Buffers

For a detailed guide, refer to the official gRPC documentation's tutorial: {% new-tab-link title="Tutorial" href="https://grpc.io/docs/languages/go/basics/" /%} at official gRPC docs.

**1. Define Your Service and RPC Methods:**

Create a `.proto` file (e.g., `customer.proto`) to define your service and the RPC methods it provides:

```protobuf
// Indicates the protocol buffer version that is being used
syntax = "proto3";
// Indicates the go package where the generated file will be produced
// assuming proto file is defined in package with path github.com/grpc/customer
option go_package = "github.com/grpc/customer";

service CustomerService {
rpc GetCustomer (CustomerFilter) returns (CustomerData) {}
}
```

**2. Specify Request and Response Types:**

For example: The CustomerFilter and CustomerData are two types of messages that will be exchanged between server and client. Users must define those for protocol buffer to serialize them when making a remote procedure call.

```go
message CustomerFilter {
int64 id = 1;
string name = 2;
// other fields that can be passed
int64 id = 1;
string name = 2;
// other fields that can be passed
}

message CustomerData {
int64 id = 1;
string name = 2;
string address = 3;
// other customer related fields
int64 id = 1;
string name = 2;
string address = 3;
// other customer related fields
}
```
coolwednesday marked this conversation as resolved.
Show resolved Hide resolved

Now run the following command to generate go code using the Go gRPC plugins:
```shell
protoc \
--go_out=. \
--go_opt=paths=source_relative \
--go-grpc_out=. \
--go-grpc_opt=paths=source_relative \
customer.proto
**3. Generate Go Code:**

Run the following command to generate Go code using the Go gRPC plugins:

```bash
protoc \
--go_out=. \
--go_opt=paths=source_relative \
--go-grpc_out=. \
--go-grpc_opt=paths=source_relative \
customer.proto
```

This command generates two files, `customer.pb.go` and `customer_grpc.pb.go`, containing the necessary code for performing RPC calls.

## Generating gRPC Handler Template using `gofr wrap grpc`

#### Prerequisite: gofr-cli must be installed

To install the CLI -
```bash
go install gofr.dev/cli/gofr@latest
```
Above command will generate two files `customer.pb.go` and `customer_grpc.pb.go` and these contain necessary code to perform RPC calls.
In `customer.pb.go` you can find `CustomerService` interface-
```go
// CustomerServiceServer is the server API for CustomerService service.
type CustomerServiceServer interface {
GetCustomer(context.Context, *CustomerFilter) (*CustomerData, error)
}

To check the installation -
```bash
gofr version
```
User needs to implement this interface to serve the content to the client calling the method.
```go
package customer

import (
"context"
)

type Handler struct {
// required fields to get the customer data
}
**1. Use the `gofr wrap grpc` Command:**

func (h *Handler) GetCustomer(ctx context.Context, filter *CustomerFilter) (*CustomerData, error) {
// get the customer data and handler error
return data, nil
}
```
```bash
gofr wrap grpc -proto=./grpc/customer.proto
```

Lastly to register the gRPC service to the GoFr server, user can call the `RegisterCustomerServiceServer` in `customer_grpc.pb.go`
to register the service giving GoFr app and the Handler struct.
```go
package main
This command leverages the `gofr-cli` to generate a `{serviceName}Server.go` file (e.g., `CustomerServer.go`)
containing a template for your gRPC server implementation, including context support, in the same directory as
that of the specified proto file.

import (
"gofr.dev/pkg/gofr"
"gofr.dev/examples/grpc-server/customer"
)
**2. Modify the Generated Code:**

func main() {
app := gofr.New()
- Customize the `CustomerGoFrServer` struct with required dependencies and fields.
- Implement the `GetCustomer` method to handle incoming requests, as required in this usecase:
- Bind the request payload using `ctx.Bind(&request)`.
- Process the request and generate a response.

customer.RegisterCustomerServiceServer(app, customer.Handler{})
## Registering the gRPC Service with Gofr

app.Run()
}
```
**1. Import Necessary Packages:**

```go
import (
"gofr.dev/pkg/gofr"
"path/to/your/generated-grpc-server/packageName"
)
```

**2. Register the Service in your `main.go`:**

```go
func main() {
app := gofr.New()

packageName.RegisterCustomerServerWithGofr(app, &customer.CustomerGoFrServer{})

app.Run()
}
```
>Note: By default, gRPC server will run on port 9000, to customize the port users can set `GRPC_PORT` config in the .env

> ##### Check out the example of setting up a gRPC server in GoFr: [Visit GitHub](https://github.com/gofr-dev/gofr/blob/main/examples/grpc-server/main.go)