前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >K8S容器应用优雅关闭-修复5003 Error

K8S容器应用优雅关闭-修复5003 Error

作者头像
运维部落
发布2021-12-31 15:12:45
8240
发布2021-12-31 15:12:45
举报
文章被收录于专栏:运维部落运维部落
  • 1、遇到的问题
  • 2、问题排查
  • 3. 根因分析
    • 3.1、SHELL 模式和 CMD 模式带来的差异
    • 3.2、直接启动应用和通过脚本启动区别
  • 4、总结

K8S容器应用优雅关闭-修复5003 Error

“运维就要无所不能,无所不会”

大家好,我是Stanley「史丹利」,今天聊技术:容器优雅关闭方案 。

1、遇到的问题

公司某服务接入效能平台后,发布过程中,页面偶尔会出现5003报错,开始以为是Nacos没有及时的将服务反注册,即POD在已经正常关闭的情况下,注册中心依然有POD信息,请求依然到已经关闭的POD中导致

5003报错

5003-error-2

2、问题排查

2.1 首先找开发同学,协助排查了反注册逻辑及相关日志,没有发现什么异常

2.2 后来偶然发现POD中的主进程PID不为1,而PID为1的进程为shell进程,这会导致容器关闭时业务进程无法接受k8s发送的SIGTERM信号,只能在等待15秒后被强行杀死

process-shell

2.3 修改了程序启动参数,通过EXEC启动模式,使应用主进程PID为1

process-exec

2.4 重新发布验证,5003报错问题修复

3. 根因分析

3.1、SHELL 模式和 CMD 模式带来的差异

通常Dockerfile中CMD和ENTRYPOINT来启动应用,启动应用有两种模式,shell 模式和 exec 模式,对应的使用 shell 模式,PID 为 1 的进程为 shell,使用 exec 模式 PID 为 1 的进程为业务本身。

SHELL模式

代码语言:javascript
复制
FROM golang as builder
WORKDIR /go/
COPY app.go    .
RUN go build app.go
FROM ubuntu
WORKDIR /root/
COPY --from=builder /go/app .
CMD ./app

这种方式构建的镜像应用启动后PID为1的进程是shell进程

EXEC模式

代码语言:javascript
复制
FROM golang as builder
WORKDIR /go/
COPY app.go    .
RUN go build app.go
FROM ubuntu
WORKDIR /root/
COPY --from=builder /go/app .
CMD ["./app"]

这种方式构建的镜像应用启动后PID为1的进程是应用进程

3.2、直接启动应用和通过脚本启动区别

在实际生产环境中,因为应用启动命令后会接很多启动参数,所以通常我们会使用一个启动脚本来启动应用,方便我们启动应用。对应的在容器内 PID 为 1 的进程为 shell 进程但 shell 程序不转发 signals,也不响应退出信号。所以在容器应用中如果应用容器中启动 shell,占据了 pid=1 的位置,那么就无法接收 k8s 发送的 SIGTERM 信号,只能等超时后被强行杀死了。启动脚本 start.sh

start.sh

代码语言:javascript
复制
$ cat > start.sh<< EOF
#!/bin/sh
sh -c /root/app
EOF

Dockerfile

代码语言:javascript
复制
FROM golang as builder
WORKDIR /go/
COPY app.go    .
RUN go build app.go
FROM alpine
WORKDIR /root/
COPY --from=builder /go/app .
ADD start.sh /root/
CMD ["/bin/sh","/root/start.sh"]
3.2.1 解决方案
方案一:通过 k8s 的 prestop 参数调用容器内进程关闭脚本,实现优雅关闭。

在前面脚本启动的dockerfile 基础上,定义一个优雅关闭的脚本,通过k8s-prestop 在关闭 POD 前调用优雅关闭脚本,实现 pod 优雅关闭。

stop.sh

代码语言:javascript
复制
#!/bin/sh
ps -ef|grep app|grep -v grep|awk '{print $1}'|xargs kill -15

通过 yaml 部署到 k8s 中

stop.sh

代码语言:javascript
复制
apiVersion: apps/v1
kind: Deployment
metadata:
  name: app-prestop
  labels:
    app: prestop
spec:
  replicas: 1
  selector:
    matchLabels:
      app: prestop
  template:
    metadata:
      labels:
        app: prestop
    spec:
      containers:
      - name: prestop
        image: xx/app:v1.0-prestop
        lifecycle:
          preStop:
            exec:
              command:
              - sh
              - /root/stop.sh
方案二:shell 脚本修改为 exec 执行

修改start.sh脚本

stop.sh

代码语言:javascript
复制
#!/bin/sh
exec ./app

shell 中添加一个 exec 即可让应用进程替代当前 shell 进程,可将 SIGTERM 信号传递到业务层,让业务实现优雅关闭。

方案三:通过第三方 init 进程传递 SIGTERM 到进程中。

使用 dump-inittini 做为容器的主进程,在收到退出信号的时候,会将退出信号转发给进程组所有进程。主要适用应用本身无关闭信号处理的场景。docker –init 本身也是集成的 tini

stop.sh

代码语言:javascript
复制
FROM golang as builder
WORKDIR /go/
COPY app.go    .
RUN go build app.go
FROM alpine
WORKDIR /root/
COPY --from=builder /go/app .
ADD start.sh tini /root/
RUN chmoad a+x start.sh && apk add --no-cache tini
ENTRYPOINT ["/sbin/tini", "--"]
CMD ["/root/tini", "--", /root/start.sh"]

4、总结

1、对于容器化应用启动命令建议使用 EXEC 模式。

2、对于应用本身代码层面已经实现了优雅关闭的业务,但有 shell 启动脚本,容器化后部署到 k8s 上建议使方案一和方案二。

3、对于应用本身代码层面没有实现优雅关闭的业务,建议使用方案三。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2021-12-28,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 运维部落 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • K8S容器应用优雅关闭-修复5003 Error
    • 1、遇到的问题
      • 2、问题排查
        • 3. 根因分析
          • 3.1、SHELL 模式和 CMD 模式带来的差异
          • 3.2、直接启动应用和通过脚本启动区别
        • 4、总结
        相关产品与服务
        容器服务
        腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档