”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > RPC 操作 EPU 使用 Protobuf 并创建自定义插件

RPC 操作 EPU 使用 Protobuf 并创建自定义插件

发布于2024-11-08
浏览:141

RPC Action EPUsing Protobuf and Creating a Custom Plugin

上一篇文章中,我使用net/rpc包实现了一个简单的RPC接口,并尝试了net/rpc自带的Gob编码和JSON编码,学习了Golang的一些基础知识远程过程调用。在这篇文章中,我将结合 net/rpc 和 protobuf 并创建我的 protobuf 插件来帮助我们生成代码,所以让我们开始吧。

本文首发于Medium MPP计划。如果您是 Medium 用户,请在 Medium 上关注我。非常感谢。

我们在工作中肯定使用过gRPC和protobuf,但是它们并没有绑定。 gRPC可以使用JSON编码,protobuf可以使用其他语言实现。

Protocol BuffersProtobuf)是一种免费开源跨平台数据格式,用于序列化结构化数据。它对于开发通过网络相互通信或存储数据的程序很有用。该方法涉及一种描述某些数据结构的接口描述语言和一个根据该描述生成源代码的程序,用于生成或解析表示结构化数据的字节流。

使用 protobuf 的示例

首先我们编写一个proto文件hello-service.proto,它定义了一条消息“String”

syntax = "proto3";
package api;
option  go_package="api";

message String {
  string value = 1;
}

然后使用protoc实用程序生成消息String的Go代码

protoc --go_out=. hello-service.proto

然后我们修改Hello函数的参数以使用protobuf文件生成的字符串。

type HelloServiceInterface = interface {  
    Hello(request api.String, reply *api.String) error  
}  

使用起来和以前没有什么不同,甚至不如直接使用string方便。那么我们为什么要使用protobuf呢?正如我前面所说,使用Protobuf定义与语言无关的RPC服务接口和消息,然后使用protoc工具生成不同语言的代码,才是它真正的价值所在。例如使用官方插件protoc-gen-go生成gRPC代码。

protoc --go_out=plugins=grpc. hello-service.proto

protoc 的插件系统

要从 protobuf 文件生成代码,我们必须安装 protoc ,但是 protoc 不知道我们的目标语言是什么,所以我们需要插件来帮助我们生成代码。 protoc的插件系统如何工作?以上面的grpc为例。

这里有一个--go_out参数。由于我们调用的插件是protoc-gen-go,因此参数称为go_out;如果名称为 XXX,则该参数将被称为 XXX_out.

protoc运行时,首先会解析protobuf文件,生成一组Protocol Buffers编码的描述性数据。它首先会判断protoc中是否包含go插件,然后会尝试在$PATH中寻找protoc-gen-go,如果找不到就会报错,然后将运行 protoc-gen-go。 protoc-gen-go 命令并通过 stdin 将描述数据发送到插件命令。插件生成文件内容后,会将Protocol Buffers编码的数据输入到stdout,告诉protoc生成特定的文件。

plugins=grpc 是 protoc-gen-go 附带的一个插件,以便调用它。如果你不使用它,它只会在Go中生成一条消息,但是你可以使用这个插件来生成grpc相关的代码。

自定义协议插件

如果我们在protobuf中添加Hello接口时序,是否可以自定义一个protoc插件直接生成代码?

syntax = "proto3";  
package api;  
option  go_package="./api";  
service HelloService {  
  rpc Hello (String) returns (String) {}  
}  
message String {  
  string value = 1;
}

客观的

对于本文,我的目标是创建一个插件,然后用于生成 RPC 服务器端和客户端代码,如下所示。

// HelloService_rpc.pb.go
type HelloServiceInterface interface {  
    Hello(String, *String) error  
}  

func RegisterHelloService(  
    srv *rpc.Server, x HelloServiceInterface,  
) error {  
    if err := srv.RegisterName("HelloService", x); err != nil {  
       return err  
    }  
    return nil  
}  

type HelloServiceClient struct {  
    *rpc.Client  
}  

var _ HelloServiceInterface = (*HelloServiceClient)(nil)  

func DialHelloService(network, address string) (  
    *HelloServiceClient, error,  
) {  
    c, err := rpc.Dial(network, address)  
    if err != nil {  
       return nil, err  
    }  
    return &HelloServiceClient{Client: c}, nil  
}  

