小知识:docker compose 服务启动顺序控制的方法

概要

dockercompose 可以方便组合多个 docker 容器服务, 但是, 当容器服务之间存在依赖关系时, docker-compose 并不能保证服务的启动顺序.

docker-compose 中的 depends_on 配置是容器的启动顺序, 并不是容器中服务的启动顺序.

问题重现

首先, 我们构造一个示例, 来演示 docker-compose 带来的问题. docker-compose.yml 文件如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
version: 2
services:
web:
image: ubuntu:14.04
depends_on:
– web
command: nc -z database 3306
database:
image: ubuntu:14.04
command: >
/bin/bash -c
sleep 5;
echo “sleep over”;
nc -lk 0.0.0.0 3306;

启动后, 可以发现, 确实是先启动 database, 后启动 web, 但是 database 中的服务是在大约 5 秒后才完成的, 所以导致 web 的启动失败.

?
1
2
3
4
5
6
7
$ docker-compose up
Creating tmp_database_1 … done
Creating tmp_database_1 …
Creating tmp_web_1   … done
Attaching to tmp_database_1, tmp_web_1
tmp_web_1 exited with code 1
database_1 | sleep over

问题解决方式 1.0

修改 web 的启动脚本, 等待 database 的端口通了之后再启动服务

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
version: 2
services:
web:
image: ubuntu:14.04
depends_on:
– database
command: >
/bin/bash -c
while ! nc -z database 3306;
do
echo “wait for database”;
sleep 1;
done;
echo “database is ready!”;
echo “start web service here”;
database:
image: ubuntu:14.04
command: >
/bin/bash -c
sleep 5;
echo “sleep over”;
nc -lk 0.0.0.0 3306;

再次启动,

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
$ docker-compose up
Creating tmp_database_1 … done
Creating tmp_database_1 …
Creating tmp_web_1   … done
Attaching to tmp_database_1, tmp_web_1
web_1    | wait for database
web_1    | wait for database
web_1    | wait for database
web_1    | wait for database
web_1    | wait for database
database_1 | sleep over
web_1    | database is ready!
web_1    | start web service here
tmp_web_1 exited with code 0

web 会在 database 启动完成, 端口通了之后才启动.

问题解决方式 2.0

上面的解决方式虽然能够解决问题, 但是在 yaml 中直接插入脚本不好维护, 也容易出错. 如果有多个依赖, 或者多层依赖的时候, 复杂度会直线上升.

所以, 要封装一个 entrypoint.sh 脚本, 可以接受启动命令, 以及需要等待的服务和端口. 脚本内容如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
#!/bin/bash
#set -x
#******************************************************************************
# @file  : entrypoint.sh
# @author : wangyubin
# @date  : 2018-08- 1 10:18:43
#
# @brief  : entry point for manage service start order
# history : init
#******************************************************************************
: ${SLEEP_SECOND:=2}
wait_for() {
echo Waiting for $1 to listen on $2…
while ! nc -z $1 $2; do echo waiting…; sleep $SLEEP_SECOND; done
}
declare DEPENDS
declare CMD
while getopts “d:c:” arg
do
case $arg in
d)
DEPENDS=$OPTARG
;;
c)
CMD=$OPTARG
;;
?)
echo “unkonw argument”
exit 1
;;
esac
done
for var in ${DEPENDS//,/ }
do
host=${var%:*}
port=${var#*:}
wait_for $host $port
done
eval $CMD

这个脚本有 2 个参数, -d 需要等待的服务和端口, -c 等待的服务和端口启动之后, 自己的启动命令

修改 docker-compose.yml, 使用 entrypoint.sh 脚本来控制启动顺序.

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
version: 2
services:
web:
image: ubuntu:14.04
depends_on:
– database
volumes:
– “./entrypoint.sh:/entrypoint.sh”
entrypoint: /entrypoint.sh -d database:3306 -c echo “start web service here”;
database:
image: ubuntu:14.04
command: >
/bin/bash -c
sleep 5;
echo “sleep over”;
nc -lk 0.0.0.0 3306;

实际使用中, 也可以将 entrypoint.sh 打包到发布的镜像之中, 不用通过 volumes 配置来加载 entrypoint.sh 脚本.

测试结果如下:

?
1
2
3
4
5
6
7
8
9
10
11
$ docker-compose up
Starting tmp_database_1 … done
Starting tmp_web_1 … done
Attaching to tmp_database_1, tmp_web_1
web_1    | Waiting for database to listen on 3306…
web_1    | waiting…
web_1    | waiting…
web_1    | waiting…
database_1 | sleep over
web_1    | start web service here
tmp_web_1 exited with code 0

补充

依赖多个服务和端口

使用上面的 entrypoint.sh 脚本, 也可以依赖多个服务和端口, -d 参数后面的多个服务和端口用逗号(,)隔开.

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
version: 2
services:
web:
image: ubuntu:14.04
depends_on:
– mysql
– postgresql
volumes:
– “./entrypoint.sh:/entrypoint.sh”
entrypoint: /entrypoint.sh -d mysql:3306,postgresql:5432 -c echo “start web service here”;
mysql:
image: ubuntu:14.04
command: >
/bin/bash -c
sleep 4;
echo “sleep over”;
nc -lk 0.0.0.0 3306;
postgresql:
image: ubuntu:14.04
command: >
/bin/bash -c
sleep 8;
echo “sleep over”;
nc -lk 0.0.0.0 5432;

执行的效果可以自行尝试.

尝试间隔的配置

每次尝试连接的等待时间可以通过 环境变量 SLEEP_SECOND 来配置, 默认 2 秒 下面的配置等待时间设置为 4 秒, 就会每隔 4 秒才去尝试 mysql 服务时候可连接.

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
version: 2
services:
web:
image: ubuntu:14.04
environment:
SLEEP_SECOND: 4
depends_on:
– mysql
volumes:
– “./entrypoint.sh:/entrypoint.sh”
entrypoint: /entrypoint.sh -d mysql:3306 echo “start web service here”;
mysql:
image: ubuntu:14.04
command: >
/bin/bash -c
sleep 4;
echo “sleep over”;
nc -lk 0.0.0.0 3306;

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

原文链接:https://www.cnblogs.com/wang_yb/p/9400291.html

声明: 猿站网有关资源均来自网络搜集与网友提供,任何涉及商业盈利目的的均不得使用,否则产生的一切后果将由您自己承担! 本平台资源仅供个人学习交流、测试使用 所有内容请在下载后24小时内删除,制止非法恶意传播,不对任何下载或转载者造成的危害负任何法律责任!也请大家支持、购置正版! 。本站一律禁止以任何方式发布或转载任何违法的相关信息访客发现请向站长举报,会员发帖仅代表会员个人观点,并不代表本站赞同其观点和对其真实性负责。本网站的资源部分来源于网络,如有侵权烦请发送邮件至:2697268773@qq.com进行处理。
建站知识

小知识:详解Docker-compose networks 的例子

2023-4-5 15:03:58

建站知识

小知识:Docker 基础网络配置详解

2023-4-5 15:18:18

0 条回复 A文章作者 M管理员
    暂无讨论,说说你的看法吧
个人中心
购物车
优惠劵
今日签到
有新私信 私信列表
搜索