GRPC is a remote call framework that uses Protobuf as the information carrier to complete data transmission between client and server. How to define Protobuf messages and build gRPC services has been covered in the previous series of articles. Today we will look at how to pass dynamic parameters when using gRPC and Protobuf.

First of all, when we say dynamic parameter, we mean a compound type field whose content is not determined at the time of defining a Protobuf message, basically the field in the message. We want to pass a combination of values like JSON object, Map dictionary, structure, etc., However, Protobuf message fields are called dynamic parameters because they are not defined when defining messages, which fields are in JSON, what type each field value is, or what type the Map dictionary key value is.

There is no standard solution for delivering dynamic parameters via Protobuf. The bytes, Map

, and proto.Struct are the three types of Protobuf message fields that I know of. Each approach has its own pros and cons, and if you happen to know a better one, feel free to discuss it in the comments.
,>

Let’s take a look at dynamic parameter passing using these three message field types.

Pass JSON object parameters using bytes

The bytes fields in a Protobuf encoded as Go code correspond to the Go slice []byte. Therefore, we can define the field types of dynamic parameters as bytes, so that after the client passes the JSON object to the server, the server can directly decode the JSON object contained in the dynamic parameters, saving a conversion from String to []byte.

For example, the info field in the Protobuf message definition below is of type Bytes

rpc UpdateRecord (UpdateRecordRequest) returns (UpdateRecordReply) {
}
    
message UpdateRecordRequest {
    int64 id = 1;
    bytes info = 2;
}
Copy the code

When using the gRPC method, the client can directly pass the info data to the server through json.Marshal encoding.

info := struct {
  name string
  age  int
} {
   name: "James",
   age: 20,
}
jsonInfo, _ := json.Marshal(info)
_ := AppService.UpdateRecord(&AppService.UpdateRecordRequest{id: 2, info: jsonInfo})
Copy the code

On the server side, you can add a parameter to verify that the correct JSON object is being passed.

func IsJSON(in []byte) bool {
 var js map[string]interface{}
 return json.Unmarshal(in, &js) == nil

}
Copy the code

After verification, you can decode the JSON object in the dynamic parameter to parse to the specific structure variable according to the actual use requirements.

type Info struct {
 name string `json:"name"`
 age int `json:"id"`
}

func (s server) UpdateRecord(ctx context.Context, reqeust *AppService.UpdateRecordRequest) (reply *AppService.UpdateRecordReply, err error) {
  if! isJson(req.Info) {// Error handling. } v := Info{} json.Unmarshal(req.Info, $v) }Copy the code

The only trouble is that each place that uses dynamic parameters has to define its own structure type for parsing JSON objects.

Dynamic parameters are passed using the Map type

If you don’t want to pass parameters through JSON objects, another commonly thought option is to define the field type of the parameter as a dictionary, which can be set to a different key-value pair each time you call it. Protobuf also happens to have a Map type.

map<key_type, value_type> map_field = N;
Copy the code

However, when defining a Map type, the value type must be fixed. Value types like Map [string]interface{} are not supported. So this method is usually used when you can determine the value type of the dictionary parameter. Otherwise, if you define map

as an integer field, the client needs to convert the data from integer to string first.
,>

Use proto.Struct to pass dynamic parameters of the structure

Struct to pass dynamic type parameters. The advantage of using it is that it appears to be Protobuf’s native support for dynamically typed data. Conversion from JSON to proto.struct can be done using the Protobuf package jsonPB.

Using a proto.Struct type requires importing its type definition in the proto file, as shown below.

syntax = "proto3";
package messages;
import "google/protobuf/struct.proto";

service UserService {
    rpc SendJson (SendJsonRequest) returns (SendJsonResponse) {}
}

message SendJsonRequest {
    string UserID = 1;
    google.protobuf.Struct Details = 2;
}

message SendJsonResponse {
    string Response = 1;
}
Copy the code

Struct is a message named Struct, which only contains a Map type field named Fileds. The Oneof feature of Protobuf specifies the type range of the Map value to approximate the completion of dynamic type support.

message Struct { // Unordered map of dynamically typed values. map<string, Value> fields = 1; } message Value { // The kind of value. oneof kind { // Represents a null value. NullValue null_value = 1; // Represents a double value. double number_value = 2; // Represents a string value. string string_value = 3; // Represents a boolean value. bool bool_value = 4; // Represents a structured value. Struct struct_value = 5; // Represents a repeated `Value`. ListValue list_value = 6; }}Copy the code

So manipulating proto.Struct in use is a bit like manipulating a dictionary, and here’s an example of how to use it.

func sendJson(userClient pb.UserServiceClient, ctx context.Context) {
    var item = &structpb.Struct{
        Fields: map[string]*structpb.Value{
            "name": &structpb.Value{
                Kind: &structpb.Value_StringValue{
                    StringValue: "James",}},"age": &structpb.Value{
                Kind: &structpb.Value_NumberValue{
                    NumberValue: 20,
                },
            },
        },
    }

    userGetRequest := &pb.SendJsonRequest{
        UserID: "A123",
        Details: item,
    }

    res, err := userClient.SendJson(ctx, userGetRequest)
}
Copy the code

conclusion

After summing up the three methods, I still think the first one is more convenient to use. The second one can only limit the value type to one, otherwise it needs to do type conversion between the client and the server. The third one is also available on the Internet with less information about the use of proto. Protobuf also has an Any type, which lets us use it without defining a message, but it’s also a bit cumbersome to carry a URL that specifies the data. This piece if the reader friends relevant experience can be discussed together.

If you like my article, please give me a thumbs up. I will share what I have learned and seen and first-hand experience through technical articles every week. Thank you for your support. Wechat search concerns the public number “network management talking BI talking” every week to teach you an advanced knowledge, there are special for the development of engineers Kubernetes tutorial.