【问题标题】:How to break-out of martini middleware如何破解马提尼中间件
【发布时间】:2020-03-14 05:11:11
【问题描述】:

简而言之,如果我为 http.Request 运行一些中间件并确定该请求应该获得 HTTP 422,我如何“突破”中间件链并提前返回不调用链中的所有中间件函数?

例如,如果我有这个:

func Routes(m *martini.ClassicMartini) {

    m.Get("/cp/users", mw.AsJson, mw.RequestTimer, ctr.GetMany)

}

如果我在上述任何中间件资金中调用return,据我所知,它仍会调用链中注册的所有中间件资金,因此始终会调用ctr.GetMany

有没有办法让请求/响应完成并告诉 martini 停止调用链中的所有函数?

如果第一个返回值是一个整数,我猜 martini 假设它是一个状态码。我目前最好的猜测是根据文档: https://github.com/go-martini/martini#middleware-handlers

我们可以使用这个:

m.Use(func(c martini.Context, w http.ResponseWriter){

    if reqIsMalformed() {
        http.Error(w, "Bad request because xyz", 422)
        return;
    }

    c.Next()

})

如果条件不满足,我们就永远不能调用c.Next()

【问题讨论】:

  • 请显示完整代码,带有一些中间件。
  • 不需要tbh,有一种机制可以在不调用声明链中的所有中间件的情况下返回,或者没有,但很简单,如果有这样的,我不知道该怎么做机制。
  • 我更新了 OP 让它更清晰,我希望它很清楚 thx

标签: go martini


【解决方案1】:

我用路由器/中间件做了一个实验,结果如下(有用的信息在最后):

func check0() {
    return
}

func check01() int {
    return 200
}

func check02() (int, string) {
    return 200, "boop"
}

func check03() bool {
    return true
}

func check04() string {
    return "04"
}

func check1(res http.ResponseWriter) string {
    return "1"
}

func check2(c martini.Context, res http.ResponseWriter) string {
    if true {
        return "hiii"
    }
    c.Next()
    return "2"
}

func check3(c martini.Context, res http.ResponseWriter) string {
    c.Next()
    return "3"
}

func check4(res http.ResponseWriter) {
    res.Write([]byte("4"))
}

func check5(c martini.Context, res http.ResponseWriter) (int, string, string) {
    res.Write([]byte("5.0"))
    c.Next()
    return 200, "5.1x", "5.1y"
}

func finish(res http.ResponseWriter) {
    fmt.Println("in finish")
    res.Write([]byte("all done"))
}

func Routes(m *martini.ClassicMartini) {
    m.Get("/cp/meta/test/middleware0", check0, finish)
    m.Get("/cp/meta/test/middleware01", check01, finish)
    m.Get("/cp/meta/test/middleware02", check02, finish)
    m.Get("/cp/meta/test/middleware03", check03, finish)
    m.Get("/cp/meta/test/middleware04", check04, finish)
    m.Get("/cp/meta/test/middleware1", check1, finish)
    m.Get("/cp/meta/test/middleware2", check2, finish)
    m.Get("/cp/meta/test/middleware3", check3, finish)
    m.Get("/cp/meta/test/middleware4", check4, finish)
    m.Get("/cp/meta/test/middleware5", check5, finish)
    m.Get("/cp/meta/echo_runtime_config", common.AsJson, common.RequestTimer, mw.BodyToMap, ctr.GetRuntimeConfig)
}

这是我点击 api 时的结果:

GET /cp/meta/test/middleware0 => 'all done'
GET /cp/meta/test/middleware01 => ''
GET /cp/meta/test/middleware03 => '<bool Value>'
GET /cp/meta/test/middleware02 => 'boop'
GET /cp/meta/test/middleware1 => '1'
GET /cp/meta/test/middleware04 => '04'
GET /cp/meta/test/middleware2 => 'hiii'
GET /cp/meta/test/middleware3 => 'all done3'
GET /cp/meta/test/middleware4 => '4'
GET /cp/meta/test/middleware5 => '5.0all done5.1x'

本来是要添加到这个问题中的。 所以这里是规则:

  1. 如果中间件函数返回 anything(也就是 func 具有非 void 返回签名),则不会调用后续中间件。
  2. 注入各种参数似乎对是否调用后续中间件(包括 martini.Context 等)没有影响。
  3. 使用 martini.Context.Next() 似乎只对在调用所有其他剩余中间件后运行挂钩有用。
  4. 如果什么都不返回,就会调用剩下的中间件,你显然不需要调用c.Next()。
  5. 如果你返回一个int作为返回列表中的第一个参数,它将被解释为一个http状态码,第二个参数,如果它是任何东西将被写入正文。如果第一个参数是一个字符串,而不是一个 int,那么它将被写入主体。我不确定是使用还是忽略了第三个参数,但它们似乎被忽略了。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-02-22
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多