programing

도커에 최소한의 플라스크 앱 배포 - 서버 연결 문제

css3 2023. 7. 4. 22:03

도커에 최소한의 플라스크 앱 배포 - 서버 연결 문제

도커 외부에서 정상적으로 실행되고 기본 포트에 바인딩되는 플라스크만 의존하는 앱이 있습니다.5000다음은 전체 소스입니다.

from flask import Flask

app = Flask(__name__)
app.debug = True

@app.route('/')
def main():
    return 'hi'

if __name__ == '__main__':
    app.run()

문제는 도커에 이 파일을 배포할 때 서버가 실행 중이지만 컨테이너 외부에서 연결할 수 없다는 것입니다.

아래는 나의Dockerfile이미지는 플라스크가 설치된 unbuntu입니다.타르는 단지 그것을 포함합니다.index.py위에 나열됨;

# Dockerfile
FROM dreen/flask
MAINTAINER dreen
WORKDIR /srv

# Get source
RUN mkdir -p /srv
COPY perfektimprezy.tar.gz /srv/perfektimprezy.tar.gz
RUN tar x -f perfektimprezy.tar.gz
RUN rm perfektimprezy.tar.gz

# Run server
EXPOSE 5000
CMD ["python", "index.py"]

다음은 배포하기 위해 수행하는 단계입니다.

$> sudo docker build -t perfektimprezy .

위의 내용물이 정상적으로 작동하는 것으로 알고 있습니다./srv이제 컨테이너에서 서버를 시작하겠습니다.

$> sudo docker run -i -p 5000:5000 -d perfektimprezy
1c50b67d45b1a4feade72276394811c8399b1b95692e0914ee72b103ff54c769

실제로 실행되고 있습니까?

$> sudo docker ps
CONTAINER ID        IMAGE                   COMMAND             CREATED             STATUS              PORTS                    NAMES
1c50b67d45b1        perfektimprezy:latest   "python index.py"   5 seconds ago       Up 5 seconds        0.0.0.0:5000->5000/tcp   loving_wozniak
$> sudo docker logs 1c50b67d45b1
    * Running on http://127.0.0.1:5000/ (Press CTRL+C to quit)
    * Restarting with stat

네, 플라스크 서버가 작동하는 것 같습니다.여기가 이상해지는 부분입니다.서버에 요청합니다.

