我刚刚开始对我的应用进行docker化。我已经建立了Dockerfile
和docker-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
答案 0 :(得分:1)
使用健康检查较为容易,但这完全取决于检查的可靠性。
另一种方法是依赖应用容器中的wait-for-it
或wait-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