为什么我的 .Dockerignore 文件不忽略文件?

G A*_*X Y 6 python django docker dockerignore

当我构建容器并检查应该被忽略的文件时,大多数文件都没有被忽略。

这是我的文件夹结构

Root/

  data/

  project/
    __pycache__/
    media/
    static/

  app/
    __pycache__/
    migrations/
    templates/

  .dockerignore
  .gitignore
  .env
  docker-compose.yml
  Dockerfile
  requirements.txt
  manage.py
Run Code Online (Sandbox Code Playgroud)

假设我想忽略__pycache__& data(创建容器时将使用命令创建数据docker-compose up)文件夹和.gitignore&.env文件。

我将在下一个.dockerignore文件中忽略这些

.git
.gitignore

.docker

*/__pycache__/
**/__pycache__/

.env/
.venv/
venv/

data/
Run Code Online (Sandbox Code Playgroud)

最终的结果是只有git&.env文件被忽略了。数据文件夹并未被忽略,但无法从容器访问。而且__pycache__文件夹也没有被忽略。

这是 docker 文件。

docker-compose.yml

version: "3.8"

services: 

  app: 
    build: .
    volumes: 
      - .:/django-app
    ports: 
      - 8000:8000
    command: /bin/bash -c "sleep 7; python manage.py migrate; python manage.py runserver 0.0.0.0:8000"
    container_name: app-container
    depends_on: 
      - db

  db:
    image: postgres
    volumes:
      - ./data:/var/lib/postgresql/data
    environment: 
      - POSTGRES_DB=${DB_NAME}
      - POSTGRES_USER=${DB_USER}
      - POSTGRES_PASSWORD=${DB_PASSWORD}
    container_name: postgres_db_container
Run Code Online (Sandbox Code Playgroud)

Dockerfile

FROM python:3.9-slim-buster

ENV PYTHONUNBUFFERED=1

WORKDIR /django-app
EXPOSE 8000

COPY requirements.txt requirements.txt

RUN apt-get update \
    && adduser --disabled-password --no-create-home userapp \
    && apt-get -y install libpq-dev \
    && apt-get -y install apt-file \
    && apt-get -y install python3-dev build-essential \
    && pip install -r requirements.txt

USER userapp
Run Code Online (Sandbox Code Playgroud)

Dav*_*aze 11

您实际上是使用 注入源代码volumes:,而不是在映像构建过程中,这不尊重.dockerignore.

像这样运行 Docker 应用程序分两个阶段进行:

  1. 您构建一个可重用的映像,其中包含应用程序运行时、任何操作系统和特定于语言的库依赖项以及应用程序代码;然后
  2. 您运行基于该映像的容器。

.dockerignore仅在第一个构建阶段才考虑该文件。COPY在您的设置中,除了文件之外,图像中实际上没有任何内容requirements.txt。相反,您可以volumes:将主机系统的一部分注入到容器中。这发生在第二阶段,并忽略.dockerignore

我建议的方法是跳过volumes:, 而是跳过COPYDockerfile 中所需的源代码。您通常还应该CMD在 Dockerfile 中指示容器将运行的默认值,而不是要求它使用docker-compose.ymldocker run命令。

FROM python:3.9-slim-buster

# Do the OS-level setup _first_ so that it's not repeated
# if Python dependencies change
RUN apt-get update && apt-get install -y ...

WORKDIR /django-app

# Then install Python dependencies
COPY requirements.txt .
RUN pip install -r requirements.txt

# Then copy in the rest of the application
# NOTE: this _does_ honor .dockerignore
COPY . .

# And explain how to run it
ENV PYTHONUNBUFFERED=1
EXPOSE 8000
USER userapp
# consider splitting this into an ENTRYPOINT that waits for the
# the database, runs migrations, and then `exec "$@"` to run the CMD
CMD sleep 7; python manage.py migrate; python manage.py runserver 0.0.0.0:8000
Run Code Online (Sandbox Code Playgroud)

这意味着,在docker-compose.yml设置中,您不需要volumes:;应用程序代码已经位于您构建的映像内。

version: "3.8"
services:
  app: 
    build: .
    ports: 
      - 8000:8000
    depends_on: 
      - db
    # environment: [PGHOST=db]
    # no volumes: or container_name:

  db:
    image: postgres
    volumes: # do keep for persistent database data
      - ./data:/var/lib/postgresql/data
    environment: 
      - POSTGRES_DB=${DB_NAME}
      - POSTGRES_USER=${DB_USER}
      - POSTGRES_PASSWORD=${DB_PASSWORD}
    # ports: ['5433:5432']
Run Code Online (Sandbox Code Playgroud)

这种方法还意味着当您的应用程序发生更改时,您需要docker-compose build一个新的映像。这在 Docker 中是正常的。

对于日常开发,一种有用的方法是在 Docker 中运行所有非应用程序依赖项,但在容器外部运行应用程序本身。

# Start the database but not the application
docker-compose up -d db

# Create a virtual environment and set it up
python3 -m venv venv
. venv/bin/activate
pip install -r requirements.txt

# Set environment variables to point at the Docker database
export PGHOST=localhost PGPORT=5433

# Run the application locally
./manage.py runserver
Run Code Online (Sandbox Code Playgroud)

执行此操作需要使数据库从 Docker 外部可见(通过ports:),并使数据库位置可配置(可能通过环境变量,在 Compose 中设置environment:)。