[Kubernetes] Deployment란
Deployment
Deployment는 ReplicaSet의 상위 개념으로, Pod와 ReplicaSet에 대한 배포를 관리합니다. 운영 중에 어플리케이션의 새 버전을 배포해야하거나 부하가 증가하면서 Pod를 추가하는 등 여러 가지 동작을 Deployment로 관리할 수 있습니다.
또한 Deployment는 배포에 대한 이력을 관리하는데 만약 배포한 새버전의 문제가 생긴 경우 Deployment를 통해 쉽게 이전 버전으로 롤백할 수 있습니다. 쿠버네티스로 서비스를 운영하는 상황이라면 ReplicaSet만으로 운영하기 보다는 대부분 Deployment단위로 Pod와 ReplicaSet을 관리하여 운영합니다.
Deployment 구성
Deployment를 구성합니다. 기본적인 구성은 다음과 같습니다.
apiVersion: apps/v1
kind: Deployment
metadata:
name: demo-deployment
spec:
replicas: 2
selector:
matchLabels:
app: nginx
template:
metadata:
labels:
app: nginx
spec:
containers:
- name: demo-container
image: nginx:latest
- .spec.replicas
- 생성할 Pod의 수를 지정해줍니다.
- .spec.selector
- 식별할 Pod를 지정해줍니다.
- .spec.selector.matchLables를 통해 식별할 Pod의 라벨을 지정해줍니다.
- .spec.template
- Pod의 구성을 정의합니다.
Deployment 생성
YAML 파일을 구성한 후 다음 명령어를 통해 Deployment를 생성합니다.
$ kubectl apply -f deployment.yaml
deployment.apps/demo-deployment created
다음 명령어를 통해 생성된 Deployment를 확인할 수 있습니다.
$ kubectl get deploy
NAME READY UP-TO-DATE AVAILABLE AGE
demo-deployment 2/2 2 2 4m5s
Deployment를 생성하게 되면 Deployment의 이름을 포함하고있는 ReplicaSet이 생성되는 것을 확인할 수 있습니다.
$ kubectl get replicaset
NAME DESIRED CURRENT READY AGE
demo-deployment-5dd594bdcb 2 2 2 10s
그리고 Deployment에 의해 생성된 Pod는 ReplicaSet의 이름을 가지고 있는 것을 확인할 수 있습니다.
$ kubectl get pod
NAME READY STATUS RESTARTS AGE
demo-deployment-5dd594bdcb-7h8tg 1/1 Running 0 31s
demo-deployment-5dd594bdcb-rjrx2 1/1 Running 0 31s
Deployment 업데이트
Deployment의 업데이트 전략에는 크게 두 가지가 있습니다.
- RollingUpdate
- Recreate
RollingUpdate
RollingUpdate 배포 전략은 무중단 배포 전략으로 기존 버전의 Pod를 하나씩 삭제하고 새 버전의 Pod를 생성하면서 순차적으로 교체하는 방법입니다. Pod가 순차적으로 교체되기 때문에 다운타임이 발생하지 않는다는 장점이 있지만, 이전 버전의 Pod와 새 버전의 Pod가 공존하는 시간이 발생하게됩니다.
다음과 같이 Rolling Update를 구성합니다.
apiVersion: apps/v1
kind: Deployment
metadata:
name: demo-deployment
spec:
replicas: 10
selector:
matchLabels:
app: nginx
strategy:
type: RollingUpdate # .spec.strategy.type을 RollingUpdate로 지정 생략시 기본값으로 RollingUpdate로 지정됨
rollingUpdate:
maxSurge: 3
maxUnavailable: 2
minReadySeconds: 10 # Pod의 Status가 Ready가 될때까지의 최소대기시간, Pod의 변화를 관찰하기위해 지정해줌
template:
metadata:
labels:
app: nginx
spec:
containers:
- name: demo-container
image: nginx:latest
RollingUpdate 방식을 사용할 때 maxSurge와 maxUnavailable을 명시하여 Rolling Update의 프로세스를 제어할 수 있습니다.
- maxSurge
- Rolling Update 시 동시에 생성할 수 있는 Pod의 최대 개수 지정, 개수 혹은 비율(%)로 지정 가능, 기본값 25%
- 개수로 지정하게 되는 경우 백분율로 올림하여 계산합니다.
- maxUnavailable이 0일 경우 0이될 수 없습니다.
- 만약 maxSurge의 값이 30% 라면 업데이트하는 동안 항상 실행(Running) 중인 파드의 수가 최대 replicas의 130%가 되도록 보장합니다.
- maxUnavailable
- Rolling Update 시 동시에 삭제할 수 있는 Pod의 최대 개수 지정, 개수 혹은 비율(%)로 지정 가능, 기본값 25%
- 개수로 지정하게 되는 경우 백분율로 내림하여 계산합니다.
- maxSurge가 0일 경우 0이될 수 없습니다.
- 만약 maxUnavailable의 값이 30% 라면 업데이트 하는 동안 항상 사용가능(Available)한 파드의 수가 최소 replicas의 70% 이상이 되도록 보장합니다.
다음 명령어를 통해 Deployment를 생성합니다.
$ kubectl apply -f deployment.yaml
Deployment의 이미지 부분을 변경합니다.
spec:
containers:
- name: demo-container
image: httpd:latest
다음 명령어를 통해 변경된 내용을 적용합니다.
$ kubectl apply -f deployment.yaml
다음 명령어를 통해 Pod의 변화를 관찰하실 수 있습니다.
$ while true;do kubectl describe rs | grep Replicas && kubectl describe rs | grep "Pods Status" && kubectl get deploy; sleep 1; done
Rolling Update의 동작 과정은 다음과 같습니다.
- v2 버전의 ReplicaSet을 생성한다. 이때 replicas는 (maxSurge + maxUnavailable)이 되고 해당 수치 만큼 Pod를 생성한다.
- v1 버전의 ReplicaSet의 replicas가 (replicas - maxUnavailable)로 변경되고, 해당 수치 만큼 Pod를 제거한다.
- 배포가 진행되는 동안 v1 버전과 v2 버전에 트래픽이 분산된다.
- v2 버전의 replicas를 1 증가시키고, v1 버전의 replicas를 1 감소시킨다. (v2 버전의 replicas가 템플릿에 정의된 replicas와 일치할때까지 반복)
- v1 버전의 replicas를 0으로 변경한 후 남은 Pod를 삭제한다.
Recreate
Recreate 배포 전략은 가장 단순한 배포 전략으로, 기존 버전의 Pod를 모두 삭제한 후 새 버전의 Pod를 생성하는 방법입니다. Recreate 배포 전략은 단순하지만 위 그림처럼 새 버전의 ReplicaSet이 생성되기 전까지 다운타임이 발생할 수 있다는 단점이 있습니다.
다음과 같이 Recreate를 구성합니다.
apiVersion: apps/v1
kind: Deployment
metadata:
name: demo-deployment
spec:
replicas: 10
selector:
matchLabels:
app: nginx
strategy:
type: Recreate # .spec.strategy.type을 Recreate로 지정
minReadySeconds: 10
template:
metadata:
labels:
app: nginx
spec:
containers:
- name: demo-container
image: nginx:latest
RollingUpdate와 마찬가지로 Deployment를 배포한 뒤 다음과 같이 Deployment의 이미지 부분을 변경합니다.
spec:
containers:
- name: demo-container
image: httpd:latest
다음 명령어를 통해 변경된 내용을 적용합니다.
$ kubectl apply -f deployment.yaml
Recreate의 동작 원리는 다음과 같습니다.
- v1 버전의 ReplicaSet의 replicas를 0으로 변경한다.
- v1 버전의 Pod가 제거 된다.
- v2 버전의 ReplicaSet이 생성된다.
- v2 버전의 Pod가 생성된다.
참고