前言
k8s
本身就支持服务滚动升级,但是如果程序没有正确的处理退出信号时,就会导致部分请求直接被中断从而影响用户体验。
滚动升级步骤
每个pod
代表一个集群中的节点,在 k8s 做rolling-update
的时候默认会向旧的pod
发送一个SIGTERM
信号,如果应用没有对SIGTERM
信号做处理的话,会立即强制退出程序,这样的话会导致有些请求还没处理完,前端应用请求错误。
先来回顾下 k8s 的滚动升级步骤:
- 启动一个新的 pod
- 等待新的 pod 进入 Ready 状态
- 创建 Endpoint,将新的 pod 纳入负载均衡
- 移除与老 pod 相关的 Endpoint,并且将老 pod 状态设置为 Terminating,此时将不会有新的请求到达老 pod
- 给老 pod 发送 SIGTERM 信号,并且等待 terminationGracePeriodSeconds 这么长的时间。(默认为 30 秒)
- 超过 terminationGracePeriodSeconds 等待时间直接强制 kill 进程并关闭旧的 pod
这里要注意,SIGTERM信号如果进程没有处理的话也其实也就会导致进程被强杀
,如果处理了但是超过terminationGracePeriodSeconds
配置的时间也一样会被强杀,所以这个时间可以根据具体的情况去设置。
SpringBoot 处理 SIGTERM 信号
在SpringBoot
中处理 SIGTERM 信号非常简单,只需要一个@PreDestroy
注解就可以监听到:
@SpringBootApplication
public class Application {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
@PreDestroy
public void shutdown() {
System.out.println("shutdown")
}
}
通过容器生命周期 hook 来优雅停止
在 pod 中容器将停止前,会执行PreStop hook
,hook 可以执行一个HTTP GET
请求或者exec
命令,并且它们执行是阻塞的,可以利用这个特性来做优雅停止。
-
调用
HTTP GET
"lifecycle": { "preStop": { "httpGet": { "path": "/shutdown", "port": 3000, "scheme": "HTTP" } } }
-
调用
exec
"lifecycle": { "preStop": { "exec": { "command": ["/bin/sh", "-c", "sleep 30"] } } }
这样的好处是可以在 k8s 层面来解决优雅停机的问题,而不需要应用程序对SIGTERM
信号做处理。
关于 PreStop 和 terminationGracePeriodSeconds
- 如果有
PreStop hook
会执行PreStop hook
。 -
PreStop hook
执行完成后会向 pod 发送SIGTERM
信号。 - 如果在
terminationGracePeriodSeconds
时间限制内,PreStop hook
没有执行完的话,一样会直接发送SIGTERM
信号,并且时间延长 2 秒。
即在有PreStop hook
的情况下,也是在terminationGracePeriodSeconds
时间限制内,在超过这个时间点之后,还会给出 2 秒进程处理SIGTERM
信号的时间,最后直接强杀。
以上情况已经过 k8s 上验证过,参考:https://kubernetes.io/docs/concepts/workloads/pods/pod/#termination-of-pods
本文首发于我的博客:https://monkeywie.cn,欢迎收藏!不定期分享
JAVA
、Golang
、前端
、docker
、k8s
等干货知识。