Docker 배포 : 쉽게 시작하기

Docker를 수동 배포 및 Watchtower 사용해 보자!

written by tiaz0128

🐋 Docker를 서버에 배포하는 방법을 알아보겠습니다. 수동 배포 부터 시작해 봅시다!

도커 이미지 생성


API 코드

배포 할 API 서버 코드를 먼저 작성하겠습니다. FastAPI로 작성된 문자열을 반환하는 간단한 코드 입니다. run.py 파일에 코드를 작성하겠습니다.

project/
└── run.py (추가 파일)
run.py
from fastapi import FastAPI
from fastapi.responses import PlainTextResponse

app = FastAPI()

@app.get("/")
def health(response_class=PlainTextResponse):
    return "Hello Docker Container"

Dockerfile 작성

Docker 이미지를 생성할 때 사용하는 Dockerfile을 작성하겠습니다.

project/
├── Dockerfile (추가 파일)
└── run.py
Dockerfile
FROM python:3.12-alpine

RUN pip install "fastapi[standard]"

WORKDIR /app

COPY . .

EXPOSE 8000

CMD ["uvicorn", "run:app", "--host", "0.0.0.0", "--port", "8000"]

Docker 이미지 생성

작성한 도커 파일(Dockerfile)을 사용하여 이미지를 빌드합니다. 이때 해당 이미지에 대한 tag를 -t 옵션으로 부여하고, 이후 도커 허브에 이미지를 push 할 때 사용합니다.

$ docker build -t tiaz0128/fast-api .

docker.com 가입


docker.com에 가입하고 로그인 합니다. 가입 할때 생성하는 Username은 도커 이미지를 저장하는 도커 허브(Docker Hub)에 사용 됩니다.

Docker Username

> GitHub와 유사한 형태의 Docker Hub

Docker Hub : 이미지 push


Docker Hub는 빌드한 도커 이미지를 저장하는 레포지토리 입니다. repositories탭에서 Username을 네임스페이스로 하는 레포지토리를 생성합니다.

여기서는 Public 레포지토리를 fast-api라는 이름으로 만들도록 하겠습니다. 참고로 비결제 계정당 하나의 Private 레포지토리를 사용 할 수 있습니다.

Docker Username

> Docker Hub : 이미지를 저장하는 원격 저장소

Docker login

빌드한 이미지를 도커 허브에 push 하겠습니다. docker.io에 등록한 이메일이나 Username으로 로그인 가능합니다.

$ docker login docerk.io

Username: tiaz0128.dev@gmail.com
Password: ****

Login Succeeded

빌드한 이미지 push

이미지를 Docker Hub에 push 하기 위해서는 docker.io에서 등록된 Username과 repository 이름을 기반으로 push 할 수 있습니다.

Docker Hub를 사용할 때는 일반적으로 Username/repository 형식을 따라야 합니다. 이는 Docker Hub가 사용자별로 네임스페이스를 관리하기 때문입니다.

$ docker push <Username>/<repository>[:tag]

$ docker push tiaz0128/fast-api

Docker tag

tag 명령어는 Docker 이미지에 태그를 추가하거나 변경하는 데 사용됩니다. 기본 사용법은 다음과 같습니다.

$ docker tag SOURCE_IMAGE[:TAG] TARGET_IMAGE[:TAG]

$ docker tag fast-api:1.0 tiaz0128/fast-api

Docker 수동 배포


어떤 환경이던 서버 환경이든 상관없습니다. 여기서는 Amazon EC2(Amazon Linux 2023 AMI)에서 진행했습니다.

Docker 설치

가장 먼저 서버에 Docker를 설치합니다.

$ sudo yum update -y

$ sudo yum -y install docker

데몬 & 권한 추가

설치 후에는 데몬을 실행 합니다. 필요시 사용자 그룹 권한을 수정 해줍니다. 사용자(ec2-user)를 docker 그룹에 추가합니다. 이를 통해 sudo 없이 docker 명령어를 실행할 수 있게 됩니다.

$ sudo systemctl restart docker

$ sudo usermod -a -G docker ec2-user

Docker pull

도커 허브에 로그인 하고 빌드 한 이미지를 pull 받습니다.

$ docker login docerk.io

Username: tiaz0128.dev@gmail.com
Password: ****

Login Succeeded
$ docker pull tiaz0128/fast-api

Docker run & stop

pull 받은 이미지로 컨테이너를 실행 합니다. 실행 중인 컨테이너가 있다면 정지 후, 새로운 컨네이터를 실행하는 방식으로 수동으로 배포가 가능합니다.

$ docker run -it -d --name fast-api --rm tiaz0128/fast-api

수동 배포의 문제점


여기까지 수동으로 도커 컨테이너를 배포해보았습니다. 그렇다면 여기서 이런 생각이 듭니다.

코드가 수정되고 이미지가 새로 만들어지면 어떻게 하지?

앞서 했던 것처럼 매번 수동으로 이미지를 pull 받고 컨테이터를 재시작해야 할 것입니다.

이런 문제를 해결하기 위한 여러가지 방법이 있지만, 여기서는 간단한 Watchtower를 사용해 보겠습니다.

Watchtower


Watchtower

Watchtower는 정기적으로 이미지 업데이트를 확인하고, 새 버전이 있으면 자동으로 컨테이너를 업데이트합니다. Watchtower 컨테이너를 실행 시켜 놓기만 하면 이미지를 가져와서 기존 컨테이너를 정상적으로 종료한 후, 처음 배포될 때 사용한 동일 옵션으로 재시작합니다.

Watchtower 기본 사용법

아래의 명령을 통해 EC2 내부에 동작 중인 fast-api라는 컨테이너를 모니터링 합니다. 여러개 컨테이너를 한번에 모니터링 또한 가능 합니다.

$ docker ps

CONTAINER ID   IMAGE                        NAMES
dad54e31fb2b   tiaz0128/fast-api:latest     fast-api
$ docker run -d \
  --name watchtower \
  -v /var/run/docker.sock:/var/run/docker.sock \
  containrrr/watchtower \
  fast-api

Watchtower 주요 옵션

남은 과제와 GitOps


Watchtower를 사용했지만 여전히 git 배포와 Docker 이미지 배포를 각각 수행해야 하는 불편함이 남아있습니다. 이러한 문제를 해결 하기 위해 GitOps 라는 방법론이 등장 합니다.

이 방법론의 핵심 아이디어는 단일 진실 원천(single source of truth)으로 git 레포지토리를 모든 것의 기준점으로 삼는 것입니다.

> 따배 : [따배GitOps] 0. 강의 소개

마무리


빌드한 이미지를 도커 허브에 push 하고 빌드된 이미지를 Docker EC2에 처음에는 수동으로 배포해보고, Watchtower를 이용하여 자동으로 배포까지 해보았습니다.

다만 이 방식도 여전히 배포 시, 문제점이 남아있었습니다. 다음은 이러한 문제까지 해결할 수 있는 좀 더 나은 Docker 배포 방법을 공부해 봅시다! 😊

참고 문헌


Docker

tiaz0128

Eat Sleep Coding.

Never Never GiveUp.

Security  |  BackEnd  |  Multi Cloud