Home  >  Article  >  Backend Development  >  GO and GRPC: Creating protobuff classes "on the fly"

GO and GRPC: Creating protobuff classes "on the fly"

PHPz
PHPzforward
2024-02-06 11:06:031230browse

GO 和 GRPC:“在飞行中”创建 protobuff 类

Question content

I am new to GRPC and cannot solve a problem. Is it possible to create a protobuff file when the application is already running? For example, I receive a json from the user like this:

"protobuf_file": "protobuf.proto", // File will be also recieved from user
"service_name": "Unary",
"method_name": "GetServerResponse",
"request_data_serializer_name": "MessageRequest",
"body": "grpc_request_data.json", // File will be also recieved from user

Here I have a .proto file, service name, method and message, and another json with data to populate the message. Now I have to open the connection and call the required method with the provided data.

TY!

Notes .proto The file (from the Get Instructions Guide) will be:

syntax = "proto3";

package protobuf_all_modes;

service Unary {
 rpc GetServerResponse(MessageRequest) returns (MessageResponse) {}
}

message MessageRequest {
 string message = 1;
}

message MessageResponse {
 string message = 1;
 int32 random_int32 = 2;
}

The second json will be:

{
    "message": "hello World!"
}

I don't know where to look for a solution. Any suggestions would be greatly appreciated


Correct answer


If anyone has the same problem, there is a very good library - https:// pkg.go.dev/github.com/jhump/[email protected]/dynamic and a sub-package https://pkg.go.dev/github. com/jhump/[email protected]/dynamic/grpcdynamic The code snippet would be: parser

func NewGrpcObject(operation *BaseOperation) *GrpcObject {
    fns, err := protoparse.ResolveFilenames([]string{"./"}, operation.ProtoFile) // prase .proto file
    if err != nil {
        log.Error(err)
    }

    parser := protoparse.Parser{}
    fds, err := parser.ParseFiles(fns...)
    if err != nil {
        log.Error(err)
    }
    descriptor := fds[0] // In my case there will be only one .proto
    pkg := descriptor.GetPackage()
    serviceDescriptor := descriptor.FindService(pkg + "." + operation.ServiceName) // name of service descriptor will be with package name first
    methodDescriptor := serviceDescriptor.FindMethodByName(operation.MethodName)
    requestDescriptor := methodDescriptor.GetInputType() // You can get types for request and response
    responseDescriptor := methodDescriptor.GetOutputType()

    return &GrpcObject{
        RequestDesc: requestDescriptor,
        MethodDesc: methodDescriptor,
        ResponseDesc: responseDescriptor,
    }
}

caller

// connect 
        conn, _ := grpc.Dial(operation.Host, grpc.WithTransportCredentials(credentials.NewTLS(c.TLSClientConfig.NewTLSConfig())))
        stub = grpcdynamic.NewStub(conn)

// call
    message := dynamic.NewMessage(operation.GrpcObject.RequestDesc) // from parser
    message.UnmarshalJSON(operation.Body) // here a JSON to fill message

    resp, err := stub.InvokeRpc(ctx, operation.GrpcObject.MethodDesc, message)
    if err != nil {
        // handle err
    }
    respMessage := dynamic.NewMessage(operation.GrpcObject.ResponseDesc) // descriptor from parser
    respMessage.ConvertFrom(resp) // convert message from raw response

The above is the detailed content of GO and GRPC: Creating protobuff classes "on the fly". For more information, please follow other related articles on the PHP Chinese website!

Statement:
This article is reproduced at:stackoverflow.com. If there is any infringement, please contact admin@php.cn delete