• 起源
  • 守护进程daemon
  • 从守护进程的角度看Docker Compose
  • Docker的解决方案
  • 思路
  • 代码
  • 结果

起源

Docker Compose提供了一个depends_on参数。

https://docs.docker.com/compose/compose-file/#depends_on

depends_on参数用于描述服务之间的依赖关系,服务依赖将导致如下行为:

  • docker-compose up按照依赖关系的顺序启动服务。
  • docker-compose up SERVICE自动包含SERVICE的依赖关系。

看起来功能强大,不过接下来又说了使用depends_on的注意事项:

  • depends_on在启动服务时并不会等待相关所依赖的服务完成。

也就是说,depends_on的主要功能是自动包含SERVICE的依赖关系, 
而所谓的按照依赖关系的顺序启动服务在实践中几乎毫无意义, 
因为很多被依赖的服务往往启动缓慢,例如数据库。

分析Docker的工作原理会发现, 
管理依赖关系的挑战在于Docker本身并不管理服务的启动的过程, 
一个Docker服务在启动entrypoint或者command进程的时候就开始了, 
一直到这个进程退出后才能被Docker识别服务结束。

Docker服务的生命周期不同于使用服务角度的声明周期, 
从使用服务的角度看,有这么几个关键点

  1. 服务的主进程启动
  2. 服务开始向外提供服务
  3. 服务向外提供服务
  4. 服务终止向外提供服务
  5. 服务的主进程结束
服务的主进程启动服务开始向外提供服务服务向外提供服务服务终止向外提供服务服务的主进程结束
从使用服务的角度看Docker服务的生命周期服务服务客户端客户端主进程启动服务开始向外提供服务请求响应服务终止向外提供服务主进程结束

然后结合Docker服务的生命周期的关键点一起看

  1. entrypoint或者command进程启动
  2. 若干启动辅助进程启动到结束
  3. 服务的主进程启动
  4. 服务开始向外提供服务
  5. 服务向外提供服务
  6. 服务终止向外提供服务
  7. 服务的主进程结束
  8. 若干结束辅助进程启动到结束
  9. entrypoint或者command进程结束
entrypoint或者command进程启动若干启动辅助进程启动到结束服务的主进程启动服务开始向外提供服务服务向外提供服务服务终止向外提供服务服务终止向外提供服务若干结束辅助进程启动到结束entrypoint或者command进程结束
结合Docker的生命周期看服务的生命周期Docker容器Docker容器启动辅助进程启动辅助进程结束辅助进程结束辅助进程服务主进程服务主进程服务工作进程服务工作进程客户端客户端管理端管理端启动启动结束启动启动服务开始向外提供服务请求响应关闭关闭服务终止向外提供服务结束结束启动结束结束

Docker可以识别entrypoint或者command进程的启动和结束, 
但是里面的细节就很难识别了。

守护进程daemon

在启动服务的过程中还有一个容易混淆的地方————守护进程daemon, 
Linux中的服务往往以守护进程daemon的形式出现。

所有守护进程的启动脚本,都放在/etc/init.d目录下面。 
init进程的主要任务,就是逐一运行这些脚本。

Linux中一个守护进程的父进程是init进程, 
因为守护进程真正的父进程在fork出子进程后就先于子进程exit退出了, 
所以守护进程是一个由init继承的孤儿进程。

这种工作方式有两个基本的用途。

  • 用于启动的父进程的运行权限较高, 
    fork子进程的同时可以设置子进程运行于不同的环境,例如不同的用户。
  • 用于启动的父进程在init中是串行的,而且可以依循一定的顺序, 
    fork具体提供服务的子进程后,父进程退出, 
    init运行下一个守护进程的启动的父进程, 
    下一个守护进程的启动的父进程就可以使用前面的守护进程的子进程提供的服务了。 
    这是和本文所讨论的相同的场景。

免责声明:本段并非专门的Linux技术分析,仅用于介绍相关知识。

从守护进程的角度看Docker Compose

既然Docker Compose中按照顺序启动服务的情况和守护进程类似, 
那么能不能使用相同的机制呢?

从技术上没问题,如果所有的服务都采用和守护进程类似的机制,例如fork出子进程, 
或者使用&运行后台进程。总之,只要保证主进程退出, 
就可以被Docker Compose识别了。

这种方式看上去很美,不过就严格限制了服务的灵活性。 
猜测,仅仅是猜测,Docker没有找到尽善尽美的解决方案,所以没有提供这个功能。

Docker的解决方案

