Home > Backend Development > Golang > GO and GRPC: Creating protobuff classes 'on the fly'

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

PHPz
Release: 2024-02-06 11:06:03
forward
1323 people have browsed it

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
Copy after login

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;
}
Copy after login

The second json will be:

{
    "message": "hello World!"
}
Copy after login

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,
    }
}
Copy after login

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
Copy after login

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!

source:stackoverflow.com
Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
Popular Tutorials
More>
Latest Downloads
More>
Web Effects
Website Source Code
Website Materials
Front End Template