Kubernetes Pod Termination

Kubernetes Pod Termination

1. Kuberentes Pod Termination

[Figure 1] Kubernetes Pod Termination

[Figure 1] Kubernetes Pod Termination

Pod가 종료되도 Pod 안에서 동작하는 App Container가 안정적으로 Request을 처리하며 Gracefully Termination을 위해서는 Pod의 종료 과정을 완전히 이해하고 적절하게 Pod를 설정해야한다. [Figure 1]은 상세한 Pod의 종료 과정을 나타내고 있다.

  1. K8s API 서버는 Pod 종료 요청을 받으면 종료 요청을 받은 Pod가 동작하는 Node의 kubelet에게 Pod 종료 요청을 전달한다. 또한 K8s API 서버는 Endpoint Slice Controller에게도 Pod 종료를 전달한다.
  2. Pod 종료 요청을 받은 Kubelet은 Pod 안에서 동작하고 있는 App Container의 PreStop Hook을 동작시킨다.
  3. Endpoint Slice Controller는 종료될 Pod를 Endpoint Slice에서 제거하여 kube-proxy가 신규 Request를 종료될 Pod로 전달되지 못하도록 만든다.
  4. PreStop Hook의 동작이 완료되면 kubelet은 SIGTERM Signal을 전송한다. SIGTERM Signal을 받은 App Container는 현재 처리중인 Request를 모두 완료하고 종료를 시도한다.
  5. 만약 SIGTERM을 받은 App Container가 종료되지 않으면 SIGKILL Signal을 받고 강제로 종료된다. kubelet은 K8s API 서버가 받은 Pod 종료 요청 시간부터 App Container에 설정된 terminationGracePeriodSeconds 시간만큼 대기후에 SIGKILL Signal을 전송한다.
[Figure 2] Kubernetes Pod Termination with Gracefully Termination

[Figure 2] Kubernetes Pod Termination with Gracefully Termination

App Container가 우아하게 종료되기 위해서는 [Figure 2]와 같이 마지막 SIGKILL 과정을 제외한 1~4의 과정이 수행되어야 한다. 1,3번의 과정은 Kubernetes 내부적으로 자동으로 수행되는 과정이지만, 2번의 과정은 Pod 내부의 App Container에 PreStop Hook을 설정하지 않으면 수행되지 않으며, 4번의 과정은 App Container에서 SIGTERM Signal을 처리하지 않도록 설정하면 수행되지 않기 때문에 관리가 필요하다. 또한 App Container가 SIGKILL을 받으면 Request가 제대로 처리되지 않고 강제로 종료될 수 있기 때문에 terminationGracePeriodSeconds의 값도 적절하게 설정해야한다.

1.1. PreStop Hook 설정

[Figure 3] Kubernetes Pod Termination without PreStop Hook

[Figure 3] Kubernetes Pod Termination without PreStop Hook

[Figure 3] 과정은 App Container에 PreStop Hook을 설정하지 않았을 경우 발생할 수 있는 App Container로 전달된 Request가 처리되지 못한 문제를 나타내고 있다. App Container에 PreStop Hook이 설정되어 있지 않으면 Pod Termination 요청을 받은 kubelet은 SIGTERM Signal을 곧바로 전송한다. SIGTERM Signal을 받은 App Container는 이후에 전달받은 Request를 처리하지 않거나, 곧바로 종료되어 제거될 수 있다.

문제는 Pod Termination 요청 Endpoint Slice Controller가 처리하고 다시 kube-proxy로 전달되고, kube-proxy가 다시 iptable/IPVS Rule을 설정하는데 시간이 소요된다는 점이다. 즉 Endpoint Slice Propagation Delay가 발생하고 이로 인해서 SIGTERM Signal을 받은 App Container는 이후에도 짧은 시간동안 다른 Pod로부터 Request를 전달 받을수 있다. [Figure 1]처럼 App Container에 PreStop Hook이 설정되어 있다면 App Container가 늦게 SIGTERM Signal을 받고 늦게 종료가 시작되기 때문에 Endpoint Slice Propagation Delay로 인해서 늦게 Request가 도착해도 문제없이 처리가 가능해 진다.

전반적인 Endpoint Slice Propagation Delay는 파악이 어렵지만 kube-proxy가 iptable/IPVS Rule을 설정하는데 소모된 시간은 kube-proxy가 제공하는 kubeproxy_sync_proxy_rules_duration_seconds Metric을 통해서 확인할 수 있다.

1
2
3
4
5
6
7
8
spec:
  terminationGracePeriodSeconds: 60
  containers:
  - name: "app"
    lifecycle:
      preStop:
        exec:
          command: ["/bin/sh","-c","sleep 5"]
[File 1] PreStop Hook sleep command Example

PreStop Hook은 SIGTERM Signal을 늦게 받기 위한 용도로 활용되기 때문에 일반적으로는 [File 1]의 내용과 같이 sleep 명령어를 활용하여 구성한다. 따라서 App Container Image에는 sleep 명령어와, Shell이 설치되어 있어야 한다. 일반적으로는 5초 정도로 설정하며, 너무 큰 값을 설정하면 Pod 종료가 늦어져 배포 속도가 느려지기 때문에 적절한 값을 설정해야 한다. 추후에는 Kubernetes 자체에서 Sleep 기능을 제공할 예정이며 자세한 내용은 Link에서 확인할 수 있다.

1.2. App Container의 SIGTERM 처리

[Figure 4] Kubernetes Pod Termination without PreStop Hook

[Figure 4] Kubernetes Pod Termination without PreStop Hook

