深入了解 REST API 中的自定义错误处理

问题描述

我来自 Node.js 的世界,在那里我从使用 Express 构建 Web 应用程序中学到了很多东西,我认为有一个很好的方法来处理错误和万一出现意外错误,有一种很棒的方法可以捕获它。

所以,我在 Go 中寻找相同的东西。我不知道我是否已经找到它,但我发现的内容来自此链接 https://astaxie.gitbooks.io/build-web-application-with-golang/content/en/11.1.html 和我阅读的一些文章,似乎许多开发人员都在使用相同的方法

我担心的不是真正的问题,是我不理解下面代码的某些部分。

type appHandler func(http.ResponseWriter,*http.Request) error

func (fn appHandler) ServeHTTP(w http.ResponseWriter,r *http.Request) {
    if err := fn(w,r); err != nil {
        http.Error(w,err.Error(),500)
    }
}

我知道可以在 go 中创建自定义类型,但严重的是我不明白这是什么意思或如何在 http.Serve

中理解它
type appHandler func(http.ResponseWriter,*http.Request) error

我没有抓住的一件事是

func (fn AppHandler) ServeHTTP(w http.ResponseWriter,r *http.Request) {

在我读过的大部分代码中,一般来说,它是一个结构体或替换类型(我的意思是 type Account int),但在这里,它是一个函数。 我想了解它如何帮助处理错误

在上面 ServeHTTP 的实现中,我们有这行 err := fn(w,r)。 拜托,你能解释一下吗?

在同一篇文章中,我们有这个代码

func viewRecord(w http.ResponseWriter,r *http.Request) error {
    c := appengine.NewContext(r)
    key := datastore.NewKey(c,"Record",r.FormValue("id"),nil)
    record := new(Record)
    if err := datastore.Get(c,key,record); err != nil {
        return err
    }
    return viewTemplate.Execute(w,record)
}

还有这一行

func init() {
    http.Handle("/view",appHandler(viewRecord))
}

拜托,你能帮我理解这个appHandler(viewRecord)吗?究竟是什么?它是一个实例化,它是铸造吗? 它应该做什么?我的意思是,如何理解似乎很关键的那条线?

最后一个问题,请。是否有可能捕获在处理请求期间可能发生的任何地方的错误?在 Node.js 中,你可以做一些类似

const app = express()

const errorHandler = (
  err: Error,req: Request,res: Response,next: NextFunction
) => {
  if (err instanceof CustomError) {
    return res.status(err.statusCode).send({ errors: err.serializeErrors() });
  }
  
  res.status(500).send({ 
    errors: [{ message: err.message || 'Something went wrong' }]
  });
}

app.use(errorHandler())

类似的东西,在 Go 中可以实现吗?

谢谢。

解决方法

在 Go 中,我们可以从任何原始类型定义自定义类型, 例子:

type A int
type Person struct{}

这对于函数来说很常见。您可以像上面示例中的 appHandler 类型一样定义具有所需函数签名的 Go 类型。我在下面添加了简单的例子。

package main

import (
    "errors"
    "fmt"
)

type IntFun func(a,b int) error
func main(){
    var addFunc IntFun
    addFunc = func(a,b int) error {
        if a ==0 || b == 0 {
            return errors.New(`zero valued inputs`)
        }
        fmt.Println(`sum := `,a+b)
        return nil
    }

    addFunc.Add(5,3) //Output: sum :=  8
    addFunc.Add(0,0) //Output: zero valued inputs
}

func (fn IntFun) Add(a,b int)  {
    err := fn(a,b)
    if err != nil {
        fmt.Println(err)
        return
    }
}

并参考user-defined-function-type-go

就你而言,

func init() {
    http.Handle("/view",appHandler(viewRecord))
}

这是http端点处理部分。 "/view" 是路径模式,其他参数是请求的处理程序。该参数应在 net/http Go 包中实现 Handler 接口。所以这就是应用处理程序类型具有 ServeHttp 函数的原因。

viewRecord 函数也兼容 type appHandler 函数类型。它处理到达“/view”端点的请求。因此它被强制转换为 appHandler 类型并传递给 http.Handle("/view",appHandler(viewRecord)) 句柄函数。

我认为这将解释这种情况。如果有什么不清楚的,请评论。