golang proto api 校验国际化 protovalidate

golang proto api 校验国际化 protovalidate
Photo by Scott Webb / Unsplash

众所周知,protobuf 原型文件扩展很多功能,比如生成 http 接口层代码,顺势就有了生成接口参数校验代码的需求。

早期可以使用https://github.com/bufbuild/protoc-gen-validate 来实现,通过生成特定的 go 代码的方式来实现校验。

github 中也提到目前趋于稳定,不会有更多新特性的支持,推荐大家使用新的版本 protovalidate,https://github.com/bufbuild/protovalidate 。该版本是protoc-gen-validate 的“精神继承者”。它不需要任何代码生成并支持自定义约束。

现在我们尝试新版本,并且增加国际化支持。

go get github.com/bufbuild/protovalidate-go

import "github.com/bufbuild/protovalidate-go"

syntax = "proto3";
package demo;

import "google/api/annotations.proto";
import "validate/validate.proto";

option go_package = "demo/app";

service App {
  rpc AppDetail(AppRequest) returns (AppItem) {
    option (google.api.http) = {
      get: "/app/detail",
    };
  }
}

message AppRequest {
  // 应用id
  int64 app_id = 1 [(buf.validate.field).int64 = {gt:  999}];
}

上图中的 validate/validate.proto 可以从仓库中复制下来https://github.com/bufbuild/protovalidate/tree/main/proto/protovalidate/buf/validate。不过官方可能推荐你使用他家 buf managed mode 来生成,这里就不展开了。

package app

import (
	"github.com/bufbuild/protovalidate-go"
)

func (a AppHTTPServerController) AppDetail(context *gin.Context, request *app.AppRequest) (*app.AppItem, error) {
    validator, err := protovalidate.New()
    if err != nil {
        log.Fatalf("failed to create validator: %v", err)
    }
    // 为每个约束设置中文,这里是一个 demo,需要进一步完成。
    var ruleMessages = map[string]string{
        "int64.gt": "{{.FieldName}}: 值必须要大于 {{.FieldValue}},当前值 {{.RuleValue}}",
    }

    type ErrorInfo struct {
        FieldName  string
        RuleValue  any
        FieldValue any
    }

    if err := validator.Validate(request); err != nil {
        fmt.Printf("Validation failed: %v\n", err)
        var valErr *protovalidate.ValidationError
        if ok := errors.As(err, &valErr); ok {
            for _, violation := range valErr.Violations {
                errmsg := template.Must(template.New("").Parse(ruleMessages[violation.Proto.GetConstraintId()])).
                    Execute(os.Stdout, ErrorInfo{
                        FieldName:  protovalidate.FieldPathString(violation.Proto.GetField()),
                        RuleValue:  violation.RuleValue.Interface(),
                        FieldValue: violation.FieldValue.Interface(),
                    })
                // 这里输出每个约束的信息
                fmt.Println(errmsg)
            }
        }
        return nil, err
    } else {
        fmt.Println("Validation passed!")
    }
}

输出:

🔹
app_id: 值必须要大于 100,当前值 999

网上鲜有 demo,这里仅作示例,未对任何框架做适配。

Read more

CLI 工具多版本管理器 - asdf

CLI 工具多版本管理器 - asdf

asdf 是一个 CLI 工具,可以根据每个项目管理多个工具或语言运行时版本。它就像 gvm、nvm、rbenv 和 pyenv(以及更多)的合一工具!只需安装您语言的插件即可! 这个随手起的名字,可能是目前地表最强版本管理器。 https://github.com/asdf-vm/asdf 一、安装 asdf 以 macbook 为示例 brew install asdf asdf -v asdf version 0.16.2 0.16.0 以上命令有较大变更,详见:https://asdf-vm.com/guide/upgrading-to-v0-16.html#breaking-changes 本文使用

By brian
git clone 复制一个整个仓库并推送到新地址

git clone 复制一个整个仓库并推送到新地址

要使用 git clone --bare 复制一个新的仓库并推送到远程仓库,可以按照以下步骤操作: 1. 克隆一个裸仓库 首先,使用 git clone --bare 命令克隆源仓库。假设源仓库的 URL 是 https://github.com/user/source-repo.git,你可以执行以下命令: bash复制 git clone --bare https://github.com/user/source-repo.git 这将创建一个新的裸仓库(没有工作区),通常会创建一个名为 source-repo.git 的目录。 2. 进入裸仓库目录 进入刚刚克隆的裸仓库目录: bash复制 cd source-repo.git 3. 添加新的远程仓库 接下来,

By brian
沪ICP备2022013452号-1