每当服务由于各种原因需要关闭时,常见的是操作系统中断,我们希望我们的服务能够优雅地关闭。

我们希望我们的golang服务停止接收新的请求,同时完成正在进行的请求并返回其响应,然后最终关闭。

我们可以通过创建一个带有cancel()回调函数的context对象和http.Server接口提供的关闭方法来实现这一任务。
通过一个channel持续监听操作系统的中断,然后调用context的cancel()函数。

我们使用该context创建一个服务器实例,然后在context完成后关闭服务。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
package main

import (
"context"
"fmt"
"log"
"net/http"
"os"
"os/signal"
"time"
)

func serve(ctx context.Context) (err error) {

mux := http.NewServeMux()
mux.Handle("/", http.HandlerFunc(
func(w http.ResponseWriter, r *http.Request) {
fmt.Fprintf(w, "okay")
},
))

srv := &http.Server{
Addr: ":6969",
Handler: mux,
}

go func() {
if err = srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
log.Fatalf("listen:%+s\n", err)
}
}()

log.Printf("server started")

<-ctx.Done()

log.Printf("server stopped")

ctxShutDown, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer func() {
cancel()
}()

if err = srv.Shutdown(ctxShutDown); err != nil {
log.Fatalf("server Shutdown Failed:%+s", err)
}

log.Printf("server exited properly")

if err == http.ErrServerClosed {
err = nil
}

return
}

func main() {

c := make(chan os.Signal, 1)
signal.Notify(c, os.Interrupt)

ctx, cancel := context.WithCancel(context.Background())

go func() {
oscall := <-c
log.Printf("system call:%+v", oscall)
cancel()
}()

if err := serve(ctx); err != nil {
log.Printf("failed to serve:+%v\n", err)
}
}