본문 바로가기

ChatGPT/Docker

[Docker] Dockerfile 작성 방법

반응형

1. Dockerfile이란?

Dockerfile은 Docker 이미지를 생성하기 위한 스크립트입니다. Dockerfile을 사용하면 누구나 쉽게 동일한 애플리케이션 환경을 설정할 수 있습니다.

Dockerfile은 애플리케이션을 컨테이너로 패키징할 때 사용됩니다. Dockerfile을 작성하면 Docker 빌드 프로세스를 자동화할 수 있으며, 일관된 빌드 및 배포를 가능하게 합니다.

Dockerfile의 장점은 다음과 같습니다.

  • 다른 환경에서도 동일한 애플리케이션 실행 가능
  • 이미지 빌드를 자동화하여 일관성 유지
  • 이미지 빌드 시간을 단축하여 배포 프로세스 개선
  • 다른 사용자와 Docker 이미지를 공유할 수 있습니다.

Dockerfile은 FROM, RUN, COPY, ADD, CMD, ENTRYPOINT 등과 같은 다양한 명령어를 사용하여 이미지를 생성합니다. 이 명령어를 적절하게 사용하여 Dockerfile을 작성하면 효과적인 이미지 빌드를 수행할 수 있습니다.

2. Dockerfile 작성을 위한 기본 지식

Dockerfile을 작성하기 위해서는 Docker, 이미지, 컨테이너 등에 대한 기본적인 이해가 필요합니다.

  • Docker: 컨테이너 기반의 오픈소스 가상화 플랫폼입니다. Docker를 사용하면 애플리케이션을 컨테이너로 패키징하고 배포할 수 있습니다.
  • 이미지: 컨테이너를 생성하기 위한 파일 시스템과 실행할 애플리케이션을 포함하는 읽기 전용 템플릿입니다. Docker 이미지는 Dockerfile을 사용하여 빌드됩니다.
  • 컨테이너: Docker 이미지를 실행한 가상 환경입니다. 컨테이너는 격리된 환경에서 애플리케이션을 실행할 수 있도록 지원합니다.

Dockerfile은 이미지를 빌드하는 데 사용되며, Dockerfile 내에는 이미지를 빌드하는 데 필요한 모든 정보가 포함됩니다. Dockerfile을 작성할 때는 베이스 이미지를 선택하고, 필요한 패키지를 설치하고, 애플리케이션을 설정하며, 노출할 포트를 지정하는 등의 작업이 필요합니다. 이후 Docker 빌드 명령어를 사용하여 Dockerfile을 실행하면 이미지를 생성할 수 있습니다.

3. Dockerfile 작성 단계

Dockerfile 작성의 기본적인 단계는 다음과 같습니다.

1. 베이스 이미지 선택

  • Dockerfile을 작성할 때는 먼저 베이스 이미지를 선택해야 합니다. 베이스 이미지는 기반이 되는 이미지로서, Dockerfile에서 FROM 명령어를 사용하여 지정합니다.

2. 필요한 패키지 설치

  • Docker 이미지를 빌드할 때는 애플리케이션을 실행하는 데 필요한 패키지를 설치해야 합니다. 이를 위해 Dockerfile에서 RUN 명령어를 사용하여 필요한 패키지를 설치합니다.

3. 애플리케이션 설정

  • 애플리케이션을 실행하기 위해서는 환경 설정이 필요합니다. 이를 위해 Dockerfile에서 ENV 명령어를 사용하여 환경 변수를 설정하거나, COPY 명령어를 사용하여 설정 파일을 복사합니다.

4. 노출할 포트 지정

  • Docker 컨테이너는 호스트 시스템과 네트워크를 공유합니다. 이를 위해 Dockerfile에서 EXPOSE 명령어를 사용하여 노출할 포트를 지정합니다.

5. 애플리케이션 실행

  • Docker 컨테이너가 실행될 때, 애플리케이션을 실행해야 합니다. 이를 위해 Dockerfile에서 CMD 명령어나 ENTRYPOINT 명령어를 사용하여 애플리케이션을 실행합니다.

Dockerfile을 작성할 때는 이러한 단계들을 고려하여 작성해야 합니다.

4. Dockerfile 작성 가이드라인

Dockerfile을 작성할 때 주의해야 할 사항과 베스트 프랙티스는 다음과 같습니다.

1. 베이스 이미지 선택

  • 베이스 이미지는 가능한 한 작고, 애플리케이션 실행에 필요한 것만 포함하는 것이 좋습니다.
  • 최신 버전의 베이스 이미지를 사용하는 것이 좋습니다.

2. RUN 명령어 사용 시 라인 수 최소화

  • 가능한 한 RUN 명령어를 하나의 명령어로 작성하는 것이 좋습니다.
  • 라인 수를 최소화하여 빌드 시간을 단축할 수 있습니다.

3. COPY 명령어 사용 시 필요한 파일만 복사

  • 가능한 한 필요한 파일만 복사하여 이미지 크기를 줄이는 것이 좋습니다.
  • COPY 명령어는 디렉토리 복사와 파일 복사를 구분할 수 있으므로 적절히 사용하는 것이 좋습니다.

