编译protobuf文件生成go代码时添加自定义的field tag

场景

需要使用go-playground/validator对入参进行更细致的验证, (一来相比其他grpc validator plugin, 这个包功能完善一些, 二来项目中使用的gin框架也是依赖于才此包, 保持统一)

当然其他场景可能有一些其他需求, 需要对protobuf文件编译出来的结构体, 添加上一些自定义的tag

问题

如果使用的是官方的protoc-gen-go, 则截止目前(2021.1.10), 仍未支持这个功能, 相关讨论可见:

https://github.com/golang/protobuf/issues/52

解决办法

1、使用gogo/protobuf, 并且使用其自己的插件

https://github.com/gogo/protobuf

demo: https://github.com/gogo/protobuf/blob/master/test/tags/tags.proto

2、使用工具修改编译出来的文件(最终采用)

使用的工具是: https://github.com/favadi/protoc-go-inject-tag

最后只需要修改编译protobuf文件的的shell脚本, 新增类似以下逻辑:

(如果你每次编译protobuf文件都是手输命令, 建议写到shell脚本里)

for file in ${pb_go_files[@]} # pb_go_files即编译出来的*.pb.go文件名
do
    if ! command -v protoc-go-inject-tag > /dev/null 2>&1; then
        `cd ~ && go get github.com/favadi/protoc-go-inject-tag` # 因为项目有可能是gomodule模式, 所以进入home目录再下载
    fi  
    protoc-go-inject-tag -input=./pb/${file}
done

简单review了一下这个工具的实现, 使用的是go/parser包对代码进行语义分析后注入的tag, 并不是特别hack的方法, 所以还是挺靠得住的

最终采用该方法的原因主要是:

  • 稳妥起见不想替换官方的protoc-gen-go,
  • 相对而言对proto文件的修改是以注释的形式, 侵入性较低, 且语法也比较简洁, 团队应该更容易接受一些