今天来学一下绑定参数和验证参数的示例,这个的 readme 写的比较直白,我这样的小白也能看明白。
绑定参数和验证参数的示例
首先看主函数:
/*
* Copyright 2022 CloudWeGo Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package main
import (
"context"
"fmt"
"github.com/cloudwego/hertz/pkg/app"
"github.com/cloudwego/hertz/pkg/app/server"
)
type Args struct {
Query string `query:"query"`
QuerySlice []string `query:"q"`
Path string `path:"path"`
Header string `header:"header"`
Form string `form:"form"`
Json string `json:"json"`
Vd int `query:"vd" vd:"$==0||$==1"`
}
func main() {
h := server.Default(server.WithHostPorts("127.0.0.1:8080"))
h.POST("v:path/bind", func(ctx context.Context, c *app.RequestContext) {
var arg Args
err := c.BindAndValidate(&arg)
if err != nil {
panic(err)
}
fmt.Println(arg)
})
h.Spin()
}
这里没什么好讲的,主要是定义了一个 Args 结构体,然后在 POST 请求里创建了一个对应的变量,传入绑定和校验的函数里,无误后把这个变量打印出来。
接下来有四个文件夹,分别为自定义错误,自定义类型解析,自定义校验函数,以及近似零开销。先看第一个,自定义错误:
/*
* Copyright 2022 CloudWeGo Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package main
import (
"context"
"fmt"
"time"
"github.com/cloudwego/hertz/pkg/app"
"github.com/cloudwego/hertz/pkg/app/client"
"github.com/cloudwego/hertz/pkg/app/server"
"github.com/cloudwego/hertz/pkg/app/server/binding"
"github.com/cloudwego/hertz/pkg/protocol"
"github.com/cloudwego/hertz/pkg/protocol/consts"
)
type BindError struct {
ErrType, FailField, Msg string
}
// Error implements error interface.
func (e *BindError) Error() string {
if e.Msg != "" {
return e.ErrType + ": expr_path=" + e.FailField + ", cause=" + e.Msg
}
return e.ErrType + ": expr_path=" + e.FailField + ", cause=invalid"
}
type ValidateError struct {
ErrType, FailField, Msg string
}
// Error implements error interface.
func (e *ValidateError) Error() string {
if e.Msg != "" {
return e.ErrType + ": expr_path=" + e.FailField + ", cause=" + e.Msg
}
return e.ErrType + ": expr_path=" + e.FailField + ", cause=invalid"
}
func init() {
CustomBindErrFunc := func(failField, msg string) error {
err := BindError{
ErrType: "bindErr",
FailField: "[bindFailField]: " + failField,
Msg: "[bindErrMsg]: " + msg,
}
return &err
}
CustomValidateErrFunc := func(failField, msg string) error {
err := ValidateError{
ErrType: "validateErr",
FailField: "[validateFailField]: " + failField,
Msg: "[validateErrMsg]: " + msg,
}
return &err
}
binding.SetErrorFactory(CustomBindErrFunc, CustomValidateErrFunc)
}
func main() {
h := server.Default(server.WithHostPorts("127.0.0.1:8080"))
h.GET("bindErr", func(ctx context.Context, c *app.RequestContext) {
type TestBind struct {
A string `query:"a,required"`
}
var req TestBind
err := c.Bind(&req)
fmt.Printf("error: %v\n", err)
})
h.GET("validateErr", func(ctx context.Context, c *app.RequestContext) {
type TestValidate struct {
B int `query:"b" vd:"$>100; msg:'C must greater than 100'"`
}
var req TestValidate
err := c.Bind(&req)
if err != nil {
panic(err)
}
err = c.Validate(&req)
fmt.Printf("error: %v\n", err)
})
go h.Spin()
time.Sleep(1000 * time.Millisecond)
c, _ := client.NewClient()
req := protocol.Request{}
resp := protocol.Response{}
req.SetMethod(consts.MethodGet)
req.SetRequestURI("http://127.0.0.1:8080/bindErr")
c.Do(context.Background(), &req, &resp)
req.SetRequestURI("http://127.0.0.1:8080/validateErr?b=1")
c.Do(context.Background(), &req, &resp)
}
这段代码里自定义了两种 error,绑定时的 error 和校验时的 error,然后用 Error() 实现了这两个接口,根据读入的 error 类型,决定处理逻辑,走的是一个函数工厂模式。底下是一套简单的测试,验证这些 error 处理逻辑对不对。