如何在Docker中使用Docker

系统 Linux
如果 Agent 已经是容器化的,那么在 Agent 上如何构建镜像呢?这就是本篇将给出的回答,如何在 Docker 中使用 Docker。

[[354736]]

 1. 典型适用场景

在 CI 中,通常会有一个 CI Engine 负责解析流程,控制整个构建过程,而将真正的构建交给 Agent 去完成。例如,Jenkins 、GitLab 均是如此。

如下图, 连接 CI Engine 的 Agent, 种类很多。这是为了满足不同项目对构建环境的要求。

同时 Agent 是动态的,构建时才需要,构建完成时即销毁。CI 非常适合实践容器、Serverless 等技术,因此在生产过程中 Agent 经常是容器化的。

那么问题就来了?如果 CI Engine 也是容器化的,在容器中如何使用 Agent 容器去构建呢?如果 Agent 已经是容器化的,那么在 Agent 上如何构建镜像呢?这就是本篇将给出的回答,如何在 Docker 中使用 Docker。

2. 两种使用模式

我们需要知道 Docker 以 C/S 模式工作,主要分为两个部分,Docker CLI 和 Docker Daemon 。Docker CLI ,也就是客户端,提供给用户命令行操作 Docker,例如 docker create/images/ps 等。Docker Damon ,也就是守护进程,负责接受用户指令,维护容器的生命周期。

2.1 Docker in Docker

Docker in Docker ,以下简称 DinD 。

如上图,可以在 Container 中直接运行一个 Docker Daemon ,然后使用 Container 中的 Docker CLI 工具操作容器。

这种方式下,容器中的 Docker Daemon 完全独立于外部,具有良好的隔离特性。看起来,Container 类似一个 VM ,但 DinD 的作者自己也不是很推荐。

主要原因还是安全问题。DinD 需要以特权模式启动,这种嵌套会带来潜在的安全风险。

这种方式下,响应命令的容器嵌套于使用 docker 命令的容器。

2.2 Docker outside of Docker

Docker outside of Docker ,以下简称 DooD 。

如上图,Docker 以 C/S 模式工作,使用时用户关注的是 C 端,而生命周期的管理在 S 端。

因此,只需要将 Container 的外部 Docker Daemon 服务挂载到 Container 。让 Container 误以为本地运行了 Docker Daemon,使用 Docker CLI 命令操作时,外部的 Docker Daemon 会响应请求。

这种方式下,响应命令的容器与使用 docker 命令的容器处于同一层级。

3. Docker 环境下的演示

3.1 DinD

  •  运行 DinD 容器 
  1. $ docker run --privileged -e DOCKER_TLS_CERTDIR="" -d --name dockerd  docker:dind  
  2. d6414f2ff0076c42de19a8a1fe122481c1a72b3bd45fd490dbe1c427414b4139 
  •  运行带 CLI 的容器链接 DinD 容器 
  1. $ docker run --rm -it --link dockerd:docker docker:latest sh 
  •  在 DinD 容器中,拉取镜像 
  1. # 拉取镜像  
  2. $ docker pull shaowenchen/devops-java-sample  
  3. # 查看镜像  
  4. $ docker images  
  5. REPOSITORY                       TAG                 IMAGE ID            CREATED             SIZE  
  6. shaowenchen/devops-java-sample   latest              fa4651c24a18        6 weeks ago         122MB 

使用起来和一个独立的 Docker Daemon 环境一样。

  •  查看外部是否受影响

键入 exit 退出容器,通过主机上的 Docker Daemon 

  1. $ docker images |grep fa4651c24a18 

符合预期。DinD 使用的是独立的 Docker Daemon,对外部的实例没有直接影响。

3.2 DooD

  •  运行一个容器 
  1. $ docker run --rm -it -v /var/run/docker.sock:/var/run/docker.sock alpine sh 
  •  安装 curl

这里为了避免安装 Docker CLI ,直接使用 curl 调用 Docker Daemon 的 API。 

  1. $ apk update && apk add curl 
  •  拉取镜像 
  1. $ curl -XPOST --unix-socket /var/run/docker.sock http://localhost/images/create?fromImage=shaowenchen/docker-robotframework&tag=latest  
  2. ...  
  3. {"status":"Status: Downloaded newer image for shaowenchen/docker-robotframework"} 
  •  查看拉取的镜像

键入 exit 退出容器,通过主机上的 Docker Daemon 

  1. $ docker images |grep robotframework  
  2. shaowenchen/docker-robotframework                              latest                         d99cfa7ee716        12 months ago       1.5GB 

 符合预期。DooD 方式直接使用的外部 Docker Daemon。

4. Kubernetes 环境下的演示

