json - 被 gin c.BindJSON 捕获时如何断言错误类型 json.UnmarshalTypeError
问题描述
我正在尝试使用 gin gonic 捕获绑定错误,并且对于来自 go-playground/validator/v10 的所有验证错误都可以正常工作,但是在解组为正确的数据类型时我遇到了捕获错误的问题。
gin.ErrorTypeBind
使用验证器标签(必需,...)时,结构字段验证不成功将返回错误类型
但如果我有一个结构
type Foo struct {
ID int `json:"id"`
Bar string `json:"bar"`
}
而且我试图传递的 json 格式错误(传递字符串而不是 id 的数字)
{
"id":"string",
"bar":"foofofofo"
}
它将失败并出现错误json: cannot unmarshal string into Go struct field Foo.id of type int
它仍然gin.ErrorTypeBind
在我的处理程序中作为绑定错误被捕获,但由于我需要区分验证错误和解组错误,我遇到了问题。
我已经尝试过验证错误的类型转换不适用于解组:
e.Err.(validator.ValidationErrors)
会恐慌
或者只是errors.Is,但这根本不会捕获错误
if errors.Is(e.Err, &json.UnmarshalTypeError{}) {
log.Println("Json binding error")
}
我这样做的目标是向用户返回格式正确的错误消息。它目前适用于所有验证逻辑,但我似乎无法使其适用于将不正确数据发送给我的 json 数据。
有任何想法吗?
编辑 :
添加示例以重现:
package main
import (
"encoding/json"
"errors"
"log"
"net/http"
"github.com/gin-gonic/gin"
"github.com/go-playground/validator/v10"
)
type Foo struct {
ID int `json:"id" binding:"required"`
Bar string `json:"bar"`
}
func FooEndpoint(c *gin.Context) {
var fooJSON Foo
err := c.BindJSON(&fooJSON)
if err != nil {
// caught and answer in the error MW
return
}
c.JSON(200, "test")
}
func main() {
api := gin.Default()
api.Use(ErrorMW())
api.POST("/foo", FooEndpoint)
api.Run(":5000")
}
func ErrorMW() gin.HandlerFunc {
return func(c *gin.Context) {
c.Next()
if len(c.Errors) > 0 {
for _, e := range c.Errors {
switch e.Type {
case gin.ErrorTypeBind:
log.Println(e.Err)
var jsonErr json.UnmarshalTypeError
if errors.Is(e.Err, &jsonErr) {
log.Println("Json binding error")
}
// if errors.As(e.Err, &jsonErr) {
// log.Println("Json binding error")
// }
// in reality i'm making it panic.
// errs := e.Err.(validator.ValidationErrors)
errs, ok := e.Err.(validator.ValidationErrors)
if ok {
log.Println("error trying to cast validation type")
}
log.Println(errs)
status := http.StatusBadRequest
if c.Writer.Status() != http.StatusOK {
status = c.Writer.Status()
}
c.JSON(status, gin.H{"error": "error"})
default:
log.Println("other error")
}
}
if !c.Writer.Written() {
c.JSON(http.StatusInternalServerError, gin.H{"Error": "internal error"})
}
}
}
}
尝试发送带有正文的发布请求
{
"id":"rwerewr",
"bar":"string"
}
interface conversion: error is *json.UnmarshalTypeError, not validator.ValidationErrors
这将起作用:
{
"id":1,
"bar":"string"
}
这将(正确地)返回 Key: 'Foo.ID' Error:Field validation for 'ID' failed on the 'required' tag
{“酒吧”:“字符串”}
解决方案
errors.Is
查找完全匹配(在您的情况下,是 的空实例json.UnmarshalTypeError
)。改用errors.As
:
var jsonErr *json.UnmarshalTypeError
if errors.As(e.Err, &jsonErr) {
log.Println("Json binding error")
}
但是,正如@LeGEC 所指出的那样,这假设 gin 的错误符合标准Unwrapper
接口,而他们不符合。
推荐阅读
- ios - 推送通知正在发送但未收到 iOS - 云功能
- python - 有没有办法阻止跟随模型中的自我跟随?
- css - CSS Grid 项目仅在 Chrome 中增长到最大高度
- latex - 引用在使用 natbib 包的乳胶中不起作用
- java - PITest 问题:属性“mainClass”是最终的,无法进一步更改
- javascript - 如何检查函数是否被 jest 调用
- javascript - ChartJS:y轴正负部分的两个标题
- python - 影响python日志记录器的导入路径
- android - ListView 项目上的 onClick 侦听器
- python - Python MySQL编程错误:1064(42000),同时使用Flask制作表格