Docker组成了在烧瓶应用程序启动之前完全启动mysql数据库的方法

时间:2018-11-10 01:41:26

标签: docker docker-compose

我刚刚开始对我的应用进行docker化。我已经建立了Dockerfiledocker-compose.yml,除了一件事以外,其他所有功能似乎都可以正常工作。有时我的flask应用程序启动得太快,并抛出连接拒绝错误(因为MySQL数据库未完全启动)。我正在使用healthcheck来检查数据库是否启动,但这似乎并不可靠(我什至确保可以看到show databases,但是mysql显然在运行状况检查通过后初始化了更多东西?确定接下来要进行哪些健康检查)。在我的输出中,我看到确实首先创建了数据库,但是在flask应用程序启动时它仍在初始化。理想情况下,当我运行docker-compose up时,我希望能够首先看到此行,

db_1_eae741771281 | 2018-11-10T00:50:21.473098Z 0 [Note] mysqld: ready for connections.

,然后启动我的flask应用程序入口点。当前,它不执行此操作。

在启动start.sh之前,是否有更可靠的方法来确保MySQL完全启动?

Dockerfile:

FROM python:3.5-alpine

RUN apk update && apk upgrade

RUN apk add --no-cache curl python build-base openldap-dev python2-dev python3-dev pkgconfig python-dev libffi-dev musl-dev make gcc

RUN pip install --upgrade pip

RUN adduser -D user

WORKDIR /home/user

COPY requirements.txt requirements.txt
RUN python -m venv venv
RUN venv/bin/pip install -r requirements.txt

COPY app app
COPY start.sh ./
RUN chmod +x start.sh

RUN chown -R user:user ./
USER user

EXPOSE 5000
ENTRYPOINT ["./start.sh"]

docker-compose.yml:

version: "2.1"
services:
  db:
    image: mysql:5.7
    ports:
      - "32000:3306"
    environment:
      - MYSQL_DATABASE=mydb
      - MYSQL_USER=user
      - MYSQL_PASSWORD=user123
      - MYSQL_ROOT_PASSWORD=user123
    volumes:
      - ./db:/docker-entrypoint-initdb.d/:ro
    healthcheck:
            test: "mysql --user=user --password=user123 --execute \"SHOW DATABASES;\""
            timeout: 20s
            retries: 20

  app:
    build: ./
    ports:
      - "5000:5000"
    depends_on:
      db:
        condition: service_healthy

start.sh

#!/bin/sh

source venv/bin/activate
# Start Gunicorn processes
echo Starting Gunicorn.

exec gunicorn -b 0.0.0.0:5000 wsgi --chdir my_app --timeout 9999 --workers 3 --access-logfile - --error-logfile - --capture-output --log-level debug

3 个答案:

答案 0 :(得分:1)

使用健康检查较为容易,但这完全取决于检查的可靠性。

另一种方法是依赖应用容器中的wait-for-itwait-for之类的项目。
由于您的连接被拒绝,因此这些脚本仅在可能建立连接后才返回,并且您的应用只能在此之后启动。

此外,如果仍然无法使用,则可以使用一个单独的脚本(在您的情况下为python)进行检查,直到数据库就绪为止,并且可以在启动烧瓶之前在start.sh中调用此脚本应用程序。

答案 1 :(得分:1)

好的,我也遇到health_check ...

的问题

也许不是最佳选择,但最可靠的解决方案是在启动应用程序之前使用MySQL客户端(mysqladmin)ping MySQL服务器。

1 -创建一个wait.sh脚本(db是您的MySQL服务名称):

#!/bin/sh

# Wait until MySQL is ready
while ! mysqladmin ping -h"db" -P"3306" --silent; do
    echo "Waiting for MySQL to be up..."
    sleep 1
done

2 -从app Dockerfile获取一个MySQL客户端:

# install mysql client, will be used to ping mysql
apt-get -y install mysql-client

3 -在您的docker-compose.yml文件中,只需将脚本添加到您的容器中(我使用了卷,但您可以继续使用COPY)并在运行之前运行wait.sh start.sh

app:
    build: ./
    ports:
      - "5000:5000"
    depends_on:
      db:
    command: bash -c "/usr/local/bin/wait.sh && /usr/local/bin/start.sh"
    volumes:
      - ./start.sh:/usr/local/bin/start.sh
      - ./wait.sh:/usr/local/bin/wait.sh

这应该有效。

如果您真的不想下载MySQL客户端,请尝试以下操作(同样,db是您的MySQL服务名称)。它在我的大多数项目中都有效,但并非全部(可能取决于发行版?):

#!/bin/sh

# Wait until MySQL is ready
while ! exec 6<>/dev/tcp/db/3306; do
    echo "Trying to connect to MySQL at 3306..."
    sleep 5
done

PS:避免将服务命名为“ app”或“ db”,如果其他容器具有相同的服务名称(即使在不同的网络中),以后可能会出现问题。

答案 2 :(得分:0)

这是多个容器的常见问题。很难控制不同容器的启动速度。在这种情况下,像Kubernetes这样的容器编排解决方案可能会为您提供帮助。 Kubernetes具有init容器的概念,它可以在依赖容器启动之前运行完成。您可以在此处找到初始化容器的示例

https://www.handsonarchitect.com/2018/08/understand-kubernetes-object-init.html

此youtube视频可能也对您有所帮助 https://www.youtube.com/watch?v=n2FPsunhuFc