4.1 DinD

  •  创建一个 dind.yaml 文件,内容如下: 
  1. apiVersion: apps/v1  
  2. kind: Deployment  
  3. metadata:  
  4.   name: dind  
  5. spec:  
  6.   replicas: 1  
  7.   selector:  
  8.     matchLabels:  
  9.       app: dind  
  10.   template:  
  11.     metadata:  
  12.       labels:  
  13.         app: dind  
  14.     spec:  
  15.       containers:  
  16.         - name: dockerd  
  17.           image: 'docker:dind'  
  18.           env:  
  19.             - name: DOCKER_TLS_CERTDIR  
  20.               value: ""  
  21.           securityContext:  
  22.             privileged: true  
  23.         - name: docker-cli  
  24.           image: 'docker:latest'  
  25.           env:  
  26.           - name: DOCKER_HOST  
  27.             value: 127.0.0.1  
  28.           command: ["/bin/sh"]  
  29.           args: ["-c", "sleep 86400;"] 
  •     创建 Deployment 
  1. $ kubectl apply -f dind.yaml 
  •  查看创建的 Pod 名 
  1. $ kubectl get pod |grep dind  
  2. dind-5446ffbc8d-68q28   2/2     Running       0          12s 
  • 进入 Pod 
  1. $ kubectl exec -it dind-5446ffbc8d-68q28  -c docker-cli sh 
  •  测试是否使用独立的 Docker Daemon 
  1. $ docker pull nginx  
  2. $ docker images  
  3. REPOSITORY          TAG                 IMAGE ID            CREATED             SIZE  
  4. nginx               latest              daee903b4e43        3 days ago          133MB 

符合预期,这里仅显示了刚拉取的 Nginx 的镜像,完全独立于主机的 Docker Daemon。

4.2 DooD

  •  创建一个 dood.yaml 文件,内容如下: 
  1. apiVersion: apps/v1  
  2. kind: Deployment 
  3. metadata:  
  4.   name: dood  
  5. spec:  
  6.   replicas: 1  
  7.   selector:  
  8.     matchLabels:  
  9.       app: dood  
  10.   template:  
  11.     metadata:  
  12.       labels:  
  13.         app: dood  
  14.     spec:  
  15.       containers:  
  16.         - image: docker:latest  
  17.           name: docker-cli  
  18.           securityContext:  
  19.             privileged: false  
  20.           command: ["/bin/sh"]  
  21.           args: ["-c", "sleep 86400;"]  
  22.           volumeMounts:  
  23.           - mountPath: /var/run/docker.sock  
  24.             name: volume-docker  
  25.       volumes:  
  26.         - hostPath:  
  27.             path: /var/run/docker.sock  
  28.             type: ""  
  29.           name: volume-docker 
  •  创建 Deployment 
  1. $ kubectl apply -f dood.yaml 
  •  查看创建的 Pod 名 
  1. $ kubectl get pod  |grep dood  
  2. dood-667d8bcfc6-d5fzf   1/1     Running   0          15s 
  •  进入 Pod 
  1. $ kubectl exec -it dood-667d8bcfc6-d5fzf  -c docker-cli sh 
  •  测试是否使用的是主机的 Docker Daemon 
  1. $ docker images |wc  
  2. 69       482      8509 

符合预期,这里 Docker 命令使用的就是外部的 Docker Daemon。

5. 参考

  1.  https://medium.com/better-programming/about-var-run-docker-sock-3bfd276e12fd
  2.  https://github.com/jpetazzo/dind 

 

责任编辑:庞桂玉 来源: 奇妙的Linux世界
相关推荐

2015-08-31 13:42:06

IDEDockerdoclipser

2015-11-26 10:57:56

DockerOpen vSwitc

2017-03-07 09:17:51

AtomicDocker远程

2016-11-03 20:06:53

UbuntuGrafanaDocker

2022-06-10 10:01:17

MacDockerLinux

2019-09-16 19:00:48

Linux变量

2014-07-02 09:47:06

SwiftCocoaPods

2020-04-09 10:18:51

Bash循环Linux

2024-09-06 11:34:15

RustAI语言

2015-08-27 09:46:09

swiftAFNetworkin

2021-03-09 07:27:40

Kafka开源分布式

2022-06-23 08:00:53

PythonDateTime模块

2021-06-09 09:36:18

DjangoElasticSearLinux

2022-05-17 08:25:10

TypeScript接口前端

2011-08-10 09:31:41

Hibernateunion

2024-01-18 08:37:33

socketasyncio线程

2018-03-09 10:07:34

mysql

2015-07-08 11:28:53

云服务平台Docker MachDocker

2018-06-26 09:15:24

Linux命令history

2018-05-16 10:32:06

Linux命令find
点赞
收藏

51CTO技术栈公众号