Hooks
Hook is a generic concept that indicates the action that accompanies an event when it is triggered.
Hertz provides a global Hook for injecting your processing logic on the server-side after triggering startup and before exiting.
StartHook
StartHook is a function to be called after the server-side has triggered a start, represented in Hertz by the type CtxErrCallback. Hertz uses the OnRun property to store the StartHook list.
// CtxErrCallback refer to it's function signatures below
OnRun []CtxErrCallback
Hook functions get triggered sequentially after triggering the startup. Once the call is completed, Hertz will officially start listening on the port, or terminate the service immediately if any error occurs.
Function signatures:
type CtxErrCallback func(ctx context.Context) error
Sample Code:
package main
import (
    "context"
    "github.com/cloudwego/hertz/pkg/app"
    "github.com/cloudwego/hertz/pkg/app/server"
    "github.com/cloudwego/hertz/pkg/common/hlog"
    "github.com/cloudwego/hertz/pkg/common/utils"
    "github.com/cloudwego/hertz/pkg/protocol/consts"
)
func main() {
    h := server.Default()
    h.OnRun = append(h.OnRun, func(ctx context.Context) error {
        hlog.Info("run the first start hook")
        return nil
    })
    h.OnRun = append(h.OnRun, func(ctx context.Context) error {
        hlog.Info("run the second start hook")
        return nil
    })
    h.OnRun = append(h.OnRun, func(ctx context.Context) error {
        hlog.Info("run the third start hook")
        return nil
    })
    h.GET("/ping", func(ctx context.Context, c *app.RequestContext) {
        c.JSON(consts.StatusOK, utils.H{"ping": "pong"})
    })
    h.Spin()
}
Note: After triggering startup, The logs of the three StartHook functions will be printed in the terminal in order.
main.go:17: [Info] run the first start hook
main.go:21: [Info] run the second start hook
main.go:25: [Info] run the third start hook
ShutdownHook
ShutdownHook is a function to be called before the server-side exiting, represented in Hertz by the type CtxCallback. Hertz uses the OnShutdown property to store the ShutdownHook list.
Hook functions get triggered simultaneously before the server-side exiting. The user can configure the max expiration time by server.WithExitWaitTime, the default is 5 seconds, and once timeout, the server is terminated.
The ShutdownHook call process is essentially a part of the Hertz Graceful Shutdown.
Function signatures:
type CtxCallback func(ctx context.Context)
Sample Code1:
package main
import (
    "context"
    "github.com/cloudwego/hertz/pkg/app"
    "github.com/cloudwego/hertz/pkg/app/server"
    "github.com/cloudwego/hertz/pkg/common/hlog"
    "github.com/cloudwego/hertz/pkg/common/utils"
    "github.com/cloudwego/hertz/pkg/protocol/consts"
)
func main() {
    h := server.Default()
    h.OnShutdown = append(h.OnShutdown, func(ctx context.Context) {
        hlog.Info("run the first shutdown hook")
    })
    h.OnShutdown = append(h.OnShutdown, func(ctx context.Context) {
        hlog.Info("run the second shutdown hook")
    })
    h.OnShutdown = append(h.OnShutdown, func(ctx context.Context) {
        hlog.Info("run the third shutdown hook")
    })
    h.GET("/ping", func(ctx context.Context, c *app.RequestContext) {
        c.JSON(consts.StatusOK, utils.H{"ping": "pong"})
    })
    h.Spin()
}
Note: Before exiting, the logs of the three ShutdownHook functions will be printed in the terminal disorderly.
main.go:17: [Info] run the first shutdown hook
main.go:23: [Info] run the third shutdown hook
main.go:20: [Info] run the second shutdown hook
Sample Code2:
package main
import (
    "context"
    "time"
    "github.com/cloudwego/hertz/pkg/app"
    "github.com/cloudwego/hertz/pkg/app/server"
    "github.com/cloudwego/hertz/pkg/common/hlog"
    "github.com/cloudwego/hertz/pkg/common/utils"
    "github.com/cloudwego/hertz/pkg/protocol/consts"
)
func main() {
    h := server.Default(server.WithExitWaitTime(time.Second * 2))
    h.OnShutdown = append(h.OnShutdown, func(ctx context.Context) {
        hlog.Info("run shutdown hook")
        time.Sleep(time.Second * 5)
    })
    h.GET("/ping", func(ctx context.Context, c *app.RequestContext) {
        c.JSON(consts.StatusOK, utils.H{"ping": "pong"})
    })
    h.Spin()
}
Note: When terminating the service, the timeout log is printed because the hook function took more than 2 seconds to execute.
hertz.go:77: [Info] HERTZ: Begin graceful shutdown, wait at most 2s ...
main.go:17: [Info] run shutdown hook
engine.go:276: [Info] HERTZ: Execute OnShutdownHooks timeout: error=context deadline exceeded
OnAccept
In netpoll libaray, OnAccept is a function to be called after connection accepted
but before adding it to epoll.
In go net library, OnAccept is executed after connection accepted but before establishing TLS connection.
OnAccept func(conn net.Conn) context.Context
Sample Code:
package main
import (
    "context"
    "github.com/cloudwego/hertz/pkg/app"
    "github.com/cloudwego/hertz/pkg/app/server"
    "github.com/cloudwego/hertz/pkg/common/utils"
    "github.com/cloudwego/hertz/pkg/protocol/consts"
    "github.com/cloudwego/hertz/pkg/common/hlog"
    "net"
)
func main() {
    h := server.New(
        server.WithOnAccept(func(conn net.Conn) context.Context {
            hlog.Info("run the onAccept")
            return context.Background()
        }),
        server.WithHostPorts("localhost:9230"))
    h.GET("", func(ctx context.Context, c *app.RequestContext) {
        hlog.Info("pong")
        c.JSON(consts.StatusOK, utils.H{"ping": "pong"})
    })
    h.Spin()
}
Note: After sending the request, OnAccept function will be printed in the terminal.
main.go:32: [Info] run the onAccept
main.go:38: [Info] pong
OnConnect
In netpoll library, OnConnect is a function to be called after adding it to epoll. The difference is that onConnect can get data but OnAccept cannot.
In go net, OnConnect is executed after establishing TLS connection.
OnConnect func(ctx context.Context, conn network.Conn) context.Context
Sample Code:
package main
import (
	"context"
	"github.com/cloudwego/hertz/pkg/app"
	"github.com/cloudwego/hertz/pkg/app/server"
	"github.com/cloudwego/hertz/pkg/common/hlog"
	"github.com/cloudwego/hertz/pkg/common/utils"
	"github.com/cloudwego/hertz/pkg/network"
	"github.com/cloudwego/hertz/pkg/protocol/consts"
)
func main() {
	h := server.New(
		server.WithHostPorts("localhost:9229"),
		server.WithOnConnect(func(ctx context.Context, conn network.Conn) context.Context {
			b, _ := conn.Peek(3)
			hlog.Info("onconnect")
			hlog.Info(b)
			return ctx
		}))
	h.GET("/ping", func(ctx context.Context, c *app.RequestContext) {
		c.JSON(consts.StatusOK, utils.H{"ping": "pong"})
	})
	h.Spin()
}
Note: After sending the request, OnConnect function will be printed in the terminal.
main.go:19: [Info] onconnect
main.go:20: [Info] [71 69 84]