Docker没有提供直接的解决方案,不过也并且给出了一篇参考文章:

https://docs.docker.com/compose/startup-order/

这篇文章很神奇,神奇之处不是提出了什么奇思妙想, 
神奇之处在于点踩的人数远远多于点赞的人数, 
今天(2018年4月3日)的数据是172踩53赞。竟然达到了3倍之多! 
这种情况在官方文档中是非常罕见的。

不过我也要赞一下,不是赞这篇文章, 
而是为Docker公司把这么一个数字暴露出来的勇气点赞。

文章内容就不讨论了,思路类似。

思路

既然官方给出的思路是服务之间检测,那么检测端口是一个方法,而且是非侵入式的。 
不过当另一个服务并没有提供端口,例如仅仅是若干操作环节中的一个步骤, 
就不能检查端口了,此时就需要借助其他的方式。 
前置的服务设置一个标志,后续的服务检查这个标志。

从这个角度思考,借助文件系统检查时间戳可能是最简单的方法之一。

  • 借助volume在不同的服务之间共享卷
  • 前置服务完成启动的时候touch一个文件作为标记
  • 后置服务不停的检查这个文件

具体涉及到两个技术细节。

  • 由于服务会重用之前的卷,因此不能仅检查文件是否存在, 
    还需要比对文件的时间戳是否晚于当前服务的启动时间
  • 当前服务的启动时间不能使用uptime,这个是主机的启动时间, 
    而是比对 /proc/1/cmdline 这个文件的时间戳

代码

https://github.com/huzhenghui/Docker-Compose-Startup-Order

version: "3"
services:
  service_1:
    image: ubuntu
    volumes:
      - Docker-Compose-Startup-Order:/Startup-Order
    command: |
      bash -c '
      echo Service 1 Start;
      sleep 10;
      echo Service 1 Up;
      touch /Startup-Order/service_1;'
  service_2:
    image: ubuntu
    volumes:
      - Docker-Compose-Startup-Order:/Startup-Order
    command: |
      bash -c '
      while [[ ! -f /Startup-Order/service_1 || /Startup-Order/service_1 -ot /proc/1/cmdline ]]; do sleep 1; done;
      echo Service 2 Start;
      sleep 10;
      echo Service 2 Up;
      touch /Startup-Order/service_2;'
  service_3:
    image: ubuntu
    volumes:
      - Docker-Compose-Startup-Order:/Startup-Order
    command: |
      bash -c '
      while [[ ! -f /Startup-Order/service_2 || /Startup-Order/service_2 -ot /proc/1/cmdline ]]; do sleep 1; done;
      echo Service 3 Start;
      sleep 10;
      echo Service 3 Up;'
volumes:
  Docker-Compose-Startup-Order:

结果

PS C:\Users\huzh\OneDrive\Docker\Docker-Compose-Startup-Order> docker-compose up
Starting dockercomposestartuporder_service_1_1 ...
Starting dockercomposestartuporder_service_3_1 ...
Starting dockercomposestartuporder_service_3_www.feihuanyule.com1 ... done
Attaching to dockercomposestartuporder_service_1_1, dockercomposestartuporder_service_2_1, dockercomposestartuporder_service_3_1
service_1_1  | Service 1 Start
service_1_1  | Service 1 Up
dockercomposestartuporder_service_1_1 www.feihuanyule.com exited with code 0
service_2_1  | Service 2 Start
service_2_1  | Service 2 Up
dockercomposestartuporder_service_2_1 www.feihuanyule.com exited with code 0
service_3_1  | Service 3 Start
service_3_1  | Service 3 Up
dockercomposestartuporder_service_www.gouyiflb.cn 3_1 exited with code 0
PS C:\Users\huzh\OneDrive\Docker\Docker-Compose-Startup-Order>