4. Docker 이미지 최적화

  • 불필요한 파일을 삭제하고, 캐시를 적극적으로 활용하여 이미지 크기를 최소화하는 것이 좋습니다.
  • 다중 빌드 스테이지를 사용하여 이미지 크기를 줄이는 것도 좋은 방법입니다.

5. 환경 변수 사용 시 변수명 규칙 지키기

  • 환경 변수의 이름은 대문자와 언더스코어(_)를 사용하여 지정하는 것이 좋습니다.

6. 명령어 실행 시 적절한 권한 부여

  • 가능한 한 최소한의 권한만을 사용하여 명령어를 실행하는 것이 좋습니다.

7. 이미지 태그 지정

  • 가능한 한 이미지 태그를 명시하여 이미지 버전을 관리하는 것이 좋습니다.

Dockerfile 작성 시 이러한 가이드라인을 따르면 효율적인 이미지 빌드를 수행할 수 있습니다.

5. Dockerfile 예제

다음은 Node.js 애플리케이션을 Docker 이미지로 빌드하는 Dockerfile 예제입니다.

FROM node:14-alpine

# 앱 디렉토리 생성
WORKDIR /usr/src/app

# 앱 의존성 설치
COPY package*.json ./
RUN npm install

# 앱 소스 추가
COPY . .

# 앱 실행
CMD [ "npm", "start" ]

위 Dockerfile은 다음과 같은 단계로 이루어져 있습니다.

  1. FROM node:14-alpine: Node.js 14 버전을 기반으로 하는 Alpine 리눅스 이미지를 베이스 이미지로 사용합니다.
  2. WORKDIR /usr/src/app: 애플리케이션 코드를 복사할 디렉토리를 생성합니다.
  3. COPY package*.json ./: 애플리케이션의 의존성을 설치하기 위해 package.json 및 package-lock.json 파일을 복사합니다.
  4. RUN npm install: 애플리케이션의 의존성을 설치합니다.
  5. COPY . .: 현재 디렉토리의 모든 파일을 애플리케이션 디렉토리로 복사합니다.
  6. CMD [ "npm", "start" ]: npm start 명령어를 사용하여 애플리케이션을 실행합니다.

이러한 Dockerfile을 작성하면 Node.js 애플리케이션을 Docker 이미지로 빌드할 수 있습니다.

6. Dockerfile 명령어

Dockerfile에서 사용하는 명령어는 다음과 같습니다.

1. FROM

  • 베이스 이미지를 지정합니다. Dockerfile은 이 베이스 이미지를 기반으로 새로운 이미지를 빌드합니다.

2. RUN

  • 쉘 명령어를 실행합니다. Dockerfile에서 RUN 명령어는 이미지를 빌드할 때 실행됩니다.

3. CMD

  • 컨테이너가 시작될 때 실행될 기본 명령어를 지정합니다. Dockerfile에서 CMD 명령어는 마지막으로 실행됩니다.

4. ENTRYPOINT

  • 컨테이너가 시작될 때 실행될 명령어를 지정합니다. Dockerfile에서 ENTRYPOINT 명령어는 CMD 명령어와 함께 사용되어 기본 명령어를 오버라이드할 수 있습니다.

5. COPY

  • 파일이나 디렉토리를 컨테이너에 복사합니다.

6. ADD

  • COPY 명령어와 비슷하지만, URL이나 압축 파일을 다운로드하여 압축을 해제한 후 컨테이너에 복사할 수 있습니다.

7. ENV

  • 환경 변수를 설정합니다.

8. ARG

  • Docker 빌드 시에 전달되는 인수(argument)를 사용할 수 있습니다.

9. EXPOSE

  • 컨테이너가 노출할 포트를 지정합니다.

10. VOLUME

  • 호스트 시스템의 디렉토리나 파일을 컨테이너에 마운트합니다.

이러한 명령어를 조합하여 Dockerfile을 작성하면 Docker 이미지를 빌드할 수 있습니다.

7. Dockerfile에서 변수 사용하기

Dockerfile에서 환경 변수를 사용하려면 ENV 명령어를 사용합니다.

다음은 환경 변수를 사용하는 Dockerfile 예제입니다.

FROM node:14-alpine

# 앱 디렉토리 생성
WORKDIR /usr/src/app

# 환경 변수 설정
ENV PORT 3000

# 앱 실행
CMD [ "npm", "start" ]

위 Dockerfile은 PORT 환경 변수를 설정하고, npm start 명령어를 사용하여 애플리케이션을 실행합니다.

Dockerfile에서 설정한 환경 변수는 RUN, CMD, ENTRYPOINT 등의 명령어에서 사용할 수 있습니다.

또한 Docker 빌드 시에 인수(argument)를 전달하여 환경 변수를 설정할 수도 있습니다. 이를 위해서는 ARG 명령어를 사용하여 인수를 정의한 후, Dockerfile 내에서 ENV 명령어를 사용하여 인수를 환경 변수로 설정해야 합니다.

import express, { Request, Response } from 'express';

const app = express();
const port = process.env.PORT ? process.env.PORT : 4000;

