Docker-compose编排微服务顺序启动
一、概述
微服务清单
| 服务名 | 端口 | 服务说明 | 依赖服务 | 启动优先级 |
| eureka-service | 8761 | 服务注册与发现 | --- | 1 |
| auth-service | 8888 | 认证服务 | eureka-service | 2 |
| user-service | 8763 | 用户服务 | eureka-service | 2 |
| gateway-service | 8081 | 网关服务 | eureka-service | 2 |
二、解决方案
针对这个问题,docker-compose官方给的建议是,通过wait-for-it.sh脚本来控制。
简单来说,这个脚本就是探测某个依赖的服务的tcp端口是否开放,否则一直等待,直到端口探测成功,才会启动后面的命令。
关于wait-for-it.sh的使用,请参考链接:
https://blog.csdn.net/wuzhong8809/article/details/82500722
环境说明
本文采用一台centos 7.6,通过docker-compose启动java spring cloud
ip地址:192.168.128.130
但是我测试了一下,使用wait-for-it.sh探测eureka的8761端口。发现虽然端口探测成功了,但是并不代表eureka完全启动好了。因此此时无法访问eureka的页面!
那么如何判断eureka完全启动好了呢?很简单,只需要访问eureka页面,http状态码为200,就表示启动好了。
curl -I -m 10 -o /dev/null -s -w %{http_code} http://eureka-server:8761
那么问题来了,还是得需要自己写检查脚本才行。
check_eureka.sh
#!/bin/bash while :
do
# 访问eureka注册中心,获取http状态码
CODE=`curl -I -m 10 -o /dev/null -s -w %{http_code} http://kitedge-eureka-server:8761`
# 判断状态码为200
if [[ $CODE -eq 200 ]]; then
# 输出绿色文字,并跳出循环
echo -e "\033[42;34m kitedge-eureka-server is ok \033[0m"
break
else
# 暂停1秒
sleep 1
fi
done # while结束时,也就是eureka启动完成后,执行容器中的run.sh。
bash /run.sh
说明:访问eureka注册中心,获取http状态码。当为200时,跳出循环,执行其他容器中的run.sh。
否则就一直循环,直到eureka页面正常为止。
目录结构
/opt/springcloud 目录结构如下:
./
├── auth
│ ├── auth-server.jar
│ ├── dockerfile
│ ├── repositories
│ └── run.sh
├── check_eureka.sh
├── docker-compose.yml
├── eureka
│ ├── dockerfile
│ ├── eureka-server.jar
│ ├── repositories
│ └── run.sh
├── gateway
│ ├── dockerfile
│ ├── gateway-server.jar
│ ├── repositories
│ └── run.sh
└── user
├── dockerfile
├── repositories
├── run.sh
└── user-service.jar
这里有4个微服务
docker-compose.yml
version: '3'
services:
eureka-server:
image: eureka-server:1
container_name: eureka-server
build: ./eureka
volumes:
- ./check_eureka.sh:/check_eureka.sh
- /data/log/tomcat:/data/log/tomcat
- /data/file:/data/file
environment:
- JVM=128m
ports:
- "8761:8761"
networks:
test_net:
aliases:
- eureka-server
restart: always
command: ["bash","/check_eureka.sh"]
healthcheck:
test: "/bin/netstat -anpt|grep 8761"
interval: 30s
timeout: 3s
retries: 1 auth-server:
image: auth-server:1
container_name: auth-server
build: ./auth
volumes:
- ./check_eureka.sh:/check_eureka.sh
- /data/log/tomcat:/data/log/tomcat
- /data/file:/data/file
environment:
- JVM=128m
ports:
- "8888:8888"
networks:
test_net:
aliases:
- auth-server
restart: always
depends_on:
- "eureka-server"
command: ["bash","/check_eureka.sh"]
healthcheck:
test: "/bin/netstat -anpt|grep 8888"
interval: 30s
timeout: 3s
retries: 1 user-service:
image: user-service:1
container_name: user-service
build: ./user
volumes:
- ./check_eureka.sh:/check_eureka.sh
- /data/log/tomcat:/data/log/tomcat
- /data/file:/data/file
environment:
- JVM=128m
ports:
- "8763:8763"
networks:
test_net:
aliases:
- user-service
restart: always
depends_on:
- "eureka-server"
command: ["bash","/check_eureka.sh"]
healthcheck:
test: "/bin/netstat -anpt|grep 8763"
interval: 30s
timeout: 3s
retries: 1 gateway-server:
image: gateway-server:1
container_name: gateway-server
build: ./gateway
volumes:
- ./check_eureka.sh:/check_eureka.sh
- /data/log/tomcat:/data/log/tomcat
- /data/file:/data/file
environment:
- JVM=128m
ports:
- "8081:8081"
networks:
test_net:
aliases:
- gateway-server
restart: always
depends_on:
- "eureka-server"
command: ["bash","/check_eureka.sh"]
healthcheck:
test: "/bin/netstat -anpt|grep 8081"
interval: 30s
timeout: 3s
retries: 1 networks:
test_net:
external: true
参数解释:
image:指定服务的镜像名称或镜像ID。如果镜像在本地不存在,Compose将会尝试拉取镜像。
container_name:容器名称,相当于docker run --name xxx,里面的--name参数。
build:指定Dockerfile所在文件夹的路径。Compose将会利用Dockerfile自动构建镜像,然后使用镜像启动服务容器。
volumes:挂载一个目录或者一个已存在的数据卷容器,相当于docker run -v xxx:xxx里面的-v参数。
environment:环境变量,相当于docker run -e xxx=xxx里面的-e参数。
ports:映射端口,相当于docker run -p xx:xx里面的-p参数。
networks:网络设置,这里表示连接test_net网桥,aliases表示设置别名。相当于docker run -it --network xx --network-alias xxx
restart:重启方式,相当于docker run --restart里面的--restart参数。
depends_on:指定依赖于哪个服务
command:覆盖容器启动后默认执行的命令,相当于docker run xxx /bin/bash里面最后一段命令。
healthcheck: 判断容器的状态是否正常
设置选项:
- test:测试命令
- --interval=DURATION (default: 30s),间隔
- --timeout=DURATION (default: 30s), 超时时间
- --start-period=DURATION (default: 0s),初始化时间
- --retries=N (default: 3),当连续失败指定次数后,容器状态会变成unhealthy
这里主要介绍eureka目录下的几个文件。
dockerfile
FROM mayan31370/openjdk-alpine-with-chinese-timezone:8-jdk
ADD eureka-server.jar /
ADD run.sh /
RUN chmod 755 run.sh && mkdir -p /data/log/tomcat /data/file EXPOSE 8761
ENTRYPOINT [ "/run.sh"]
repositories
#https://mirrors.aliyun.com/alpine/v3.6/main/
#https://mirrors.aliyun.com/alpine/v3.6/community/
#aliyun
https://mirrors.aliyun.com/alpine/v3.6/main/
https://mirrors.aliyun.com/alpine/v3.6/community/
run.sh
#!/bin/sh
java -Xms${JVM} -Xmx${JVM} -XX:MetaspaceSize=50M -XX:MaxMetaspaceSize=256m -jar /eureka-server.jar
再来看auth下的几个文件。
dockerfile
FROM mayan31370/openjdk-alpine-with-chinese-timezone:8-jdk
ADD auth-server.jar /
ADD repositories /etc/apk/repositories
ADD run.sh /
RUN chmod 755 run.sh && mkdir -p /data/log/tomcat /data/file && apk update && apk add bash curl EXPOSE 8888
#ENTRYPOINT [ "/run.sh"]
注意:这里注释掉了ENTRYPOINT ,因为它由check_eureka.sh脚本来启动。注意看此脚本的最后一行,它就是来执行run.sh的。
repositories
内容同上,这里就不重复贴了
run.sh
#!/bin/sh
java -Xms${JVM} -Xmx${JVM} -XX:MetaspaceSize=50M -XX:MaxMetaspaceSize=256m -jar /auth-server.jar
另外2个微服务,user和gateway。和auth也是类似的,只不过dockerfile和run.sh里面的jar包名不一样,其他内容都是一样的。
启动服务
创建bridge网络
docker network create test_net
一切准备就绪了,先build,再启动。
cd /opt/springcloud
docker-compose build
docker-compose up -d
等待几分钟,查看容器状态
# docker-compose ps
Name Command State Ports
--------------------------------------------------------------------------------------
auth-server bash /check_eureka.sh Up (healthy) 0.0.0.0:8888->8888/tcp
eureka-server /run.sh bash /check_eureka.sh Up (healthy) 0.0.0.0:8761->8761/tcp
gateway-server bash /check_eureka.sh Up (healthy) 0.0.0.0:8081->8081/tcp
user-service bash /check_eureka.sh Up (healthy) 0.0.0.0:8763->8763/tcp
本文参考连接:
https://blog.csdn.net/wuzhong8809/article/details/82500722
https://www.jianshu.com/p/d2f39703bcb5
Docker-compose编排微服务顺序启动的更多相关文章
- 使用Docker Compose编排微服务
一般微服务架构会包含若干个微服务,而每个微服务可以有多个实例,如果每个微服务都有手动启停,那么效率就很低.维护量比较大. 所以我们可以使用Docker Compose来轻松.高效地管理容器. 一.安装 ...
- 使用Docker Compose编排Spring Cloud微服务
文章目录 微服务构建实例 简化Compose的编写 编排高可用的Eureka Server 编排高可用Spring Cloud微服务集群及动态伸缩 微服务项目名称 项目微服务中的角色 microser ...
- 使用Docker compose编排Laravel应用
前言 Laravel官方开发环境推荐的是Homestead(其实就是一个封装好的Vagrant box),我感觉这个比较重,于是自己用Docker compose编排了一套开发环境,在这里分享下. 环 ...
- Docker从入门到掉坑(二):基于Docker构建SpringBoot微服务
本篇为Docker从入门到掉坑第二篇:基于Docker构建SpringBoot微服务,没有看过上一篇的最好读过 Docker 从入门到掉坑 之后,阅读本篇. 在之前的文章里面介绍了如何基于docker ...
- Docker Compose编排工具部署lnmp实践及理论(详细)
目录 一.理论概述 编排 部署 Compose原理 二.使用docker compose 部署lnmp 三.测试 四.总结 一.理论概述 Docker Compose是一个定义及运行多个Docker容 ...
- SpringCloud微服务实战——搭建企业级开发框架(三十五):SpringCloud + Docker + k8s实现微服务集群打包部署-集群环境部署
一.集群环境规划配置 生产环境不要使用一主多从,要使用多主多从.这里使用三台主机进行测试一台Master(172.16.20.111),两台Node(172.16.20.112和172.16.20.1 ...
- Docker(二十六)-Docker Compose编排容器
1. 前言 Docker Compose 是 Docker 容器进行编排的工具,定义和运行多容器的应用,可以一条命令启动多个容器. 使用Compose 基本上分为三步: Dockerfile 定义应用 ...
- Docker系列之(五):使用Docker Compose编排容器
1. 前言 Docker Compose 是 Docker 容器进行编排的工具,定义和运行多容器的应用,可以一条命令启动多个容器. 使用Compose 基本上分为三步: Dockerfile 定义应用 ...
- 基于docker部署的微服务架构(四): 配置中心
原文:http://www.jianshu.com/p/b17d65934b58%20 前言 在微服务架构中,由于服务数量众多,如果使用传统的配置文件管理方式,配置文件分散在各个项目中,不易于集中管理 ...
随机推荐
- impala支持的数据库里的double(float)类型,通过迁移inceptor后,类型的值都变成了null
impala支持的数据库里的double(float)类型,通过迁移inceptor后,double类型的值都变成了null. 通过查阅日志发现默认将double转换成Decimal(38,10)然而 ...
- 2019牛客暑期多校训练营(第五场)I.three points 1(暴力几何)
题意:现在给你一个矩形边框 一个三角形的三边长 现在问你能否把三角形放入矩阵边框中 并且输出三个点的坐标 思路:我们可以发现如果一定有解 我们就可以让一个点在左下角(0,0)处 还有一个点在矩形边上 ...
- Codeforces Round #651 (Div. 2) A. Maximum GCD(数论)
题目链接:https://codeforces.com/contest/1370/problem/A 题意 有 $n$ 个数大小分别为 $1$ 到 $n$,找出两个数间最大的 $gcd$ . 题解 若 ...
- Luogu 2017 Autumn Camping 游记
颓得不行的我到D2才想起来自己可以写一篇low得不能再low的游记,然后就动笔了...... Day0 愉快地看着三联,想着别人放一天我放四天的悠闲生活,内心甚是平静.然而晚上回到家就开始浪了,看完了 ...
- Luogu T7468 I liked Matrix!
题目链接 题目背景 无 题目描述 在一个n*m 的矩阵A 的所有位置中随机填入0 或1,概率比为x : y.令B[i]=a[i][1]+a[i][2]+......+a[i][m],求min{B[i] ...
- HDOJ 3398
这个题坑了太久太久啊!!!!!贡献了得有30+WA才发现 原来是因为在乘法中有溢出导致一直TLE啊.... 但是到最后也不知道有个问题怎么解决的. 就是在getp()中的num值的诡异的改变! #in ...
- Strongly connected HDU - 4635 原图中在保证它不是强连通图最多添加几条边
1 //题意: 2 //给你一个有向图,如果这个图是一个强连通图那就直接输出-1 3 //否则,你就要找出来你最多能添加多少条边,在保证添加边之后的图依然不是一个强连通图的前提下 4 //然后输出你最 ...
- 牛客编程巅峰赛S1第5场 - 青铜&白银 B.完全平方数的尾巴 (暴力)
题意:有一个数\(x\),判断其是否能有某个完全平方数$mod$1000得到. 题解:直接写个for判断一下就好了,因为对1000取模,所以枚举到1000即可. 代码: class Solution ...
- git忽略规则以及.gitignore文件不生效解决办法
正文 Git忽略规则: #此为注释 – 内容被 Git 忽略 .sample # 忽略所有 .sample 结尾的文件 !lib.sample # 但 lib.sample 除外 /TODO # 仅仅 ...
- SpringBoot整合shiro-MD5盐值加密
为什么要进行密码加密? 在我们的日常生活中,许多人有着在不同网站上使用相同密码的坏习惯(包括我也是qaq),假如应用程序或服务器出现漏洞,数据被窃取,用户的明文密码直接被暴露给黑客.显然后果将不堪设想 ...