Protobuf for Message Payloads in Messaging
NATS message payloads are byte slices, so any kind of serialization strategy can be applied. This example shows a simple way to define message types using the Protocol Buffers IDL, generate code for the target language, and then use it with NATS.
If you are new to Protobuf, you can get started using one of the official tutorials.
The protobuf file for example looks as follows:
syntax = "proto3";
option go_package = ".;main";
package main;
message GreetRequest {
string name = 1;
}
message GreetReply {
string text = 1;
}
Click the link to the example’s source code to view the generated code.
Code
package main
import (
"fmt"
"os"
"time"
"github.com/nats-io/nats.go"
"google.golang.org/protobuf/proto"
)
func main() {
Use the env varibale if running in the container, otherwise use the default.
url := os.Getenv("NATS_URL")
if url == "" {
url = nats.DefaultURL
}
Create an unauthenticated connection to NATS.
nc, _ := nats.Connect(url)
defer nc.Drain()
Create an async subscription that will respond to greet
requests.
This will unmarshal the request and reply with a new message.
nc.Subscribe("greet", func(msg *nats.Msg) {
var req GreetRequest
proto.Unmarshal(msg.Data, &req)
rep := GreetReply{
Text: fmt.Sprintf("hello %q!", req.Name),
}
data, _ := proto.Marshal(&rep)
msg.Respond(data)
})
Allocate the request type which is a generate protobuf type.
and Marshal it using the proto
package.
req := GreetRequest{
Name: "joe",
}
data, _ := proto.Marshal(&req)
Messages are published to subjects. Although there are no subscribers, this will be published successfully.
msg, _ := nc.Request("greet", data, time.Second)
var rep GreetReply
proto.Unmarshal(msg.Data, &rep)
fmt.Printf("reply: %s\n", rep.Text)
}
Output
Network 60e7166b_default Creating Network 60e7166b_default Created Container 60e7166b-nats-1 Creating Container 60e7166b-nats-1 Created Container 60e7166b-nats-1 Starting Container 60e7166b-nats-1 Started reply: hello "joe"!