当前位置: 首页 > 面试题库 >

如何将Go中间件模式与错误返回请求处理程序结合在一起?

洪子晋
2023-03-14
问题内容

我熟悉这样的Go中间件模式:

// Pattern for writing HTTP middleware.
func middlewareHandler(next http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        // Our middleware logic goes here before executing application handler.
        next.ServeHTTP(w, r)
        // Our middleware logic goes here after executing application handler.
    })
}

例如,如果我有一个loggingHandler:

func loggingHandler(next http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        // Before executing the handler.
        start := time.Now()
        log.Printf("Strated %s %s", r.Method, r.URL.Path)
        next.ServeHTTP(w, r)
        // After executing the handler.
        log.Printf("Completed %s in %v", r.URL.Path, time.Since(start))
    })
}

和一个简单的handleFunc:

func handleFunc(w http.ResponseWriter, r *http.Request) {
    w.Write([]byte(`Hello World!`))
}

我可以这样合并它们:

http.Handle("/", loggingHandler(http.HandlerFunc(handleFunc)))
log.Fatal(http.ListenAndServe(":8080", nil))

很好

但是我喜欢处理程序能够像正常函数一样返回错误的想法。这使错误处理变得容易得多,因为如果有错误,我可以只返回错误,或者在函数末尾仅返回nil。

我这样做是这样的:

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

func (f errorHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
    err := f(w, r)
    if err != nil {
        // log.Println(err)
        fmt.Println(err)
        os.Exit(1)
    }
}

func errorHandle(w http.ResponseWriter, r *http.Request) error {
    w.Write([]byte(`Hello World from errorHandle!`))
    return nil
}

然后像这样包装它来使用它:

http.Handle("/", errorHandler(errorHandle))

我可以使这两种模式分别工作,但是我不知道如何将它们组合在一起。我喜欢我能够将中间件与Alice之类的库链接在一起。但是如果他们也可以返回错误,那就太好了。我有办法实现这一目标吗?


问题答案:

我也喜欢这种HandlerFuncs返回错误的模式,它更加整洁,您只需编写一次错误处理程序即可。只需将中间件与包含的处理程序分开考虑,就不需要中间件传递错误。中间件就像一条链,依次执行每个中间件,然后最后一个中间件是知道您的处理程序签名并适当处理错误的中间件。

因此,以最简单的形式,使中间件保持完全相同,但最后插入一个具有这种形式的中间件(并且不执行另一种中间件,而是执行特殊的HandlerFunc):

// Use this special type for your handler funcs
type MyHandlerFunc func(w http.ResponseWriter, r *http.Request) error


// Pattern for endpoint on middleware chain, not takes a diff signature.
func errorHandler(h MyHandlerFunc) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
       // Execute the final handler, and deal with errors
        err := h(w, r)
        if err != nil {
            // Deal with error here, show user error template, log etc
        }
    })
}

然后像这样包装您的函数:

moreMiddleware(myMiddleWare(errorHandler(myhandleFuncReturningError)))

这意味着这个特殊的错误中间件只能包装您的特殊功能签名,并且位于链的末尾,但这很好。另外,我会考虑将这种行为包装在您自己的多路复用器中,以使其更简单并避免传递错误处理程序,并让您更轻松地构建中间件链,而不必在路由设置中进行难看的包装。

我认为,如果您使用的是路由器库,则可能需要对此模式的显式支持。您可以在此路由器中以修改后的形式查看此示例的运行方式,该示例完全使用您要使用的签名,但是可以处理中间件链并无需手动包装即可执行:

https://github.com/fragmenta/mux/blob/master/mux.go



 类似资料:
  • 我有一个web服务,它从另一个系统接收xml事件,使用特定的工作流处理它们,并将潜在错误列表作为HTTP响应发送回。 事件处理工作流由几个使用Guava的EventBus实现的处理程序(例如:预处理器、持久器和验证器)组成。处理程序相互发送事件。类似这样: 问题是:如何将处理结果从Validator处理程序深度返回到起点(Request estHandler),以便用户接收HTTP响应? 我考虑两

  • 我在Tomcat中部署了一个Java的Web应用程序。 如果我的URL是这样的,我会收到来自Tomcat 8的400个错误请求 编码为 但是如果我从 URL 中删除 a:b:ab,a:b:abc 和 renditionFilter=cmis:thumbnail,application/pdf,image/bmp,image/gif,image/jpeg,image/png,那么它就可以工作了,这个

  • 因此,我有以下内容,这似乎令人难以置信的骇客,我一直在想,Go有比这更好的设计库,但我找不到Go处理JSON数据POST请求的例子。它们都是表格帖子。 下面是一个示例请求: 下面是嵌入日志的代码: 肯定有更好的办法,对吧?我只是很难找到最好的做法。 对搜索引擎来说,Go也被称为Golang,这里提到了它,这样其他人就可以找到它。)

  • 我真的不知道我的java代码中的错误在哪里。我必须使用REST API登录Kofax Total Agility。为此,我尝试使用postman测试我的json是否正确构建。以下是我的登录JSON: 我得到了肯定的回答: 到目前为止,一切顺利。为此,我创建了模型: 对于响应: 这些类应该允许我构建 json。现在,我创建了一个方法,用于生成请求对象并期望响应对象。 当我调用这部分代码时,我注意到我

  • 我正在尝试一个从Android到Spring MVC web服务的简单帖子,但我收到了对URL的帖子请求,结果是400(错误请求):调用错误处理程序org。springframework。网状物客户HttpClientErrorException:400错误请求 这是我的Android代码: 这是我的控制器: 可能是什么问题?