控制Docker Compose的启动顺序的一个思路的更多相关文章

  1. docker compose 服务启动顺序控制

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

  2. asp.net core容器&mysql容器network互联 & docker compose方式编排启动多个容器

    文章简介 asp.net core webapi容器与Mysql容器互联(network方式) docker compose方式编排启动多个容器 asp.net core webapi容器与Mysql ...

  3. Docker Compose 启动mysql,redis,rabbitmq

    这里使用的centos7,首先切换到root. sudo -s 首先去设置下载镜像,否则下载这三个东西要很久,而且可能失败. vim /etc/docker/daemon.json 内容如下: { & ...

  4. Docker学习笔记之常用的 Docker Compose 配置项

    0x00 概述 与 Dockerfile 一样,编写 Docker Compose 的配置文件是掌握和使用好 Docker Compose 的前提.编写 Docker Compose 配置文件,其本质 ...

  5. docker-compose下的java应用启动顺序两部曲之二:实战

    上篇回顾 本文是<docker-compose下的java应用启动顺序两部曲>的终篇,在上一篇<docker-compose下的java应用启动顺序两部曲之一:问题分析>中,我 ...

  6. 5种常见的Docker Compose错误

    在构建一个容器化应用程序时,开发人员需要一种方法来引导他们正在使用的容器去测试其代码.虽然有几种方法可以做到这一点,但 Docker Compose 是最流行的选择之一.它让你可以轻松指定开发期间要引 ...

  7. Docker学习笔记之使用 Docker Compose 管理容器

    0x00 概述 通过之前的介绍,我们已经基本掌握了构建.运行容器的方法,但这还远远不够,由于 Docker 采用轻量级容器的设计,每个容器一般只运行一个软件,而目前绝大多数应用系统都绝不是一个软件所能 ...

  8. docker-compose下的java应用启动顺序两部曲之一:问题分析

    在docker-compose编排多个容器时,需要按实际情况控制各容器的启动顺序,本文是<docker-compose下的java应用启动顺序两部曲>的第一篇,文中会分析启动顺序的重要性, ...

  9. docker和docker compose安装使用、入门进阶案例

    一.前言 现在可谓是容器化的时代,云原生的袭来,导致go的崛起,作为一名java开发,现在慌得一批.作为知识储备,小编也是一直学关于docker的东西,还有一些持续继承jenkins. 提到docke ...

随机推荐

  1. oracle分区表按时间自动创建

    表分区是一种思想,分区表示一种技术实现.当表的大小过G的时候可以考虑进行表分区,提高查询效率,均衡IO.oracle分区表是oracle数据库提供的一种表分区的实现形式.表进行分区后,逻辑上仍然是一张 ...

  2. js 去掉下划线,后首个字母变大写

    1.驼峰转连字符: var s = "fooStyleCss";  s = s.replace(/([A-Z])/g,"-$1").toLowerCase(); ...

  3. 生鲜水果商城PC手机微信完整版源码2018版(免费)

    采用php+mysql架构,含有PC.手机.微信三端,只需要修改一下数据库配置,并恢复一下数据即可使用,还有微信.支付宝等接口,如有问题请在文章下面留言一下,我看到会协助一下的,下载包里面含有详细的安 ...

  4. DirectX11与DirectX12在古墓丽影暗影中的表现

    最近在关注这两个图形API,因为感兴趣,也算是初学者. 以下内容仅供参考. 使用古墓丽影暗影游戏,分别对这两个进行比较,得出的结论如下图(此笔记本散热很差,更改散热应该比下图结果好些): 首先看可以很 ...

  5. jupyter notebook 使用cmd命令窗口打开

    第一步:将文件路径改为你需要使用文件所在的路径 第二部:   jupyter notebook

  6. numpy切片和布尔型索引

    numpy 标签(空格分隔): numpy 数据挖掘 切片 数组切片是原始数组的视图.这意味着数据不会被复制,视图上的任何修改都会直接反映到源数组上 In [16]: arr Out[16]: arr ...

  7. UVa 1583 - Digit Generator 解题报告 - C语言

    1.题目大意 如果a加上a的各个数字之和得到b,则说a是b的生成元.给出n其中$1\le n\le 100000$,求其最小生成元,若没有解则输出0. 2.思路 使用打表的方法打出各个数字a对应的b, ...

  8. Linux内核设计笔记11——定时器

    定时器与时间管理笔记 内核中的时间 时钟中断:内核中的系统定时器以某种频率触发中断,该频率可以通过编程预定. 节拍率HZ:时钟中断的频率称为节拍率. 节拍:相邻两次中断的时间间隔称为节拍,1/节拍率. ...

  9. HADOOP docker(四):安装hive

    1.hive简介2.安装hive2.1 环境准备2.1.1 下载安装包2.1.2 设置hive用户的环境变量2.1.3 hive服务端配置文件2.1.4 hive客户端配置文件2.1.4 分发hive ...

  10. "Hello world!"团队第一次会议

    今天是我们"Hello world!"团队第一次召开会议,今天的会议可能没有那么正式,但是我们一起确立了选题——基于WEB的售票系统.博客内容是: 1.会议时间 2.会议成员 3. ...