Development, Architecture/Docker, k8s

[쿠버네티스] 스테이트풀셋

ggyongi 2024. 7. 27. 21:07
반응형

스테이트풀셋

주요 용도

안정된 프레임워크에서 동작하는 애플리케이션에 스케일링 기능을 제공하는 파드 컨트롤러.

주로 데이터베이스, 메시지 큐, 로그 처리 시스템 등 상태 정보를 유지해야 하는 애플리케이션에 사용

 

특징

  • 고유한 네트워크 ID: 스테이트풀셋에서 생성되는 각 파드는 규칙적인 이름과 고유한 네트워크 ID(도메인 네임)를 가진다. 예를 들어, 파드 이름이 statefulset-0, statefulset-1, statefulset-2와 같이 부여된다.
  • 순차적 생성 및 종료: 파드는 순서대로 생성되고 종료된다. 첫 번째 파드가 Running 상태가 되면 그 다음 파드가 생성되는 식으로 동작다.
apiVersion: apps/v1
kind: StatefulSet
metadata:
  name: example-statefulset
spec:
  serviceName: "example"
  replicas: 3
  selector:
    matchLabels:
      app: example
  template:
    metadata:
      labels:
        app: example
    spec:
      containers:
      - name: example
        image: example-image
        ports:
        - containerPort: 80
  volumeClaimTemplates:
  - metadata:
      name: example-storage
    spec:
      accessModes: ["ReadWriteOnce"]
      resources:
        requests:
          storage: 1Gi

 

 

 

스테이트풀셋에서 초기화 컨테이너 활용

스테이트풀셋에서 초기화 컨테이너(Init Containers)는 파드가 시작되기 전에 실행되어 필요한 초기화 작업을 수행한다. 이는 데이터베이스 스키마 설정, 초기 데이터 로드 등과 같은 작업을 포함할 수 있다. 만약 이 초기화 작업이 StatefulSet의 다른 파드와 통신해야 한다면, 헤드리스 서비스가 필요하다. 헤드리스 서비스를 사용하면 초기화 컨테이너가 특정 파드에 직접 접근할 수 있습니다. 헤드리스는 실제 주소와 연결된 IP주소가 없는 대신 스테이트풀셋 안에 있는 각 파드의 IP주소가 반환되기 때문이다.

apiVersion: v1
kind: Service
metadata:
  name: my-service
  labels:
    app: myapp
spec:
  clusterIP: None
  selector:
    app: myapp
  ports:
  - port: 80
    name: http

 

apiVersion: apps/v1
kind: StatefulSet
metadata:
  name: my-statefulset
spec:
  serviceName: "my-service"
  replicas: 3
  selector:
    matchLabels:
      app: myapp
  template:
    metadata:
      labels:
        app: myapp
    spec:
      initContainers:
      - name: init-myservice
        image: busybox
        command: ['sh', '-c', 'echo Initializing...; sleep 5']
      containers:
      - name: myapp
        image: myapp:latest
        ports:
        - containerPort: 80
      volumeMounts:
      - name: my-storage
        mountPath: /data
  volumeClaimTemplates:
  - metadata:
      name: my-storage
    spec:
      accessModes: ["ReadWriteOnce"]
      resources:
        requests:
          storage: 1Gi

 

위의 예시에서, my-service는 헤드리스 서비스로 설정되었고 my-statefulset의 각 파드는 my-service를 포함한 고유한 네트워크 ID를 갖게 된다(todo-db-0.my-service.default.svc.cluster.local). 초기화 컨테이너는 필요에 따라 이 네트워크 ID를 사용하여 특정 파드에 접근할 수 있습니다.

 

볼륨 클레임 템플릿으로 스토리지 요청

각 파드의 영구볼륨클레임마다 별도의 스토리지가 마운트되게 하기위해 volumeClaimTemplates 필드를 기술해야함

볼륨 클레임 템플릿을 포함한 스테이트풀셋을 배치하면 각 파드마다 영구볼륨 클레임이 생성되며 파드와 영구볼륨클레임이 연결됨. 예를 들어 파드 0이 대체되었다면 새로운 파드 0에도 기존 영구볼륨클레임이 사용됨

apiVersion: apps/v1
kind: StatefulSet
metadata:
  name: example-statefulset
spec:
  serviceName: "example"
  replicas: 3
  selector:
    matchLabels:
      app: example
  template:
    metadata:
      labels:
        app: example
    spec:
      containers:
      - name: example
        image: example-image
        ports:
        - containerPort: 80
        volumeMounts:
        - name: example-storage
          mountPath: /data
  volumeClaimTemplates:
  - metadata:
      name: example-storage
    spec:
      accessModes: ["ReadWriteOnce"]
      resources:
        requests:
          storage: 1Gi

 

 

주의점

스테이트풀셋은 애플리케이션에 안정적인 환경을 제공하는 것이 목적인 리소스라 업데이트에 있어서는 유연성이 다소 떨어짐.

볼륨 클레임을 추가하는 등의 변경이 있으면 기존 스테이트풀셋을 업데이트할 수 없음(경고 발생함)

 

예시) 기존 정의

 ... 생략 ...
 volumeClaimTemplates:
  - metadata:
      name: www
    spec:
      accessModes: ["ReadWriteOnce"]
      resources:
        requests:
          storage: 1Gi

 

여기서 아래처럼 스토리지 크기를 변경한다.

 ... 생략 ...
 volumeClaimTemplates:
  - metadata:
      name: www
    spec:
      accessModes: ["ReadWriteOnce"]
      resources:
        requests:
          storage: 2Gi  # 변경 시도: 스토리지 크기 변경

 

이제 다시 이 yaml파일을 통해 스테이트풀셋을 배포하면 아래와 같은 경고가 뜸

The StatefulSet "web" is invalid: spec: Forbidden: updates to statefulset spec for fields other than 'replicas', 'template', and 'updateStrategy' are forbidden

 

그래서 이런 변경이 필요하다면 먼저 기존 스테이트풀셋을 삭제하고 다시 새로운 스테이트풀셋을 생성해야함

kubectl delete statefulset web --cascade=orphan   // 기존꺼 삭제

kubectl apply -f new-statefulset.yaml   // 다시 새로운거 생성