쿠버네티스는 보안측면에서도 다양한 기능 제공
RBAC를 기반으로 하는 서비스 어카운트
서비스 어카운트 : 사용자 또는 어플리케이션 하나에 해당
RBAC(Role based Access Control) : 특정명령을 실행할 수 있는 권한 (서비스를 읽을 권리, 포드를 생성할 권리)
지금까지 사용했던 kubectl 명령어는 최상위 권한에 해당
실제로 cat ~/.kube/config를 보면 kubectl 권한 등이 들어있음.
그래서 여러명의 사용자가 쿠버네티스 접근 -> kubectl을 제공하는 것은 옳지않음.
우리가 실제로 kubectl
명령어를 사용하면 내부구현은 다음과 같은 일이 일어남.
Auth를 구현하기 위해서는 Service Account 방법 외에도 Oauth
, 인증서
방법등이 있다고함
서비스 어카운트는 체계적으로 권한을 관리하기 위한 오브젝트
네임 스페이스에 속하는 오브젝트
kubectl create sa {계정명}
# or
kubectl create serviceaccount {계정명}
kubectl get sa {계정명}
# or
kubectl get serviceaccount {계정명}
### ex
kubectl get services --as system:serviceaccount:{네임스페이스}:{계정명}
서비스 어카운트는 부여된 Role이외의 명령어는 실행할 수 없다.
쿠버네티스에서 권한을 부여하는 방법은 크게 2가지
롤 (Role)
네임스페이스에 속하는 오브젝트로 다른 네임스페이스에 속하는 오브젝트들에 대한 권한을 정의
클러스터 롤 (Cluster Role)
클러스터 단위의 권한을 정의
네임스페이스에 속하지 않는 전역적인 쿠버네티스 오브젝트
롤과 클러스터 롤은 모두 쿠버네티스 오브젝트
따라서
kubectl get role
No resources found.
# 쿠버네티스 현재 네임스페이스에 속하는 권한 출력
kubectl get clusterrole
#전역적이므로, 클러스터 자체에 존재하는 모든 클러스터 롤의 목록을 출력
apiVersion: rbac.authorization.k8s.io/v1
kind: Role
metadata:
namespace: default
name: service-reader
rules:
- apiGroups: [""]
resources: ["services"]
verbs: ["get","list"]
kubectl apply -f service-reader-role.yaml
롤을 적용했다고 해서, 서비스 어카운트에 해당 권한이 부여되는 것은 아님.
apiVersion: rbac.authorization.k8s.io/v1
kind: RoleBinding
metadata:
name: service-reader-rolebinding
namespace: default
subjects:
- kind: ServiceAccount # 권한을 부여할 대상이 ServiceAccount
name: ehdrms2034 # ehdrms2034이라는 이름의 서비스 어카운트에 권한을 부여
namespace: default
roleRef:
kind: Role # Role에 정의된 권한을 부여
name: service-reader # service-role라는 이름의 Role을 대상에 연결
apiGroup: rbac.authorization.k8s.io
kubectl apply -f rolebiding-service-reader.yaml
kubectl get services --as system:serviceaccount:default:ehdrms2034
### 부여받지 않은 권한 사용하면 안됨
kubectl get deployment --as system:serviceaccount:default:ehdrms2034
클러스터 롤은 네임스페이스에 한정되지 않은 오브젝트에 권한을 부여하기 위해 사용됨
기본적으로 롤은 네임스페이스에 한정된 오브젝트에 권한을 부여하기 위해 사용
#node는 namespace에 한정되지 않은 오브젝트인데, role로 권한을 부여받은 사용자가 명령을 수행하면 오류가남.
kubectl get nodes --as system:serviceaccount:default:ehdrms2034
이럴 경우는 클러스터 롤을 사용함.
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
namespace: default # namespace에 한정되지 않는다 했는데 왜 부여하는지..?
name: nodes-reader
rules:
- apiGroups: [""]
resources: ["nodes"]
verbs: ["get", "list"]
kubectl apply -f nodes-reader-clusterrole.yaml
#클러스터 롤 권한 확인하기
kubectl describe clusterrole {클러스터 롤 네임}
#ex kubectl describe clusterrole nodes-reader
클러스터 롤 역시 바인딩이 필요함
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRoleBinding
metadata:
name: nodes-cluster-clusterrolebinding
namespace: default
subjects:
- kind: ServiceAccount
name: ehdrms2034
namespace: default
roleRef:
kind: ClusterRole
name: nodes-reader
apiGroup: rbac.authorization.k8s.io
kubectl apply -f clusterrolebinding-nodes-reader.yaml
#### 권한 부여 확인해보기
kubectl get nodes --as system:serviceaccount:default:ehdrms2034
curl {쿠버네티스 마스터 IP}:6443/apis -k
그런데 403 에러가 발생
System:anoymous에 대한 인증 실패가 뜸.
"내가 서비스 어카운트 XX이다" 라는 권한 정보를 같이 보내줘야하는데, 해당 인증 정보는 Secret을 가지고 있습니다.
kubectl get secrets
서비스 어카운트 생성할때, 시크릿도 같이 생성됨.
해당 서비스 어카운트와 연결된 시크릿을 확인하는 방법은 다음과 같다.
kubectl describe sa {서비스 어카운트 명}
# Moutable secrets: ~~
# Tokens : ~
kubectl describe secret {시크릿명}
# ca.crt, namespace, token
ca.crt : 쿠버네티스 클러스터의 공개 인증서 저장
namespace : 해당 서비스 어카운트가 존재하는 네임스페이스 저장
token: kube - api server에 인증할 jwt 인증에 사용. 일반적으로 base 64 인코딩으로 저장이 되어있음.
따라서 token을 가지고 api server에 요청을 수행하려면 다음과 같이 수행합니다.
export decoded_token=$(kubectl get secret {secret-name} -o jsonpath='{.data.token}' | base64 -d)
curl {마스터 IP}:6443/apis --header "Authorization: Bearer. $decoded_token" -k
이전의 nginx 인그레스 컨트롤러가 인그레스의 생성을 동적으로 감지해서 nginx 라우팅 규칙을 변경해줌
위 동작은 어떻게 수행을 하고 있었던 것일까
kubectl get svc
를 해보면 default로 생성이 되어있는 kubernetes
서비스가 확인되어있는 것을 볼 수 있음.
kubectl describe pods {포드명}
### Mounts: 확인
spec:
serviceAccountName: {sa 명}
쿠버네티스는 서비스 어카운트 외에도 유저와 그룹의 개념이 존재.
따라서 롤 바인딩을 정의할 때 ServiceAccount 대신 User나 Group을 사용할 수 도 있음.
...
subjects:
- kind: User
name: ehdrms2034
...
## or
subjects:
- kind: Group
name: 그룹명
앞서 나왔던
kubectl get services --as system:serviceaccount:default:ehdrms2034
에서
system:serviceaccount
가 그룹명이라고 볼수 있고
system:serviceaccount:default:ehdrms2034
가 유저명이라고 볼 수 있음.