func (p *HelloServiceClient) Hello(  
    in String, out *String,  
) error {  
    return p.Client.Call("HelloService.Hello", in, out)  
}

这会将我们的业务代码更改为如下所示

// service
func main() {  
    listener, err := net.Listen("tcp", ":1234")  
    if err != nil {  
       log.Fatal("ListenTCP error:", err)  
    }  
    _ = api.RegisterHelloService(rpc.DefaultServer, new(HelloService))  
    for {  
       conn, err := listener.Accept()  
       if err != nil {  
          log.Fatal("Accept error:", err)  
       }  
       go rpc.ServeConn(conn)  
    }  
}  

type HelloService struct{}  

func (p *HelloService) Hello(request api.String, reply *api.String) error {  
    log.Println("HelloService.proto Hello")  
    *reply = api.String{Value: "Hello:"   request.Value}  
    return nil  
}
// client.go
func main() {  
    client, err := api.DialHelloService("tcp", "localhost:1234")  
    if err != nil {  
       log.Fatal("net.Dial:", err)  
    }  
    reply := &api.String{}  
    err = client.Hello(api.String{Value: "Hello"}, reply)  
    if err != nil {  
       log.Fatal(err)  
    }  
    log.Println(reply)  
}

根据生成的代码,我们的工作量已经小很多了,出错的机会也已经很小了。一个好的开始。

根据上面的api代码,我们可以拉出一个模板文件:

const tmplService = `  
import (  
    "net/rpc")  
type {{.ServiceName}}Interface interface {  
func Register{{.ServiceName}}(  
    if err := srv.RegisterName("{{.ServiceName}}", x); err != nil {        return err    }    return nil}  
    *rpc.Client}  
func Dial{{.ServiceName}}(network, address string) (  
{{range $_, $m := .MethodList}}  
    return p.Client.Call("{{$root.ServiceName}}.{{$m.MethodName}}", in, out)}  
`

整个模板很清晰,里面有一些占位符,比如MethodName、ServiceName等,我们稍后会介绍。

如何开发插件?

Google发布了Go语言API 1,引入了新的包google.golang.org/protobuf/compile R/protogen,大大降低了插件开发的难度:

  1. 首先我们创建一个go语言项目,如protoc-gen-go-spprpc
  2. 然后我们需要定义一个protogen.Options,然后调用它的Run方法,并传入一个func(*protogen.Plugin)错误回调。主流程代码到此结束。
  3. 我们还可以设置protogen.Options的ParamFunc参数,这样protogen会自动帮我们解析命令行传递的参数。从标准输入读取和解码 protobuf 信息、将输入信息编码到 protobuf 以及写入 stdout 等操作均由 protogen 处理。我们需要做的是与protogen.Plugin交互,实现代码生成逻辑。

每个服务最重要的是服务的名称,然后每个服务都有一套方法。对于服务定义的方法来说,最重要的是方法的名称,以及输入参数的名称和输出参数类型。我们先定义一个ServiceData来描述服务的元信息:

// ServiceData 
type ServiceData struct {  
    PackageName string  
    ServiceName string  
    MethodList  []Method  
}
// Method 
type Method struct {  
    MethodName     string  
    InputTypeName  string  
    OutputTypeName string  
}

然后是主逻辑,以及代码生成逻辑,最后是调用tmpl生成代码。

func main() {  
    protogen.Options{}.Run(func(gen *protogen.Plugin) error {  
       for _, file := range gen.Files {  
          if !file.Generate {  
             continue  
          }  
          generateFile(gen, file)  
       }  
       return nil  
    })  
}  

// generateFile function definition
func generateFile(gen *protogen.Plugin, file *protogen.File) {  
    filename := file.GeneratedFilenamePrefix   "_rpc.pb.go"  
    g := gen.NewGeneratedFile(filename, file.GoImportPath)  
    tmpl, err := template.New("service").Parse(tmplService)  
    if err != nil {  
       log.Fatalf("Error parsing template: %v", err)  
    }  
    packageName := string(file.GoPackageName)  
// Iterate over each service to generate code
    for _, service := range file.Services {  
       serviceData := ServiceData{  
          ServiceName: service.GoName,  
          PackageName: packageName,  
       }  
       for _, method := range service.Methods {  
          inputType := method.Input.GoIdent.GoName  
          outputType := method.Output.GoIdent.GoName  

          serviceData.MethodList = append(serviceData.MethodList, Method{  
             MethodName:     method.GoName,  
             InputTypeName:  inputType,  
             OutputTypeName: outputType,  
          })  
       }  
// Perform template rendering
       err = tmpl.Execute(g, serviceData)  
       if err != nil {  
          log.Fatalf("Error executing template: %v", err)  
       }  
    }  
}