Linux 환경에서 SIGTERM Signal Handler가 설정되지 않는 Application (Process)는 SIGTERM Signal을 받는 순간 죽으며, App Container도 동일하다. [Figure 4]는 App Container가 SIGTERM Handler가 설정되지 않았을 때를 나타내고 있다. SIGTERM을 받자마자 App Container가 제거되기 때문에, 현재 처리중인 Request를 제대로 처리하지 못하고 종료될 수 있다. 따라서 App Container는 SIGTERM Signal을 받더라도 현재 처리중 Request를 완료하고 죽도록 설정되어 있어야 한다.

1
2
server:
  shutdown: graceful
[File 2] SpringBoot SIGTEM Handler Configuration

대부분의 App Server Framework에서는 SIGTERM Signal을 손쉽게 처리할 수 있는 Gracefully Termination 설정을 제공하기 때문에, 직접 SIGTERM Handler를 작성할 필요는 없으며, App Container의 경우에도 동일하다. [File 2]는 SpringBoot의 예제를 나타내고 있다. Spring Boot에 shutdown: graceful을 설정하면 SIGTERM Signal을 수신하는 순간 신규 Request의 처리는 거절되며, 현재 처리중인 모든 Request가 완료 된 이후에 종료된다.

SpringBoot 뿐만이 아니라 대부분의 App Server Framework에서는 Gracefully Termination이 동작하면 SIGTERM Signal을 수신하는 순간 신규 Request도 거절하기 때문에, SIGTERM Signal을 App Container로 전송한 이후에는 신규 Request도 App Container로 전달되면 안되며, 이러한 역할은 PreStop Hook이 수행한다.

App Container가 SIGTERM Signal을 처리하지 않는 상태에서 Gracefully Termination을 수행하기 위해서는 PreStop Hook의 지속 시간을 App Container의 Request 처리 시간 이상으로 늘리는 방법이 존재한다. PreStop Hook이 길어질 수록 App Container가 SIGTERM Signal을 받는 시간이 늘어나고 그만큼 현재 처리중인 Request를 완료할 수 있는 시간을 얻을 수 있기 때문이다. 하지만 PreStop Hook이 길어질 수록 Pod 종료 시간도 길어지고 그 만큼 Pod 배포 시간도 증가하기 때문에 가능하면 App Container에서 SIGTERM Signal Handler를 설정하는 방법이 권장된다.

1.3. SIGKILL 설정

terminationGracePeriodSeconds은 kubelet이 PreStop Hook을 실행하고 난뒤 SIGKILL Singal을 전송하기 전까지 대기하는 시간이다. Linux 환경에서 SIGTERM Signal과 달리 SIGKILL Signal을 받는 Application (Process)는 반드시 죽는다. 따라서 terminationGracePeriodSeconds 값은 PreStop Hook 시간과 App Container에서 대부분의 요청을 처리하는 시간의 합보다 반드시 커야한다. terminationGracePeriodSeconds의 기본값은 30초이며 Pod 내부의 각 Container마다 설정할 수 없고 Pod 전체에만 설정이 가능하다.

1.4. Gracefully Termination 고려사항

Pod의 Gracefully Termination을 위해서는 App Container의 PreStop Hook의 지속 시간, Pod의 terminationGracePeriodSeconds의 지속 시간 등 다양한 지속 시간들을 적절하게 설정해야 한다. 이러한 지속 시간들을 정확하게 결정하는 공식은 존재하지 않으며 다양한 요소에 따라서 변경이 될 수 있다.

PreStop Hook의 경우에는 일반적으로 5초로 설정하지만 Kubernetes Cluster 내부에서 Endpoint Slice의 변경이 많다면 5초 이상으로 설정이 필요할 수 있다. 또는 App Container에서 SIGTERM Signal Handler가 설정되어 있지 않아면 App Container의 Request 처리 시간 이상으로 PreStop Hook을 설정해야 한다. Pod의 terminationGracePeriodSeconds 지속 시간은 App Container의 App Container의 Request 처리 시간으로 결정된다. 다양한 고려 요소들이 존재하고 명확한 공식이 존재하지 않기 때문에 일반적으로는 Pod 재시작을 반복하는 환경에서도 Request 처리에 문제가 없는지를 검증하며, Heuristic 방식으로 지속 시간을 설정한다.

Pod의 Gracefully Termination을 설정해도 Kubernetes는 Pod가 종료될때는 항상 Gracefully Termination이 수행되는 것을 보장하지는 않는다. 대표적인 예로 Kubernetes Cluster의 특정 Node가 장애로 죽을 경우에, 죽은 Node에서 동작하던 Pod는 당연하게도 Gracefully Termination을 수행하지 못하고 종료된다. 또는 App Container가 대부분의 Request를 1초 안에 처리하지만 간혹 특정 Request의 경우에는 오랜 시간이 소요되어 Pod에 설정된 terminationGracePeriodSeconds 시간 이후에도 Request을 완료하지 못한다면 SIGKILL Signal을 통해서 강제로 종료될 수 있다.

Gracefully Termination을 수행하지 못한 경우에 발생할 수 있는 Business Logic의 영향은 Client 또는 Sidecar Container에서 재요청을 수행하여 대부분 완화가 가능하다. 따라서 Gracefully Termination을 위한 지속 시간을 설정할때도 일반적으로는 모든 Request를 100% 완전히 처리하며 종료되는 것을 목표로 긴 지속 시간을 설정하는것 보다는, 99%와 같이 일부 Request의 완료를 포기하고 대신 짧은 지속 시간을 설정하여 빠른 배포가 가능하도록 설정하는 방식이 더 올바른 방식이다.

2. 참조