title: 控制Docker Compose的启动顺序的一个思路
date: 2018-04-03 12:34:56
tags: [“Docker”]
categories: [“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. 服务的主进程结束
Created with Raphaël 2.2.0 服务的主进程启动 服务开始向外提供服务 服务向外提供服务 服务终止向外提供服务 服务的主进程结束
服务 客户端 主进程启动 服务开始向 外提供服务 请求 响应 服务终止向 外提供服务 主进程结束 服务 客户端 从使用服务的角度看Docker服务的生命周期

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

  1. entrypoint或者command进程启动
  2. 若干启动辅助进程启动到结束
  3. 服务的主进程启动
  4. 服务开始向外提供服务
  5. 服务向外提供服务
  6. 服务终止向外提供服务
  7. 服务的主进程结束
  8. 若干结束辅助进程启动到结束
  9. entrypoint或者command进程结束
Created with Raphaël 2.2.0 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_1 ... 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 exited with code 0
service_2_1  | Service 2 Start
service_2_1  | Service 2 Up
dockercomposestartuporder_service_2_1 exited with code 0
service_3_1  | Service 3 Start
service_3_1  | Service 3 Up
dockercomposestartuporder_service_3_1 exited with code 0
PS C:\Users\huzh\OneDrive\Docker\Docker-Compose-Startup-Order>
Logo

权威|前沿|技术|干货|国内首个API全生命周期开发者社区

更多推荐