f8b21af4be
10_Wiki/Topics 대규모 정리: - 오류 캡처/미완성 stub 문서 227개 제거 - 교차폴더 중복 43클러스터 병합 (63파일 → redirect) - 링크명 정규화: 깨진 링크 수정·redirect 직결·개념 매핑 ~2,400건 - 카테고리 MOC 6개 신규 생성 - Graph 섹션 미해결 related-keyword 링크 10,058건 제거 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
197 lines
4.9 KiB
Markdown
197 lines
4.9 KiB
Markdown
---
|
|
id: devops-kubernetes-basics
|
|
title: Kubernetes — Deployment / Service / Ingress
|
|
category: Coding
|
|
status: draft
|
|
source_trust_level: B
|
|
verification_status: conceptual
|
|
created_at: 2026-05-09
|
|
updated_at: 2026-05-09
|
|
tags: [devops, kubernetes, k8s, vibe-coding]
|
|
tech_stack: { language: "YAML / kubectl", applicable_to: ["DevOps"] }
|
|
applied_in: []
|
|
aliases: [Deployment, Service, Ingress, ConfigMap, Secret, HPA, probes]
|
|
---
|
|
|
|
# Kubernetes Basics
|
|
|
|
> Container orchestrator. **Pod = 컨테이너 그룹, Deployment = pod replica 관리, Service = 안정 endpoint, Ingress = 외부 노출**. Probes / resources / HPA 가 production hygiene.
|
|
|
|
## 📖 핵심 개념
|
|
- Namespace: 논리 분리.
|
|
- Pod: 1+ 컨테이너 단위, 같은 IP.
|
|
- ReplicaSet: pod 수 유지. (Deployment 가 사용)
|
|
- Deployment: rolling update.
|
|
- Service: stable IP / DNS, load balance.
|
|
- Ingress: HTTP path / host routing.
|
|
|
|
## 💻 코드 패턴
|
|
|
|
### Deployment
|
|
```yaml
|
|
apiVersion: apps/v1
|
|
kind: Deployment
|
|
metadata:
|
|
name: api
|
|
namespace: prod
|
|
spec:
|
|
replicas: 3
|
|
selector: { matchLabels: { app: api } }
|
|
strategy:
|
|
type: RollingUpdate
|
|
rollingUpdate: { maxUnavailable: 0, maxSurge: 1 }
|
|
template:
|
|
metadata: { labels: { app: api } }
|
|
spec:
|
|
containers:
|
|
- name: api
|
|
image: ghcr.io/myco/api:1.2.3
|
|
ports: [{ containerPort: 8080 }]
|
|
env:
|
|
- { name: NODE_ENV, value: production }
|
|
- { name: DB_URL, valueFrom: { secretKeyRef: { name: db, key: url } } }
|
|
resources:
|
|
requests: { cpu: 100m, memory: 256Mi }
|
|
limits: { cpu: 500m, memory: 512Mi }
|
|
readinessProbe:
|
|
httpGet: { path: /healthz, port: 8080 }
|
|
initialDelaySeconds: 5
|
|
periodSeconds: 5
|
|
livenessProbe:
|
|
httpGet: { path: /livez, port: 8080 }
|
|
initialDelaySeconds: 30
|
|
periodSeconds: 10
|
|
startupProbe:
|
|
httpGet: { path: /healthz, port: 8080 }
|
|
failureThreshold: 30
|
|
periodSeconds: 2
|
|
```
|
|
|
|
### Service
|
|
```yaml
|
|
apiVersion: v1
|
|
kind: Service
|
|
metadata: { name: api, namespace: prod }
|
|
spec:
|
|
type: ClusterIP
|
|
selector: { app: api }
|
|
ports: [{ port: 80, targetPort: 8080 }]
|
|
```
|
|
|
|
### Ingress
|
|
```yaml
|
|
apiVersion: networking.k8s.io/v1
|
|
kind: Ingress
|
|
metadata:
|
|
name: api
|
|
namespace: prod
|
|
annotations:
|
|
cert-manager.io/cluster-issuer: letsencrypt
|
|
spec:
|
|
ingressClassName: nginx
|
|
tls:
|
|
- hosts: [api.myco.com]
|
|
secretName: api-tls
|
|
rules:
|
|
- host: api.myco.com
|
|
http:
|
|
paths:
|
|
- path: /
|
|
pathType: Prefix
|
|
backend: { service: { name: api, port: { number: 80 } } }
|
|
```
|
|
|
|
### ConfigMap & Secret
|
|
```yaml
|
|
apiVersion: v1
|
|
kind: ConfigMap
|
|
metadata: { name: app-config }
|
|
data:
|
|
LOG_LEVEL: info
|
|
FEATURE_X: "true"
|
|
---
|
|
apiVersion: v1
|
|
kind: Secret
|
|
metadata: { name: db }
|
|
type: Opaque
|
|
stringData:
|
|
url: postgres://app:pw@db:5432/app
|
|
```
|
|
|
|
### HPA (autoscale)
|
|
```yaml
|
|
apiVersion: autoscaling/v2
|
|
kind: HorizontalPodAutoscaler
|
|
metadata: { name: api }
|
|
spec:
|
|
scaleTargetRef: { apiVersion: apps/v1, kind: Deployment, name: api }
|
|
minReplicas: 2
|
|
maxReplicas: 20
|
|
metrics:
|
|
- type: Resource
|
|
resource: { name: cpu, target: { type: Utilization, averageUtilization: 70 } }
|
|
```
|
|
|
|
### PDB (graceful disruption)
|
|
```yaml
|
|
apiVersion: policy/v1
|
|
kind: PodDisruptionBudget
|
|
metadata: { name: api }
|
|
spec:
|
|
minAvailable: 2
|
|
selector: { matchLabels: { app: api } }
|
|
```
|
|
|
|
### kubectl 자주 쓰는
|
|
```bash
|
|
kubectl get pods -n prod
|
|
kubectl logs -f deployment/api -n prod
|
|
kubectl describe pod <pod> -n prod
|
|
kubectl exec -it <pod> -n prod -- sh
|
|
kubectl rollout restart deployment/api -n prod
|
|
kubectl rollout undo deployment/api -n prod
|
|
kubectl top pods -n prod
|
|
kubectl port-forward svc/api 8080:80 -n prod
|
|
```
|
|
|
|
### Helm chart 구조
|
|
```
|
|
charts/api/
|
|
Chart.yaml
|
|
values.yaml # 기본값
|
|
values-prod.yaml # 환경별 override
|
|
templates/
|
|
deployment.yaml
|
|
service.yaml
|
|
ingress.yaml
|
|
```
|
|
|
|
## 🤔 의사결정 기준
|
|
| 상황 | 도구 |
|
|
|---|---|
|
|
| 단순 배포 | raw YAML + kustomize |
|
|
| Reusable / 환경 다양 | Helm chart |
|
|
| 다중 환경 GitOps | Argo CD / Flux |
|
|
| 복잡한 cron | Job / CronJob |
|
|
| Stateful (DB) | StatefulSet (또는 DB 매니지드) |
|
|
| Sidecars (proxy, log) | initContainer / sidecar |
|
|
|
|
## ❌ 안티패턴
|
|
- **Liveness만, Readiness 없음**: warmup 안 끝났는데 트래픽.
|
|
- **Resources 없음**: 한 pod 가 노드 장악 / OOM.
|
|
- **Limit = Request**: throttling 심해짐. limit 좀 여유.
|
|
- **Latest tag**: 재현 불가. semver tag.
|
|
- **Secret YAML 에 평문**: SOPS / Sealed Secrets / External Secrets.
|
|
- **Replicas 1 + PDB minAvailable 1**: 노드 drain 못 함.
|
|
- **maxUnavailable 1 + replicas 2**: 50% 다운. 0/1 권장.
|
|
- **HPA 없음 prod**: 트래픽 spike 죽음.
|
|
|
|
## 🤖 LLM 활용 힌트
|
|
- Probes 3종 + resources + HPA + PDB 항상.
|
|
- Helm + Argo CD GitOps.
|
|
- ConfigMap 비밀X, Secret 평문X (SOPS).
|
|
|
|
## 🔗 관련 문서
|
|
- [[DevOps_Terraform_Patterns]]
|
|
- [[Backend_Cron_Patterns]]
|