$> curl 127.0.0.1:5000 -v
* Rebuilt URL to: 127.0.0.1:5000/
* Hostname was NOT found in DNS cache
*   Trying 127.0.0.1...
* Connected to 127.0.0.1 (127.0.0.1) port 5000 (#0)
> GET / HTTP/1.1
> User-Agent: curl/7.35.0
> Host: 127.0.0.1:5000
> Accept: */*
>
* Empty reply from server
* Connection #0 to host 127.0.0.1 left intact
curl: (52) Empty reply from server

빈 답장...하지만 프로세스가 실행되고 있습니까?

$> sudo docker top 1c50b67d45b1
UID                 PID                 PPID                C                   STIME               TTY                 TIME                CMD
root                2084                812                 0                   10:26               ?                   00:00:00            python index.py
root                2117                2084                0                   10:26               ?                   00:00:00            /usr/bin/python index.py

이제 서버에 들어가서 확인해 보겠습니다...

$> sudo docker exec -it 1c50b67d45b1 bash
root@1c50b67d45b1:/srv# netstat -an
Active Internet connections (servers and established)
Proto Recv-Q Send-Q Local Address           Foreign Address         State
tcp        0      0 127.0.0.1:5000          0.0.0.0:*               LISTEN
tcp        0      0 127.0.0.1:47677         127.0.0.1:5000          TIME_WAIT
Active UNIX domain sockets (servers and established)
Proto RefCnt Flags       Type       State         I-Node   Path
root@1c50b67d45b1:/srv# curl -I 127.0.0.1:5000
HTTP/1.0 200 OK
Content-Type: text/html; charset=utf-8
Content-Length: 5447
Server: Werkzeug/0.10.4 Python/2.7.6
Date: Tue, 19 May 2015 12:18:14 GMT

괜찮아요...하지만 외부에서는 그렇지 않습니다.
내가 뭘 잘못하고 있는 거지?

문제는 로컬 호스트 인터페이스에만 바인딩되므로0.0.0.0외부에서 컨테이너에 액세스할 수 있도록 하려면 다음과 같이 하십시오.변경하는 경우:

if __name__ == '__main__':
    app.run()

로.

if __name__ == '__main__':
    app.run(host='0.0.0.0')

그건 작동할 거야.

이는 호스트의 모든 인터페이스에 바인딩되며, 경우에 따라 보안 위험이 될 수 있습니다. 특정 인터페이스에 바인딩하는 방법에 대한 자세한 내용은 https://stackoverflow.com/a/58138250/4332 을 참조하십시오.

사용 시flask대신 명령합니다.app.run당신은 통과할 수 있습니다.--host호스트를 변경하는 옵션입니다.도커의 줄은 다음과 같습니다.

CMD ["flask", "run", "--host", "0.0.0.0"]

또는

CMD flask run --host 0.0.0.0

도커 컨테이너에 둘 이상의 네트워크 인터페이스가 있습니다.예를 들어, 내 컨테이너에는 다음이 있습니다.

$ ip addr
1: lo: <LOOPBACK,UP,LOWER_UP> mtu 65536 qdisc noqueue state UNKNOWN group default qlen 1000
    link/loopback 00:00:00:00:00:00 brd 00:00:00:00:00:00
    inet 127.0.0.1/8 scope host lo
       valid_lft forever preferred_lft forever
32: eth0@if33: <BROADCAST,MULTICAST,UP,LOWER_UP> mtu 1500 qdisc noqueue state UP group default 
    link/ether 02:42:ac:11:00:02 brd ff:ff:ff:ff:ff:ff link-netnsid 0
    inet 172.17.0.2/16 brd 172.17.255.255 scope global eth0
       valid_lft forever preferred_lft forever

뛰면docker network inspect bridge위의 출력에서 두 번째 인터페이스를 사용하여 컨테이너가 브리지에 연결되어 있는 것을 확인할 수 있습니다.이 기본 브리지는 호스트의 도커 프로세스에도 연결됩니다.

따라서 다음 명령을 실행해야 합니다.

CMD flask run --host 172.17.0.2

호스트 컴퓨터에서 도커 컨테이너에서 실행 중인 플라스크 앱에 액세스합니다.교체하다172.17.0.2컨테이너의 특정 IP 주소가 무엇이든 간에.

호스트를 다음으로 수정해야 합니다.0.0.0.0도커 파일에 있습니다.이것은 최소한의 예입니다.

# Example of Dockerfile

FROM python:3.8.5-alpine3.12

WORKDIR /app

EXPOSE 5000
ENV FLASK_APP=app.py

COPY . /app
RUN pip install -r requirements.txt

ENTRYPOINT [ "flask"]
CMD [ "run", "--host", "0.0.0.0" ]

그리고 파일app.py이라

# app.py
from flask import Flask

app = Flask(__name__)

@app.route("/")
def home():
    return "Hello world"


if __name__ == "__main__":
    app.run()

다음으로 컴파일합니다.

docker build . -t deploy_flask

로 실행합니다.

docker run -p 5000:5000 -t -i deploy_flask:latest

다음을 사용하여 응답을 확인할 수 있습니다.curl http://127.0.0.1:5000/ -v

우선 파이썬 스크립트에서 코드를 변경해야 합니다.

app.run()

로.

app.run(host="0.0.0.0")

둘째, 도커 파일에서 마지막 줄은 다음과 같아야 합니다.

CMD ["flask", "run", "-h", "0.0.0.0", "-p", "5000"]

호스트 시스템의 경우0.0.0.0:5000작동하지 않습니다. 그러면 당신은 시도해야 합니다.localhost:5000

참고 - CMD 명령은 올바른 명령이어야 합니다.CMD 명령은 컨테이너 실행을 위한 기본값을 제공하기 때문입니다.

다른 답변을 기반으로 하는 방법:

두 대의 컴퓨터가 있다고 상상해 보세요.각 컴퓨터에는 공용 IP인 네트워크 인터페이스(WiFi 등)가 있습니다.각 컴퓨터에는 127.0.0.1의 루프백/로컬 호스트 인터페이스가 있습니다.이것은 "그냥 이 컴퓨터"를 의미합니다.

시스템 A의 127.0.0.1에 나열한 경우 시스템 B에서 실행될 때 127.0.0.1을 통해 연결할 수 없습니다.결국, 당신은 A 컴퓨터의 로컬, 개인 주소에서 청취를 요청했습니다.

도커는 기술적으로 동일한 컴퓨터이지만 리눅스 커널은 각 컨테이너가 자체적으로 분리된 네트워크 스택으로 실행되도록 허용합니다.따라서 컨테이너에 있는 127.0.0.1은 호스트가 아닌 다른 컴퓨터에 있는 127.0.0.1과 동일합니다. 이 컴퓨터에 연결할 수 없습니다.

긴 버전, 다이어그램 포함: https://pythonspeed.com/articles/docker-connection-refused/

빠른 독자를 위해 세 가지 빠른 확인 사항:

  1. 도커 파일에서 포트를 노출했는지 확인합니다.
  2. 하여 명령어떠한 명령어를 사용합니다.flask run --host=0.0.0.0
  3. run 하기 " " " "docker run -it -p5000:5000 yourImageName

를 내경호를다바인딩로로 입니다.0.0.0.0로컬 환경에서만 작동했으며 서버에 배포할 때 실패했습니다.

그러면 포트를 교체했을 때 작동합니다.--network=host:

이전:

docker run -d -p 5000:5000 <docker_image>

이후:

docker run -d --network=host <docker_image>

저는 ps를 했습니다. 나는 여전히 그것을 사용했습니다.0.0.0.0:5000플라스크 앱을 실행할 때 용기 안에 있습니다.

언급URL : https://stackoverflow.com/questions/30323224/deploying-a-minimal-flask-app-in-docker-server-connection-issues