[Docker Compose定义运行多个Docker容器]

Docker Compose

定义和运⾏多个 Docker 容器的应⽤(Defining and running multi-container Dockerapplications)

	我们知道使⽤⼀个 Dockerfile 模板⽂件,可以让⽤户很⽅便的定义⼀个单独的应⽤容器。然⽽,在⽇常⼯作
中,经常会碰到需要多个容器相互配合来完成某项任务的情况。例如要实现⼀个 Web 项⽬,除了 Web 服务容器本身,
往往还需要再加上后端的数据库服务容器,甚⾄还包括负载均衡容器等。

	Compose 恰好满⾜了这样的需求。它允许⽤户通过⼀个单独的 docker-compose.yml 模板⽂件(YAML格
式)来定义⼀组相关联的应⽤容器为⼀个项⽬(project)。

	docker-compose是容器编排工具,需要注意的是,docker-compose只是单机的容器编排工具。
	容器编排:管理多个容器

Compose 中有两个重要的概念

服务 ( service )

⼀个应⽤的容器,实际上可以包括若⼲运⾏相同镜像的容器实例。

docker 多个进程 多个docker compose_docker 多个进程

项目 ( project )

由⼀组关联的应⽤容器组成的⼀个完整业务单元,在 docker-compose.yml ⽂件中定义

Compose 的默认管理对象是项⽬,通过⼦命令对项⽬中的⼀组容器进⾏便捷地⽣命周期管理。

	Compose 项⽬由 Python 编写,实现上调⽤了 Docker 服务提供的 API 来对容器进⾏管理。因此,只要所操作的平台⽀持 Docker API,就可以在其上利⽤ Compose 来进⾏编排管理。

安装docker-compose

1、下载

curl -L https://get.daocloud.io/docker/compose/releases/download/1.24.0/docker-compose-`uname -s`-`uname -m` > /usr/local/bin/docker-compose

2、修改权限

chmod +x /usr/local/bin/docker-compose

3、文件更名

mv /usr/local/bin/docker-compose /usr/local/bin/docker-compose-Linux-x86_64

4、创建软连接

ln -s /usr/local/bin/docker-compose-Linux-x86_64 /usr/local/bin/docker-compose

5、下载bash命令补全

curl -L https://raw.githubusercontent.com/docker/compose/1.25.5/contrib/completion/bash/docker-compose > /etc/bash_completion.d/docker-compose

6、测试docker-compose是否安装成功

docker-compose --version

显示如下结果,表示成功

	# docker-compose version 1.24.0, build 0aa59064

卸载docker-compose

如果是⼆进制包⽅式安装的,删除⼆进制⽂件即可。

rm /usr/local/bin/docker-compose /usr/local/bin/docker-compose-Linux-x86_64

docker-compose的使用

模板⽂件是使⽤ Compose 的核⼼,涉及到的指令关键字也⽐较多。但⼤家不⽤担⼼,这⾥⾯⼤部分指令跟
docker run 相关参数的含义都是类似的。

	默认的模板⽂件名称为 docker-compose.yml ,格式为 YAML 格式。

docker-compose.yaml模板

参考⽂档:https://github.com/yeasy/docker_practice/blob/master/compose/compose_file.md

version: "3.0"
services:
  django:
    image: django:v3
    ports:
      - "8000:8000"
    container_name: django
    networks:
      - django
    depends_on:
      - mysql

  mysql:
    image: mysql:5.7
    environment:
      MYSQL_ROOT_PASSWORD: 123456
    container_name: mysql
    volumes:
    - "/root/docker-compose/mysql/data:/var/lib/mysql"
    networks:
      - django
    depends_on:
      - redis

  redis:
    image: redis:latest
    container_name: redis
    networks:
      - django

networks:
  django:

image :指定启动的镜像名称或ID

指定为要启动的镜像名称或镜像 ID。如果镜像在本地不存在, Compose 将会尝试拉取这个镜像(公网下载)。

build :构建镜像启动

每个服务都必须通过 image 指令指定镜像或 build 指令(需要 Dockerfile)等来⾃动构建⽣成镜像。
	
	如果使⽤ build 指令,在 Dockerfile 中设置的选项(例如: CMD , EXPOSE , VOLUME , ENV 等) 将
会⾃动被获取,⽆需在 docker-compose.yml 中重复设置。

	build:
	  context: ./dir
	  dockerfile: Dockerfile-alternate
	  args:
	    buildno: 1
            	
    context : 指令指定 Dockerfile 所在⽂件夹的路径。
    
    dockerfile :指令指定 Dockerfile ⽂件名。
    
    arg :指令指定构建镜像时的变量。

ports :暴露端口信息

使⽤宿主端⼝:容器端⼝格式,或者仅仅指定容器的端⼝(宿主将会随机选择端⼝)都可以。

ports:
 - "3000"
 - "8000:8000"
 - "49100:22"
 - "127.0.0.1:8001:8001"

expose : 暴露端口,但不映射到宿主机,只被连接的服务访问。仅可以指定内部端口为参数

