[Docker] Docker에 Flask App 배포하기
📌Index
Flask란 마이크로 서비스 웹 프레임워크로 간단한 웹 사이트, 혹은 간단한 API 서버를 만드는 데에
특화되어있다. (python 3.7 이상 필요)
✔️ 환경 구성
Flask
환경 구성을 위해 아래의 3가지를 설치하자
- python3
- python3-pip
- python3-venv
pip
는 파이썬 패키지를 설치하고 관리하는 도구이다.
pip
는python2
의 패키지 관리자pip3
는python3
의 패키지 관리자
$ sudo apt install python3-pip python3-venv
가상 환경
가상환경(virtual environment)는 Python
에만 존재하는 개념으로, 가상환경을 사용하면 가상 환경에서만 패키지를 설치하고 관리할 수 있다. Flask
는 우리가 만드려는 App에만 필요하기 때문에 Global 전체에 설치하는 것이 불필요하고, 해당되는 App에만 사용할 수 있는 가상 환경을 만들고, 가상 환경에서만 사용할 수 있는 패키지를 따로 관리한다(Isolation)
가상환경 만들기
가상환경을 위한 별도의 디렉토리를 만들어주는 것은 중요하다
$ mkdir -p /python/hello-flask
$ cd /python/hello-flask
가상환경/프로젝트를 생성
$ python3 -m venv venv
venv
는 파이썬이 관리하므로, 직접적으로 건들이지 않는 것이 좋음-m venv
에서venv
는 모듈명(패키지명)이고, 뒤의venv
는 가상환경의 이름- 가상환경의 이름을 꼭
venv
로 설정할 필요는 없으나 일반적으로venv
로 지정 - 가상 환경에 패키지를 설치하면
/venv/lib/site-packages/
에 설치됨(isolation)
가상환경 활성화
$ . venv/bin/activate
- 가상환경 비활성화(가상환경에서 벗어나기) :
deactivate
- 관리하는 패키지 목록 확인하기 :
pip3 list
(venv)
를 통해 가상 환경에 들어온 것을 확인- 작업 디렉토리에서 가상환경을 활성화를 시켜줘야함
- 참고) 가상환경에서 디렉토리를 빠져나와도 가상환경은 유지됨
- 터미널을 닫았다가 다시 열면 가상환경에서 나와지기 때문에 꼭 promt를 통해 본인이 가상환경인지 확인해야함
Flask 설치
$ pip3 install Flask
✔️ Application 만들기
Minimal Application
a minimal application을 참조하여, 아래와 같이 hello.py
파일을 작성하자.
hello.py
from flask import Flask
app = Flask(__name__)
@app.route("/")
def hello_world():
return "<p>Hello, World!</p>"
파일 내용 설명
from flask import Flask
: Flask 패키지를 가져옴@app.route("/")
:app.route
decorator
로, root(/) 경로로 오면 아래의 함수를 실행하도록 설정- Linux의
DocumentRoot
라고 생각하면 이해하기 쉬울 것 같다
- Linux의
def hello_world()
: 함수정의, return 값만 있음
Flask 실행
$ export FLASK_APP=hello
$ flask run
* Serving Flask app 'hello' (lazy loading)
* Environment: production
WARNING: This is a development server. Do not use it in a production deployment.
Use a production WSGI server instead.
* Debug mode: off
* Running on http://127.0.0.1:5000 (Press CTRL+C to quit)
127.0.0.1 - - [11/May/2022 05:16:16] "GET / HTTP/1.1" 200 -
flask run
==python3 -m flask run
접속 확인
단, 개발중에 외부에 노출되면 위험하기 때문에,
기본적으로 외부에서는 접속이 불가능하고 내부에서만 접속 가능하도록 구성되어있다.
$ curl localhost:5000
<p>Hello, World!</p>%
다음 명령을 통해 외부에서의 접속을 허용할 수 있다.
$ flask run --host='0.0.0.0' [--port='8080']
Rendering Templates
rendering-templates을 참조하여 hello.py
를 작성해보자
hello.py
from flask import Flask
from flask import render_template
app = Flask(__name__)
@app.route('/')
def hello_world():
return "<p>Hello, World!</p>"
@app.route('/hello/')
@app.route('/hello/<name>')
def hello(name=None):
return render_template('hello.html', name=name)
파일 내용 설명
- root(/)로 접근하면
Hello, World
가 출력 /hello/
뒤에 변수(name)을 적으면 함수(hello)의 매개변수로 전달됨
/template/hello.html
을 다음과 같이 작성해보자
<!doctype html>
<title>Hello from Flask</title>
{% if name %}
<h1>Hello {{ name }}!</h1>
{% else %}
<h1>Hello, World!</h1>
{% endif %}
파일 내용 설명
{% if name %}
조건문:name
이라는 변수가 있으면Hello {{ name }}!
을 출력{% else %}
: 없으면Hello, World!
을 출력
Flask 실행
$ flask run --host='0.0.0.0' --port='8080'
접속 확인
/hello/
뒤의 변수 /encore
가 들어가서 출력된 것을 확인할 수 있다.
✔️ Freeze
현재 설치된 패키지들을 보존하여 저장할 수 있다.
$ pip3 freeze > requirements.txt
$ cat requirements.txt
click==8.1.3
Flask==2.1.2
importlib-metadata==4.11.3
itsdangerous==2.1.2
Jinja2==3.1.2
MarkupSafe==2.1.1
Werkzeug==2.1.2
zipp==3.8.0
매번 애플리케이션을 개발할 때마다 필요한 패키지들을 설치하는 것은 매우 번거럽고 어렵다.
따라서 필요한 패키지들을 requirements.txt
파일로 만들어 놓고, 이 파일을 읽어서 패키지 목록을 설치할 수 있다.
$ pip3 install -r requirements.txt
파이썬 소스코드는 항상 reauirements.txt
파일을 제공한다.
개발 완료 후, 코드를 배포하기 전에 freeze
해서 requirements.txt
파일을 생성해야,
이후 사용하는 사람이 어떤 패키지의 어떤 버전을 사용했는지 알고 대응할 수 있다.
✔️ Flask App으로 Docker 이미지 만들기
앞서 생성한flask app
을 이용하여 진행해보자
Dockerfile
FROM python:3.9-slim-buster
COPY . /app
WORKDIR /app
RUN python3 -m venv venv && . venv/bin/activate
RUN pip3 install -r requirements.txt
ENTRYPOINT ["python3", "-m", "flask", "run"]
CMD ["--host=0.0.0.0"]
⭐참고 사항
RUN
작성 시, ;
로 명령들을 나열하는 것도 가능하나
AND 연산의 &&
또는 OR 연산의 ||
를 사용하는 것도 가능하다.
A ; B
가 A
실행 여부와 관계없이 B
가 실행되는 반면에,
A && B
의 경우 A
가 성공해야만 B
가 실행되고, A || B
는 A
가 실패해야만 B
가 실행된다.
.dockerignore
venv/
Dockerfile
.dockerignore
$ docker build -t myflask .
.dockerfile
이 제대로 작용했는지 확인하기 위해 빌드한 이미지로 컨테이너를 생성해서 복사된 파일 목록을 확인해보자
$ docker run -it myflask bash
root@3af01fa6732b:/# ls app/
__pycache__ hello.py requirements.txt templates
이미지 myflask
로 docker run
해보자
$ docker run -d -p 80:5000 myflask
b54715c7057143a8d90676be1b48e5e252c6f1233340f1b9ec2991cca5c5df9d
$ docker ps
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
b54715c70571 myflask "python3 -m flask ru…" 5 seconds ago Up 4 seconds 0.0.0.0:80->5000/tcp, :::80->5000/tcp condescending_cohen
접속 확인