如何 运行 将烧瓶和芹菜放入一个容器中?

How to run Flask AND celery in one container?

我想用 docker 在 digitalocean 上部署我的小 Web 应用程序。问题是,我没有把烧瓶和芹菜分开,所以所有东西都在一个容器里。 每次我 运行 docker-compose up Web 容器崩溃 ("exited with code 0").

项目结构如下:

.
.
├── docker-compose.yml
├── nginx
│   ├── dockerfile
│   └── nginx.conf
└── web
    ├── app.ini
    ├── bot
    │   ├── __init__.py
    │   ├── main.py
    │   ├── __pycache__
    │   └── timer.py
    ├── celeryd.pid
    ├── config.py
    ├── dockerfile
    ├── dockerignore
    ├── flask_app.py
    ├── __init__.py
    ├── requirements.txt
    ├── tasks.py
    └── webapp
        ├── bot
        ├── __init__.py
        ├── models.py
        ├── __pycache__
        ├── static
        ├── templates
        └── users

这是docker-compose.yml:

version: "3.7"

services:

  rabbitmq:
    image: rabbitmq:3-management
    hostname: rabbitmq
    container_name: rabbitmq
    environment:
      RABBITMQ_ERLANG_COOKIE: 'SWQOKODSQALRPCLNMEQG'
      RABBITMQ_DEFAULT_USER: 'user123'
      RABBITMQ_DEFAULT_PASS: 'password123'
      RABBITMQ_DEFAULT_VHOST: '/webapp'
    ports:
      - "15672:15672"
      - "5672:5672"
    volumes:
      - rabbitmq_storage:/var/lib/rabbitmq
    networks:
      netzwerk123:
        aliases:
          - rabbitmq

  db:
    image: mariadb
    restart: always
    container_name: db
    environment:
      MYSQL_ROOT_PASSWORD: 'password123'
      MYSQL_USER: 'user123'
      MYSQL_PASSWORD: 'password123'
      MYSQL_DATABASE: 'webapp'
    ports:
      - "3306:3306"
    volumes:
      - db_storage:/var/lib/mysql
    networks:
      netzwerk123:
        aliases:
          - db


  nginx:
    build: ./nginx  
    container_name: nginx
    restart: always
    ports:  
      - "80:80"
    networks:
      netzwerk123:
        aliases:
          - ngnix


  web:
    build: ./web  
    container_name: web
    hostname: web
    restart: always
    environment:    
      FLASK_DEBUG: 'True'
      FLASK_APP: 'flask_app.py'
    command: celery -A tasks.celery worker --loglevel=info --detach
    expose:
      - 8080
    networks:
      netzwerk123:
        aliases:
          - web
    depends_on:     
      - nginx
      - db
      - rabbitmq
    links:
      - rabbitmq
    tty: true
    stdin_open: true


  adminer:        #only for development purposes
    image: adminer
    container_name: adminer
    restart: always
    ports:
      - "8081:8080"
    depends_on:
      - db
    networks:
      netzwerk123:
        aliases:
          - adminer



volumes:                                    
  db_storage:
  rabbitmq_storage:

networks:
  netzwerk123:

如果我评论 "command: celery -A tasks.celery worker --loglevel=info --detach" 一切 运行 都很好,因为芹菜不是 运行ning。网页功能齐全,数据库处于活动状态。一切都很好,除了芹菜。

通过取消注释命令行,docker 抛出 "web exited with error code 0"。

我猜这是因为 docker 在成功执行命令后退出了容器。所以我面临两个问题:

1) 如何将运行 celery 放在同一容器的celery 后台,这样才不会出现错误码0?

2) 如何运行 celery AND flask 并列在容器中?

-> 我已经尝试了以下,但同样的错误:

command: bash -c "uwsgi app.ini && celery -A tasks.celery worker --loglevel=info --detach"

非常感谢任何帮助:)

编辑: 我将芹菜 + 烧瓶放在一个容器中的原因是我的 task.py。在这里,我从 web.webapp.models 导入所有 sqlalchemy 模型。所以我需要访问 flask 目录,目前我不知道如何将它分成两个容器

如果您在两个不同进程之间 "share" 唯一需要的是代码,那么它已经在构建的 Docker 映像中。您可以很容易地 运行 同一图像中的两个不同容器,但使用不同的命令。

services:
  web:
    build: ./web  
    environment:    
      FLASK_DEBUG: 'True'
      FLASK_APP: 'flask_app.py'
    command: flask run --host=0.0.0.0
    depends_on:     
      - nginx
      - db
      - rabbitmq
  celery:
    build: ./web  
    restart: always
    command: celery -A tasks.celery worker --loglevel=info
    depends_on:     
      - db
      - rabbitmq

请注意,这两个 build: 是同一张图片。另外,我将 celery worker 设为 --detach,因此它 运行 是前台进程;只要 command: 仍然是 运行ning,容器就会保持 运行ning。

(要使此片段在原始 docker-compose.yml 文件的上下文中工作,您需要彻底删除整个文件中的所有 networks: 块,但它会正常工作:Docker Compose 创建一个默认网络并使用其服务块的名称将所有服务附加到该网络。我还删除了许多其他无关紧要的选项。)