expose:
 - "3000"
 - "8000"

command : 覆盖容器启动后默认执行的命令

设置容器启动命令

command: echo "hello world"

container_name : 指定容器名称

默认将会使⽤ 项⽬名称_服务名称_序号 这样的格式。

container_name: docker-web-container

depends_on : 解决容器的依赖、启动先后的问题

version: '3'

services:
  web:
    build: .
    depends_on:
      - db
      - redis
      
  redis:
    image: redis
    
  db:
    image: postgres
    
上面会先启动 redis db 再启动 web

注意: web 服务不会等待 redis db 「完全启动」之后才启动。

dns :自定义 DNS 服务器

可以是⼀个值,也可以是⼀个列表

dns: 8.8.8.8

dns:
 - 8.8.8.8
 - 114.114.114.114

environment : 设置环境变量

你可以使⽤数组或字典两种格式

environment:
  RACK_ENV: development
  SESSION_SECRET:
environment:
  - RACK_ENV=development
  - SESSION_SECRET

只给定名称的变量会⾃动获取运⾏ Compose 主机上对应变量的值,可以⽤来防⽌泄露不必要的数据。

如果变量名称或者值中⽤到 true|false,yes|no 等表达 布尔 含义的词汇,最好放到引号⾥,避免 YAML
⾃动解析某些内容为对应的布尔语义。这些特定词汇,包括

y|Y|yes|Yes|YES|n|N|no|No|NO|true|True|TRUE|false|False|FALSE|on|On|ON|off|Off|OFF

networks : 配置容器连接的⽹络

version: "3"

services:
  some-service:
  networks:  # 给启动的容器设置两个网桥
    - some-network
    - other-network
    
networks:      # 设置两个网桥
  some-network:
  other-network:

network_mode : 设置网络模式

使⽤和 docker run 的 --network 参数⼀样的值。

network_mode: "bridge"
network_mode: "host"
network_mode: "none"
network_mode: "service:[service name]"
network_mode: "container:[container name/id]"

env_file :从文件中获取环境变量

从⽂件中获取环境变量,可以为单独的⽂件路径或列表。

如果通过 docker-compose -f FILE ⽅式来指定 Compose 模板⽂件,则 env_file 中变量的路径会基于模板
⽂件路径。

如果有变量名称与 environment 指令冲突,则按照惯例,以后者为准。

env_file: .env

env_file:
  - ./common.env
  - ./apps/web.env
  - /opt/secrets.env
  
环境变量⽂件中每⼀⾏必须符合格式,⽀持 # 开头的注释⾏。

	# common.env: Set development environment
	PROG_ENV=development

volumes :数据卷所挂载路径设置

和 -v 差不多,不过-v是宿主主机映射到容器,两边同步。
而volumes直接就是两边数据同步。

healthcheck : 通过命令检查容器是否健康运行

healthcheck:

  test: ["CMD", "curl", "-f", "http://localhost"] # 执行的命令
  interval: 1m30s  # 间隔时间
  timeout: 10s     # 设置超时时间
  retries: 3       # 设置允许失败次数

  test: ["CMD", "ping", "-c1","-t1", "http://localhost"] # 执行的命令
  # -c1 : 只 ping 一次
  # -t1 :时间是一秒钟,超出1秒则超时

docker 多个进程 多个docker compose_docker 多个进程_02

docker-compose参数

启动:docker-compose up # 启动docker-compose.yaml文件里面的容器
	
删除:docker-compose down # 删除docker-compose.yaml文件里面的容器

重启:docker-compose restart [容器的名称...] # 重新启动其中的一个容器

进入容器:docker-compose exec [参数] [SERVICE] [CMD] # 默认使用-T(类似 -it )	
	
停止:docker-compose stop [SERVICE] # 停⽌已经处于运⾏状态的容器,但不删除它。通过 docker-compose start 可以再次启动这些容器查看各个服务容器内运⾏的进程

删除容器:docker-compose rm [SERVICE]  # 删除服务,停止之后才能删除

	-f, --force 强制直接删除,包括⾮停⽌状态的容器。⼀般尽量不要使⽤该选项。
	-v 删除容器所挂载的数据卷。

容器列表:docker-compose ps # 查看容器列表

启动:docker-compose start [SERVICE...] # 启动已经存在的服务容器

查看容器进程:docker-compose top  # 查看各个服务容器内运⾏的进程

暂停运行:docker-compose pause # 暂停服务容器

恢复运行:docker-compose unpause # 回复暂停状态的服务容器

可视化工具

version: '3.0'
services:
  portainer:
    image: portainer/portainer-ce
    ports:
    - "8001:8000"
    - "9001:9000"
    container_name: portainer
    networks:
      - "portainer"
    volumes:
    - "/var/run/docker.sock:/var/run/docker.sock"
    - "/root/docker-compose/portainer_data:/data"
    healthcheck:
      test: ["CMD", "ping", "-c1", "-t1", "portainer"]
      interval: 30s
      timeout: 10s
      retries: 3

networks:
  portainer: