使用 HTTP Middleware 可以有效的对 HTTP 的请求体,响应体进行拦截,做一些自定义的操作。

在 go-zero 中使用 HTTP Middleware 共有两种方式。

  1. 全局 Middleware 配置
  2. 路由组 Middleware 配置

接下来以快速开始中最基础的 greet 项目做演示。

准备工作

首先准备一个最基本的 greet 演示项目。以下代码均由 goctl 生成,未进行编辑。

项目结构如下:

$ tree                           
.
├── etc
│   └── greet-api.yaml
├── go.mod
├── go.sum
├── greet.api
├── greet.go
└── internal
    ├── config
    │   └── config.go
    ├── handler
    │   ├── greethandler.go
    │   └── routes.go
    ├── logic
    │   └── greetlogic.go
    ├── svc
    │   └── servicecontext.go
    └── types
        └── types.go

greet/greet.go:

package main

import (
    "flag"
    "fmt"

    "greet/internal/config"
    "greet/internal/handler"
    "greet/internal/svc"

    "github.com/tal-tech/go-zero/core/conf"
    "github.com/tal-tech/go-zero/rest"
)

var configFile = flag.String("f", "etc/greet-api.yaml", "the config file")

func main() {
    flag.Parse()

    var c config.Config
    conf.MustLoad(*configFile, &c)

    ctx := svc.NewServiceContext(c)
    server := rest.MustNewServer(c.RestConf)
    defer server.Stop()

    handler.RegisterHandlers(server, ctx)

    fmt.Printf("Starting server at %s:%d...\n", c.Host, c.Port)
    server.Start()
}

我们先在 Greet 方法中加入一行日志输出,用来代表我们的业务处理逻辑

greet/internal/logic/greetlogic.go :

package logic

import (
    "context"

    "greet/internal/svc"
    "greet/internal/types"

    "github.com/tal-tech/go-zero/core/logx"
)

type GreetLogic struct {
    logx.Logger
    ctx    context.Context
    svcCtx *svc.ServiceContext
}

func NewGreetLogic(ctx context.Context, svcCtx *svc.ServiceContext) GreetLogic {
    return GreetLogic{
        Logger: logx.WithContext(ctx),
        ctx:    ctx,
        svcCtx: svcCtx,
    }
}

func (l *GreetLogic) Greet(req types.Request) (*types.Response, error) {
    // 在这里加入业务逻辑,我们用打印日志来代表业务逻辑
    l.Infof("name: %v", req.Name)
    return &types.Response{}, nil
}

全局 Middleware 配置

首先我们创建一个 Middleware 方法

func middlewareDemoFunc(next http.HandlerFunc) http.HandlerFunc {
    return func(w http.ResponseWriter, r *http.Request) {
        logx.Info("request ... ")
        next(w, r)
        logx.Info("reponse ... ")
    }
}

然后将其注册到 go-zero 的 rest 中

unc main() {
    
    ···
    
    server := rest.MustNewServer(c.RestConf)
    defer server.Stop()
    
    server.Use(
        middlewareDemoFunc,
    )
    
    ···

    server.Start()
}

启动项目,并使用 curl 进行请求,终端输出如下:

curl:

$ curl -i http://localhost:8888/greet/from/you
HTTP/1.1 200 OK
Content-Type: application/json
Date: Fri, 23 Oct 2020 07:42:48 GMT
Content-Length: 14

{"message":""}

server:

$ go run greet.go -f etc/greet-api.yaml
Starting server at 0.0.0.0:8888...
{"@timestamp":"2020-10-23T15:42:48.113+08","level":"info","content":"request..."}
{"@timestamp":"2020-10-23T15:42:48.113+08","level":"info","content":"name: you","trace":"373939c095f9c52f","span":"0"}
{"@timestamp":"2020-10-23T15:42:48.113+08","level":"info","content":"reponse..."}
{"@timestamp":"2020-10-23T15:42:48.113+08","level":"info","content":"200 - /greet/from/you - [::1]:52864 - curl/7.64.1 - 0.3ms","trace":"373939c095f9c52f","span":"0"}

路由组 Middleware 配置

路由组 Middleware 只对特定的路由有效,使用路由组 Middleware 只需我们在注册路由时使用 rest.WithMiddleware() 或 rest.WithMiddlewares() 方法传入我们定义的 Middleware 即可。

goctl 支持生成路由组 Middleware

在 goctl 中可使用以下语法来定义路由组 Middleware

greet/greet.api :

···

@server(
    middleware: GreetMiddleware1, GreetMiddleware2
)
service greet-api {
  @handler GreetHandler
  get /greet/from/:name(Request) returns (Response);
}

···

使用 goctl 命令重新生成路由代码

$ goctl api go -api ./greet.api -dir .
etc/greet-api.yaml exists, ignored generation
internal/config/config.go exists, ignored generation
greet.go exists, ignored generation
internal/svc/servicecontext.go exists, ignored generation
internal/types/types.go exists, overwrite it?
Enter to overwrite or Ctrl-C to cancel...
internal/handler/greethandler.go exists, ignored generation
internal/handler/routes.go exists, overwrite it?
Enter to overwrite or Ctrl-C to cancel...
internal/logic/greetlogic.go exists, ignored generation
Done.

具体代码如下:

没有使用 Middleware 的路由组

greet/internal/handler/routes.go: 

···

func RegisterHandlers(engine *rest.Server, serverCtx *svc.ServiceContext) {
    engine.AddRoutes(
        []rest.Route{
            {
                Method:  http.MethodGet,
                Path:    "/greet/from/:name",
                Handler: greetHandler(serverCtx),
            },
        },
    )
}

···

加入 Middleware 的路由组

greet/internal/handler/routes.go:

···

func RegisterHandlers(engine *rest.Server, serverCtx *svc.ServiceContext) {
    engine.AddRoutes(
        rest.WithMiddlewares(
            []rest.Middleware{serverCtx.GreetMiddleware1, serverCtx.GreetMiddleware2},
            []rest.Route{
                {
                    Method:  http.MethodGet,
                    Path:    "/greet/from/:name",
                    Handler: greetHandler(serverCtx),
                },
            }...,
        ),
    )
 )
    
···

注意:goctl 命令行工具不会覆盖 greet/internal/svc/servicecontext.go 文件,而使用 goctl 定义的 Middleware 方法在 greet/internal/svc/servicecontext.go 中,因此我们需要手动修改 greet/internal/svc/servicecontext.go 文件,添加我们所需要的 serverCtx.GreetMiddleware1serverCtx.GreetMiddleware2 方法。在首次生成代码的新项目中,由于不存在 internal/svc/servicecontext.go 文件,goctl 会生成 Middleware 方法的声明,具体实现需要自己来完成。

这里的 serverCtx.GreetMiddleware1serverCtx.GreetMiddleware2 需要我们自己进行实现。

我们简单实现一下。

 

package svc

import (
    "greet/internal/config"
    "net/http"

    "github.com/tal-tech/go-zero/core/logx"
    "github.com/tal-tech/go-zero/rest"
)

type ServiceContext struct {
    Config           config.Config
    GreetMiddleware1 rest.Middleware
    GreetMiddleware2 rest.Middleware
}

func NewServiceContext(c config.Config) *ServiceContext {
    return &ServiceContext{
        Config:           c,
        GreetMiddleware1: greetMiddleware1,
        GreetMiddleware2: greetMiddleware2,
    }
}

func greetMiddleware1(next http.HandlerFunc) http.HandlerFunc {
    return func(w http.ResponseWriter, r *http.Request) {
        logx.Info("greetMiddleware1 request ... ")
        next(w, r)
        logx.Info("greetMiddleware1 reponse ... ")
    }
}

func greetMiddleware2(next http.HandlerFunc) http.HandlerFunc {
    return func(w http.ResponseWriter, r *http.Request) {
        logx.Info("greetMiddleware2 request ... ")
        next(w, r)
        logx.Info("greetMiddleware2 reponse ... ")
    }
}

启动服务并使用 curl 进行请求,可以观察到如下输出:

curl:

$ curl -i http://localhost:8888/greet/from/you
HTTP/1.1 200 OK
Content-Type: application/json
Date: Fri, 23 Oct 2020 08:14:27 GMT
Content-Length: 14

{"message":""}

server:

$ go run greet.go -f etc/greet-api.yaml
Starting server at 0.0.0.0:8888...
{"@timestamp":"2020-10-23T16:14:27.655+08","level":"info","content":"greetMiddleware1 request ... "}
{"@timestamp":"2020-10-23T16:14:27.655+08","level":"info","content":"greetMiddleware2 request ... "}
{"@timestamp":"2020-10-23T16:14:27.655+08","level":"info","content":"name: you","trace":"102aee4a4a935210","span":"0"}
{"@timestamp":"2020-10-23T16:14:27.655+08","level":"info","content":"greetMiddleware2 reponse ... "}
{"@timestamp":"2020-10-23T16:14:27.655+08","level":"info","content":"greetMiddleware1 reponse ... "}
{"@timestamp":"2020-10-23T16:14:27.655+08","level":"info","content":"200 - /greet/from/you - [::1]:53701 - curl/7.64.1 - 0.3ms","trace":"102aee4a4a935210","span":"0"}