app.get('/', (req: Request, res: Response) => {
  res.send('Hello, world!');
});

app.listen(port, () => {
  console.log(`Express is listening at http://localhost:${port}`);
});

 

반응형

8. Dockerfile에서 빌드 캐시 활용하기

Dockerfile 빌드 시 캐시를 활용하여 빌드 시간을 단축할 수 있습니다. 이를 위해서는 Dockerfile을 작성할 때 캐시를 적극적으로 활용해야 합니다.

Dockerfile에서는 빌드 단계별로 캐시를 저장합니다. 따라서 Dockerfile의 앞부분에 있는 명령어를 변경하면 뒤에 있는 명령어는 모두 다시 실행해야 합니다.

다음은 캐시를 활용하는 Dockerfile 예제입니다.

FROM node:14-alpine

# 앱 디렉토리 생성
WORKDIR /usr/src/app

# 앱 의존성 설치
COPY package*.json ./
RUN npm install

# 앱 소스 추가
COPY . .

# 앱 실행
CMD [ "npm", "start" ]

위 Dockerfile에서는 npm install 명령어를 실행하는 부분에서 캐시를 활용합니다. 이전에 이미 실행된 적이 있고, 이후에 변경된 부분이 없다면 이전에 생성된 캐시를 사용하여 빌드 시간을 단축할 수 있습니다.

또한, COPY 명령어를 사용할 때 변경된 파일만 복사하도록 주의하여 Docker 이미지 크기를 줄일 수 있습니다.

코드 변경 후 빌드한 모습 ( 5번째 단계부터 진행)

9. Dockerfile에서 ARG 사용하기

Dockerfile에서 ARG 명령어를 사용하면 빌드 타임에 인수(argument)를 전달할 수 있습니다.

다음은 ARG를 사용하는 Dockerfile 예제입니다.

FROM node:14-alpine

# 앱 디렉토리 생성
WORKDIR /usr/src/app

# 환경 변수 설정
ARG PORT
ENV PORT $PORT

# 앱 의존성 설치
COPY package*.json ./
RUN npm install

# 앱 소스 추가
COPY . .

# 앱 실행
CMD [ "npm", "start" ]

위 Dockerfile에서는 ARG를 사용하여 PORT 변수를 정의하고, ENV 명령어를 사용하여 환경 변수를 설정합니다. Docker 빌드 시에 --build-arg 옵션으로 인수를 전달하면 Dockerfile 내에서 ARG로 정의한 변수를 사용할 수 있습니다.

예를 들어 다음과 같이 Dockerfile을 빌드하면서 PORT 인수를 전달할 수 있습니다.

$ docker build --build-arg PORT=3000 -t my-image:latest .

이렇게 빌드 타임에 인수를 전달하여 Dockerfile을 작성하면 다양한 환경에서 동일한 Dockerfile을 사용할 수 있습니다.

10. Dockerfile에서 인증 정보 처리하기

Dockerfile에서 인증 정보를 처리하는 방법은 여러 가지가 있습니다.

1. Docker secrets

  • Docker secrets는 Docker에서 제공하는 비밀 정보 관리 시스템입니다. Docker secrets를 사용하면 인증 정보를 안전하게 저장하고, 컨테이너가 실행될 때 비밀 정보를 전달할 수 있습니다.

2. 환경 변수

  • 인증 정보를 환경 변수로 설정하여 Dockerfile에서 사용할 수 있습니다. 이 경우, 인증 정보가 Dockerfile 내에 노출될 수 있으므로 보안에 취약할 수 있습니다.

3. ARG

  • Dockerfile에서 ARG 명령어를 사용하여 빌드 타임에 인수를 전달할 수 있습니다. 이를 활용하여 Docker 빌드 시에 인증 정보를 전달할 수 있습니다.

4. 볼륨

  • 인증 정보를 볼륨으로 마운트하여 컨테이너에서 사용할 수 있습니다. 이 경우, 볼륨 내부의 파일에 인증 정보를 저장하고, 컨테이너가 실행될 때 이 파일을 참조하여 인증을 수행합니다.

이러한 방법 중에서 Docker secrets를 사용하여 인증 정보를 안전하게 관리하는 것이 가장 좋은 방법입니다.

11. 결론

Dockerfile은 Docker 이미지를 빌드하는 데 필요한 설정 파일입니다. Dockerfile을 작성하면 Docker 빌드 시스템을 통해 Docker 이미지를 생성할 수 있습니다.

Dockerfile을 작성하는 방법은 다양하지만, 기본적인 작성 방법을 숙지하고, 캐시를 적극적으로 활용하는 등의 최적화 기법을 적용하여 빠르고 안정적인 Docker 이미지를 빌드할 수 있습니다.

또한 Dockerfile에서 환경 변수, ARG 등을 활용하여 빌드 타임에 인수를 전달하거나 인증 정보를 안전하게 관리할 수 있습니다.

Dockerfile은 Docker 이미지를 빌드하는 데 필수적인 요소이므로, Docker를 활용한 애플리케이션 개발 및 배포에 필수적인 기술입니다.

 

이글은 ChatGPT 도움을 받아 작성되었습니다.

반응형