调试插件

最后,我们将编译好的二进制执行文件 protoc-gen-go-spprpc 放入 $PATH 中,然后运行 ​​protoc 生成我们想要的代码。

protoc --go_out=.. --go-spprpc_out=.. HelloService.proto

因为protoc-gen-go-spprpc必须依赖protoc才能运行,所以调试起来有点棘手。我们可以使用

fmt.Fprintf(os.Stderr, "Fprintln: %v\n", err)

打印错误日志进行调试。

概括

这就是本文的全部内容。我们首先使用protobuf实现了一个RPC调用,然后创建了一个protobuf插件来帮助我们生成代码。这为我们学习protobuf RPC打开了大门,也是我们深入了解gRPC的途径。希望大家都能掌握这项技术。

参考

  1. https://taoshu.in/go/create-protoc-plugin.html
  2. https://chai2010.cn/advanced-go-programming-book/ch4-rpc/ch4-02-pb-intro.html
版本声明 本文转载于:https://dev.to/huizhou92/rpc-action-ep2-using-protobuf-and-creating-a-custom-plugin-2j9j?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 除了“if”语句之外:还有什么地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    除了“if”语句之外:还有什么地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    无需强制转换即可上下文转换为 bool您的类定义了对 bool 的显式转换,使您能够在条件语句中直接使用其实例“t”。然而,这种显式转换提出了一个问题:“t”在哪里可以在不进行强制转换的情况下用作 bool?上下文转换场景C 标准指定了四种值可以根据上下文转换为的主要场景bool:语句:if、whi...
    编程 发布于2024-12-26
  • 在 Go 中使用 WebSocket 进行实时通信
    在 Go 中使用 WebSocket 进行实时通信
    构建需要实时更新的应用程序(例如聊天应用程序、实时通知或协作工具)需要比传统 HTTP 更快、更具交互性的通信方法。这就是 WebSockets 发挥作用的地方!今天,我们将探讨如何在 Go 中使用 WebSocket,以便您可以向应用程序添加实时功能。 在这篇文章中,我们将介绍: WebSocke...
    编程 发布于2024-12-26
  • 如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    MySQL配置不正确:相对路径的问题在Django中运行python manage.py runserver时,可能会遇到以下错误:ImproperlyConfigured: Error loading MySQLdb module: dlopen(/Library/Python/2.7/site-...
    编程 发布于2024-12-26
  • 如何在 HTML 表格中有效地使用 Calc() 和基于百分比的列?
    如何在 HTML 表格中有效地使用 Calc() 和基于百分比的列?
    在表格中使用 Calc():克服百分比困境创建具有固定宽度列和可变宽度列的表格可能具有挑战性,尤其是在尝试在其中使用 calc() 函数。在 HTML 中,使用 px 或 em 设置固定列宽非常简单。但是,对于可变宽度列,通常使用百分比 (%) 单位。然而,当在表中使用 calc() 时,百分比似乎...
    编程 发布于2024-12-26
  • Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta:列偏移的删除和恢复Bootstrap 4 在其 Beta 1 版本中引入了重大更改柱子偏移了。然而,随着 Beta 2 的后续发布,这些变化已经逆转。从 offset-md-* 到 ml-auto在 Bootstrap 4 Beta 1 中, offset-md-*...
    编程 发布于2024-12-26
  • 如何在PHP中通过POST提交和处理多维数组?
    如何在PHP中通过POST提交和处理多维数组?
    在 PHP 中通过 POST 提交多维数组当使用具有可变长度的多列和行的 PHP 表单时,有必要进行转换输入到多维数组中。这是解决这一挑战的方法。首先,为每列分配唯一的名称,例如:<input name="topdiameter[' current ']" type=&qu...
    编程 发布于2024-12-26
  • for(;;) 循环到底是什么以及它是如何工作的?
    for(;;) 循环到底是什么以及它是如何工作的?
    揭秘神秘的 for(;;) 循环在古老的代码库深处,你偶然发现了一个令人困惑的奇特 for 循环你的理解。其显示如下:for (;;) { //Some stuff }您深入研究在线资源,但发现自己陷入沉默。让我们剖析这个神秘的构造。for 循环的结构Java 中的 for 循环遵循特定的语...
    编程 发布于2024-12-25
  • Java 的 Scanner.useDelimiter() 如何使用正则表达式?
    Java 的 Scanner.useDelimiter() 如何使用正则表达式?
    Java 中使用 Scanner.useDelimiter 了解分隔符Java 中的 Scanner 类提供了 useDelimiter 方法,允许您指定分隔符(字符或模式)来分隔代币。然而,使用分隔符可能会让初学者感到困惑。让我们用更简单的术语来分解它。考虑片段:sc = new Scanner(...
    编程 发布于2024-12-25
  • 如何在 Android 中显示动画 GIF?
    如何在 Android 中显示动画 GIF?
    在 Android 中显示动画 GIF尽管最初误解 Android 不支持动画 GIF,但实际上它具有解码和显示动画的能力显示它们。这是通过利用 android.graphics.Movie 类来实现的,尽管这方面没有广泛记录。要分解动画 GIF 并将每个帧作为可绘制对象合并到 AnimationD...
    编程 发布于2024-12-25
  • 为什么我在运行 phpize 时出现“找不到 config.m4”错误?
    为什么我在运行 phpize 时出现“找不到 config.m4”错误?
    解决 phpize 中的“找不到 config.m4”错误运行 phpize 时遇到“找不到 config.m4”错误是可能阻碍 ffmpeg 等扩展安装的常见问题。以下是解决此错误并让 phpize 启动并运行的方法。先决条件:您已经安装了适合您的 PHP 版本的必要开发包,例如 php- Deb...
    编程 发布于2024-12-25
  • 打印时如何在每页上重复表头?
    打印时如何在每页上重复表头?
    在打印模式下重复表格标题当表格在打印过程中跨越多个页面时,通常需要有标题行(TH 元素)在每页上重复,以便于参考。 CSS 提供了一种机制来实现此目的。解决方案:使用 THEAD 元素CSS 中的 THEAD 元素是专门为此目的而设计的。它允许您定义一组应在每个打印页面上重复的标题行。使用方法如下:...
    编程 发布于2024-12-25
  • 尽管代码有效,为什么 POST 请求无法捕获 PHP 中的输入?
    尽管代码有效,为什么 POST 请求无法捕获 PHP 中的输入?
    解决 PHP 中的 POST 请求故障在提供的代码片段中:action=''而不是:action="<?php echo $_SERVER['PHP_SELF'];?>";?>"检查 $_POST数组:表单提交后使用 var_dump 检查 $_POST 数...
    编程 发布于2024-12-25
  • 如何在 PHP 中组合两个关联数组,同时保留唯一 ID 并处理重复名称?
    如何在 PHP 中组合两个关联数组,同时保留唯一 ID 并处理重复名称?
    在 PHP 中组合关联数组在 PHP 中,将两个关联数组组合成一个数组是一项常见任务。考虑以下请求:问题描述:提供的代码定义了两个关联数组,$array1 和 $array2。目标是创建一个新数组 $array3,它合并两个数组中的所有键值对。 此外,提供的数组具有唯一的 ID,而名称可能重合。要求...
    编程 发布于2024-12-25
  • 为什么 `cout` 会误解 `uint8_t` 以及如何修复它?
    为什么 `cout` 会误解 `uint8_t` 以及如何修复它?
    深入分析:为什么 uint8_t 无法正确打印您遇到了 uint8_t 变量的值无法正确打印的问题库特。经过调查,您发现将数据类型更改为 uint16_t 可以解决该问题。此行为源于 uint8_t 的基本性质以及 cout 处理字符数据的方式。uint8_t 在内部存储一个无符号 8 位整数。当您...
    编程 发布于2024-12-25
  • MySQLnd 是否已启用?
    MySQLnd 是否已启用?
    确定 MySQLnd 驱动程序状态在 PHP 中使用 MySQL 时,验证 MySQLnd 是否是活动数据库驱动程序至关重要。与旧版驱动程序相比,此高级驱动程序提供了增强的性能和功能。使用 MySQLi 检查 MySQLnd对于 MySQLi 连接,您可以使用以下代码来确定 MySQLnd状态:$m...
    编程 发布于2024-12-25

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3