1

我在 Go 中的 Web 应用程序(使用 Gorillamuxnegroni)有大约 20 个处理程序,根据应该应用的中间件函数分为三组。具体来说:

  • 第 1 组:静态请求(根本没有中间件)

    GET   /favicon.ico
    GET   /files
    GET   /files/index.html
    GET   /files/favicon.ico
    
  • 第 2 组:应该只有 CORS 中间件,没有身份验证的请求:

    GET   /
    GET   /login
    POST  /login
    GET   /auth-configuration
    GET   /service-status
    
  • 第 3 组:应同时应用 CORS 和身份验证中间件的请求:

    GET   /articles
    POST  /articles
    PUT   /articles/etc
    PATCH /articles/etc
    

这是我设置 HTTP 服务器的代码:

func run() {

    negroniStack := setUpNegroni()

    bindAddr := // ...

    http.ListenAndServe(bindAddr, negroniStack)
}

func setUpNegroni() negroni.Negroni {

    negroniStack := negroni.Negroni{}

    staticNegroni := setUpRoutesAndMiddlewareForStaticRequests()
    loginNegroni  := setUpRoutesAndMiddlewareForLogin()
    serviceNegroni = setUpRoutesAndMiddlewareForService()

    negroniStack.UseHandler(&staticNegroni)
    negroniStack.UseHandler(&loginNegroni)
    negroniStack.UseHandler(&serviceNegroni)

    return negroniStack
}

func setUpRoutesAndMiddlewareForStaticRequests() negroni.Negroni {

    staticNegroni := negroni.Negroni{}
    staticRouter := mux.NewRouter()

    staticRouter.PathPrefix("/files").HandlerFunc(staticHandler)
    staticRouter.Path("/favicon.ico").HandlerFunc(staticHandler)

    staticNegroni.UseHandler(staticRouter)
    return staticNegroni
}

func setUpRoutesAndMiddlewareForLogin() negroni.Negroni {

    authNegroni := negroni.Negroni{}

    corsMiddleware := cors.New(cors.Options{
        AllowedMethods:     []string{"GET", "HEAD", "POST", "PUT", "PATCH", "DELETE"},
        AllowCredentials:   true,
        OptionsPassthrough: false,
    })

    authNegroni.Use(corsMiddleware)

    authRouter := mux.NewRouter()

    authRouter.HandleFunc("/login", HandlePostAuth).Methods("POST")
    authRouter.HandleFunc("/login", HandleGetAuth) // GET

    authNegroni.UseHandler(authRouter)

    return authNegroni
}

func setUpRoutesAndMiddlewareForService() negroni.Negroni {

    serviceNegroni := negroni.Negroni{}

    corsMiddleware := cors.New(cors.Options{
        AllowedMethods:     []string{"GET", "HEAD", "POST", "PUT", "PATCH", "DELETE"},
        AllowCredentials:   true,
        OptionsPassthrough: false,
    })
    serviceNegroni.Use(corsMiddleware)

    serviceNegroni.UseFunc(jwtMiddleware)

    serviceRouter := mux.NewRouter()
    serviceRouter.HandleFunc("/articles", HandleGetArticles).Methods("GET")
    serviceRouter.HandleFunc("/articles", HandlePostArticles).Methods("POST")
    // etc

    serviceNegroni.UseHandler(serviceRouter)

    return serviceNegroni
}

根据Negroni 文档中的“路由特定中间件”部分,我认为这是正确的,其中说:

如果您有一组需要执行特定中间件的路由,您可以简单地创建一个新的 Negroni 实例并将其用作您的路由处理程序。

但是,当我发出请求并使用调试器时,我看到它(*Negroni).ServeHTTP被多次调用。例如,如果我请求,GET /favicon.ico则该staticHandler函数被正确调用并调用WriteHeader(200),但之后它会调用下一个调用,mux.Router该调用WriteHeader(404)会在终端中打印出警告,因为标头被写入了两次(http: multiple response.WriteHeader calls

如果它用于不存在的路由,NotFoundHandler则调用 Gorilla 默认值 3 次(每次调用一次mux.Router)。

请求完成后,如何让 Negroni 停止调用其他处理程序?

...如果我的 Negroni 实例配置错误,为什么它不在初始化期间执行检查以警告我配置无效?

我的理解是,negroni.UseandUseFunc用于设置中间件(每个请求都会调用它们),而UseHandler用于设置终端处理程序(每个请求只调用 1 个,或回退到 404)。如果我正确理解了这种情况,那么出于某种原因,它将我的终端处理程序视为中间件。

4

1 回答 1

2

UseHandler文档(https://godoc.org/github.com/urfave/negroni#Negroni.UseHandler

UseHandler 将 http.Handler 添加到中间件堆栈中。处理程序按照它们添加到 Negroni 的顺序被调用。

因此,您在这里看到的似乎是预期的行为。

您基本上是在创建不同的 negroni 实例并将它们链接起来,因此您的 finalnegroniStack是一个中间件本身,它将执行您添加的其他中间件。

我相信您想要做的是使用实际路由器创建路由,然后将适当的中间件(使用 negroni)添加到每个路由。

如果您查看从文档链接的示例,这就是他们在该部分中所做的事情(https://github.com/urfave/negroni#route-specific-middleware)。

router.PathPrefix("/admin").Handler(negroni.New(
   Middleware1,
   Middleware2,
   negroni.Wrap(adminRoutes),
))

请注意,它们不是嵌套 negroni 实例,而是仅创建一个应用于所需路由的实例。

于 2018-05-03T02:56:41.213 回答