Kubernetes 사용하는 경우, 공식 docker hub 를 사용하지 않고, 별도로 구축한 private registry 를 사용하는 방법을 설명합니다. 기본적으로 kubernetes 는 공식 registry 를 바라보도록 설정되어 있습니다.
Container 환경에 대해서 학습하기 시작하면서 가장 처음 접하게 되는 것이 아마도 docker 일 것입니다. docker 는 container 환경을 이용하여 특정 프로그램 등을 구동하는 container 구동 환경 그 자체라고 할 수 있습니다. docker 를 사용하게 되면 처음으로 어떤 이미지를 다운로드 받고(pull), 이를 구동(run) 하는 실습이 흔히 이루어집니다.
조금 더 발전시켜서 진행한다면 특정 이미지에서 일부분을 수정하여 나만의 이미지를 만들게 됩니다. 다운로드 받은 이미지의 특정 부분을 수정하는 방법이 있을 수 있고, Dockerfile 이라는 이미지 빌드용 파일을 수정하여 커스텀 이미지를 생성할 수 있을 것입니다.
다운받은 이미지에서 일부를 수정하는 방식은 대게의 경우 netty와 같은 경량 WAS를 다운로드 받고 index.html
파일을 수정하여 변경된 사항이 적용된 이미지를 또 하나의 별도 이미지로 만듭니다.
그렇지 않고 Dockerfile 을 수정하는 경우, 이미 작성해놓은 index.html
파일을 복사하게 COPY
구문을 사용하여 복사하게 하고 이를 빌드, 구동하게 할 것입니다.
다 만들고 난 이미지는 어떻게 할까요?
다른 사람에게 공유하거나, 혹은 다음에 설명할 kubernetes 와 같이 다중화 솔루션을 구동하는 데 사용하기 위해서 이미지를 hub
에 업로드하게 됩니다. 이렇게 이미지가 접근 가능한 어떠한 곳에 위치해야 필요한 시점(scale-out)에 다운로드 받아서 확장하여 사용할 수 있을 것입니다. 그렇다면 이미지를 업로드하게 되는 그 어떤 곳은 무엇일까요?
기본적으로는 아미지를 생성하여 업로드하는 곳으로 docker 공식 registry 를 사용하게 됩니다. 우리가 hub.docker.com
을 통해서 검색하고 다운로드 받는 모든 이미지들은 이 공식 registry 에 업로드되어 있습니다.
그런데, Public network를 사용할 수 없는 상황, 즉 private cloud 를 구축해서 사용하는 경우이거나, 혹은 gitlab 같은 CI/CD 를 별도로 구축해서 사용하는 경우에는 private repository 를 사용해야 하는 경우가 생기고, 이럴 때에는 구동 환경, 즉 kubernetes 가 바라보는 registry 경로를 다르게 설정해야 합니다.
개요
여기서는 단순히 docker registry 를 변경하는 방법만을 다루지는 않습니다. 그 과정에 있었던 개발 과정 중, 도움이 될 만한 것들 까지도 다 정리를 해 두겠습니다. kubernetes 를 처음 사용한다면, 아래 설명에서 이야기하는 것들을 모두 docker-compose 라고 생각해도 무방할 것입니다. 다만 문법이 조금씩 상이하기 때문에 조금은 어색하게 느껴질 수도 있으니, 문법에 맞게 생각해야 합니다.
또한 docker-compose 조차도 생소하다면 이 부분에 대해서는 다른 글을 조금 검색해 보시고 오시기를 권해 드립니다. 몰라도 그만인 부분이라고 할 수 있지만, docker, container, kubernetes 를 학습하는 데 도움이 되는 부분이고 자주 사용하게 되므로 짚고 넘어가는 것이 좋습니다.
kubernetes 에 대한 글을 적으면서 반복적으로 설명하겠지만, kubernetes 에서 pod, node 등의 동일한 역할을 하는 집합들은 동일한 namespace 로 관리해야 편합니다. 정식 가이드는 어떻게 하고 있는지는 확인해 보아야 하겠습니다만, namespace의 역할 자체가 동일 기능들을 묶어놓는, 일종의 tag 개념으로 사용(인식)되기 때문입니다. 어떠한 정책을 적용하는 등의 작업에 있어서 namespace 단위로 적용할 수 있으며 따라서 이를 잘 적용해 두어야 향후 중복작업을 방지할 수 있습니다.
deployment, service 등을 생성할 때에 namespace 를 별도로 설정하지 않으면 kubernetes 에 있는 default namespace를 사용하게 됩니다. 만약 하나의 집단만을 관리할 것이라면 이렇게 설정한 채로 default namespace를 그대로 사용해도 동작에는 지장이 없겠지만, 유지보수 측면이나 다른 사람이 관리하는 것을 감안한다면 별도의 namespace를 생성해서 관리해 주는 것이 좋습니다.
현재의 상황에 대해서 국한하는 것이 아니라, 단 하나의 namespace만 필요한 정도로 작은 시스템이라고 하더라도 namespace를 생성하기를 권장합낟.
namespace 생성
이제 namespace를 생성하겠습니다. namespace는 일종의 tag같은 개념이기 때문에 그 안에 포함될 deployment, pod, container 등의 이름들을 포괄할 수 있는 것으로 정하겠습니다.
다음 명령어로 namespace를 생성합니다. 이 이름이 모든 서비스들에 적용될 것이므로, 적절한 형태로 지어주는 것이 좋습니다.
kubectl create namespace mynamespace
이렇게 하면 별다른 미시지 없이 namespace생성이 완료됩니다. 여기서는 mynamespace
라는 이름으로 namespace를 생성해 주었고, 이 namespace안에 포함하게 될 pod, deployment 등의 모든 요소에는 metadata
부분에 namespace : mynamespace
를 추가해 줍니다. (yml파일 내에 정의합니다)
secret 생성
일반적인 경우에는 docker hub 에 보여지는 public registry 에서 이미지를 다운로드 받고, 구동하게 된다고 앞서 설명했습니다. 이 docker hub
의 경우에는 public 이미지는 로그인 등의 인증 절차가 없고, private image 에 대해서는 로그인 정보를 물어보게 됩니다. 당연한 절차입니다.
만약 별도로 구축한 docker registry 에 로그인 정보가 필요하다면 이렇게 로그인을 할 수 있는 계정 정보를 넣어주어야 합니다. 만약 docker 엔진 자체에서 사용하다면 docker 에, 그렇지 않고 kubenetes에서 사용하는 경우라면 kubernetes에 이 계정 정보를 넣어 주어야 구동 시점에 정상적으로 image를 가져오게 될 것입니다. kubernetes 에서 이렇게 로그인 정보 등을 가지고 있기 위해서는 secret이라는 구성요소를 활용합니다.
특정 docker registry 에 접근하게 하려면 우선 kubernetes 의 요소 중에서 secret을 생성해 주어야 합니다. 이것이 secret 이라고 불려지는 이유는, id / password 등을 이용해서 특정 docker registry 에 로그인하는 정보를 담고 있기 때문입니다.
여기서는 단순히 id / password를 이용해서 로그인 하는 것으로 예시를 들었습니다. 실제 환경에서는 password 대신에 github / gitlab 등에서 발생하는 token 등으로 대체하여야 보안 문제를 해소할 수 있습니다. github에서 secret token 을 생성하고 이용하는 방법에 대해서는 추후 다루도록 하겠습니다.
kubectl create secret registry-credentials docker-registry-login --namespace=mynamespace --docker-server=<server-address>:<port> --docker-username={아이디} --docker-password={패스워드} --docker-email={이메일}
위의 명령어를 분석해보면 다음과 같습니다.
- kubectl create secret : kubernetes 의 구성요소 중 secret 을 생성
- registry-credentials docker-registry-login : registry-credentials 라는 것으로 생성할 것인데, 이것은 docker registry 에 로그인하는 정보를 가지고 있음
이제 이 생성할 secret 에 부가정보를 추가해 주는 부분입니다.
- –namespace=mynamespace : namespace 정의
- –docker-server=<server-address>:<port> : 접속할 서버 정보(사설로 구축한 docker registry 서버 정보)
- –docker-username={아이디} : 로그인할 아이디
- –docker-password={패스워드} : 로그인할 패스워드
- –docker-email={이메일} : 이메일 정보(필수값 아닐 수 있지만, 넣어줍니다.)
yaml 파일에 적용
다음으로는 kubernetes의 deployment 를 생성하는 yaml(yml) 파일을 수정합니다. 이미 설정해 두었던 yaml 파일이 있다면 다른 부분은 별도로 수정할 필요가 없고, 아래와 같이 containers 를 설정해 주는 부분만 추가해 주면 됩니다.
...(중략)
template:
metadata:
labels:
app: appapp
spec:
imagePullSecrets:
- name: registry-credentials
containers:
- name: namenamename
image: image:tag
imagePullPolicy: Always
env:
...(후략)
위에서 중요하게 보아야 하는 부분은, imagePullSecrets 부분입니다. 만약 public registry 를 그대로 사용한다면 저 부분 자체가 없고, spec 다음에 containers 부분이 바로 나타날 것입니다만, 여기서는 private registry 를 사용하려고 하는 것이므로 위와 같이 imagePullSecrets 부분을 추가해 주고, name 부분에 앞서 생성한 secret 의 이름을 지정해 줍니다.
이렇게 하면, 해당 deployment 가 생성되는 시점에 image를 가져오는 것을 해당 registry 에서 가져오게 됩니다.
제언
이러한 방식으로 구성하면 공식 public docker registry 를 바라보지 않고, private registry 를 바라보게 변경할 수 있습니다. kubernetes 를 사용할 환경을 구성하고 나면(예를 들어, AWS 에서 EKS 인스턴스 생성), 우선 들어가서 namespace 를 생성하고, 사용할 대상이 되는 registry 에 대한 secret을 미리 생성해 두는 것도 좋아 보입니다.
물론 여기에는 istio 같은 네트워크 관련 설정들도 모두 포함하여 기반 작업을 모두 진행한 뒤에, application이 구동될 수 있도록 구성해 주는 것이 좋겠습니다.
참고
아래는 kubernetes 사용 시 참고할 만 한 내용들입니다. 명령어가 익숙하지 않은 경우를 대비에 아래와 같이 추가로 정리합니다.
namespace 삭제 방법
kubectl delete namespace mynamespace
secret 삭제 방법
kubectl delete secret docker-registry-login -n mynamespace
Leave a comment