containerd的create container的API如下所示:

type CreateContainerRequest struct {

  Id       string
  BundlePath  string
  Checkpoint   string
  Stdin     string
  Stdout     string
  Stderr     string
  Labels    []string
  NoPivotRoot  bool
  Runtime    string
  RuntimeArgs []string
  CheckpointDir string
}

  

StartTask结构如下所示:

type StartTask struct {

  baseTask

  ID      string
  BundlePath  string
  Stdout     string
  Stderr     string
  Stdin     string
  StartResponse chan StartResponse
  Labels     []string
  NoPivotRoot  bool
  Checkpoint   *runtime.Checkpoint
  CheckpointDir  string
  Runtime     string
  RuntimeArgs   []string
}

  

1、containerd/api/grpc/server/server.go

func (s *apiServer) CreateContainer(ctx context.Context, c *types.CreateContainerRequest) (*types.CreateContainerResponse, error)

(1)、根据c填充获得e := &supervisor.StartTask{},并调用s.sv.SendTask(e)

(2)、调用 r := <-e.StartResponse,再调用apiC ,err := createAPIContainer(r.Container, false)获取创建的容器实例

(3)、return &types.CreateContainerResponse{Container: apiC,}

2、containerd/supervisor/create.go

func (s *Supervisor) start(t *StartTask) error

(1)、根据t的内容创建容器,runtime.New主要根据ContainerOpts填充结构container获得容器实例,再将创建state并写入状态文件中。

container, err := runtime.New(runtime.ContainerOpts{

  Root:    s.stateDir,
  ID:     t.ID,
  Bundle:   t.BundlePath,
  Runtime:   rt,
  RuntimeArgs: rtArgs,   Shim:    s.shim,   Labels:    t.Labels,
  NoPivotRoot: t.NoPivotRoot,
  Timeout:   s.timeout
})

  

(2)、注册新增加的容器,调用s.containers[t.ID] = &containerInfo{container: container,}

(3)、根据新获得的container实例和t的内容,填充获得startTask,再调用s.startTask <- task 交由worker处理

task := &startTask{

  Err:      t.ErrorCh(),
  Container:   container,
  StartResponse: t.StartResponse,
  Stdin:      t.Stdin,
  Stdout:     t.Stdout,
  Stderr:     t.Stderr
}

  

3、containerd/supervisor/worker.go

// Start runs a loop in charge of starting new containers

func (w *Worker) Start()

containerd在刚创建时,就启动了10个goroutine,用于处理startTasks,这里的Start函数就是从startTasks这个channel中获取任务,并处理

(1)、调用process, err := t.Container.Start(t.checkpointPath, runtime.NewStdio(t.Stdin, t.Stdout, t.Stderr))启动容器,NewStdio()仅仅只是将参数封装到一个统一的Stdio结构中

(2)、分别调用w.s.monitor.MonitorOOM(t.Container),w.s.monitorProcess(process)对容器进行监控,调用t.StartResponse <- StartResponse{Container: t.Container}返回创建成功的容器实例

4、containerd/runtime/container.go

func (c *container) Start(checkpointPath string, s Stdio) (Process, error)

(1)、创建processRoot := filepath.Join(c.root, c.id, InitProcessID)目录

(2)、容器创建命令 cmd := exec.Command(c.shim, c.id, c.bundle, c.runtime),配置cmd目录为processRoot,spec, err := c.readSpec()

(3)、生成config如下:

config := &processConfig {

  checkpoint:    checkpointPath,
  root:       processRoot,
  id:        InitProcessID,
  c:         c,
  stdio:       s,
  spec:       spec,
  processSpec:   specs.ProcessSpec(spec.Process), }

  

(4)、根据config,调用p, err := newProcess(config),生成process实例

(5)、最后调用c.createCmd(InitProcessID, cmd, p),并返回 return p, nil

5、containerd/runtime/process.go

func newProcess(config *processConfig) (*process, error)

(1)、根据config生成process实例,p := &process{root: config.root, id: config.id, container: config.c, ....., cmdDoneCh: make(chan struct{}), state: Running,}

(2)、创建状态文件os.Create(filepath.Join(config.root, "process.config")),生成ps := ProcessState{...},并将ps的内容写入状态文件中

(3)、调用exit, err := getExitPipe(filepath.Join(config.root, ExitFile))和control ,err := getControlPipe(filepath.Join(config.root, ControlFile))生成两个pipe文件,并分别将exit和control赋值给p.exitPipe和p.controlPipe,最后 return p

6、containerd/runtime/process.go

getExitPipe和getControlPipe生成两个FIFO文件,其中Exit函数中的FIFO是只读的,而Control函数中的FIFO是读写的

7、containerd/runtime/container.go

func (c *container) createCmd(pid string, cmd *exec.Cmd, p *process)

(1)、p.cmd = cmd, 再调用 cmd.Start()

(2)、构建defer函数,其中生成一个goroutine,其中调用p.cmd.Wait(),再调用same, err := p.isSameProcess(),如果same为true并且p.pid > 0则再进行一些处理

(3)、调用c.waitForCreate(p, cmd),c.processes[pid] = p

8、containerd/runtime/container.go

func (c *container) waitForCreate(p *process, cmd *exec.Cmd) error

(1)、该函数先启动一个goroutine用于从pidfile中读取pid

(2)、select,从(1)中的goroutine接收结果,或者超时,当超时时,调用cmd.Process.Kill()和cmd.Wait()

docker containerd 中的create 容器操作的更多相关文章

  1. JAVA中的集合容器操作类

    目录 JAVA中的集合容器操作类 List集合 ArrayList的操作方法说明 LinkedList Stack Set Map Queue 总结 JAVA中的集合容器操作类 Java容器类库总共分 ...

  2. docker学习笔记2:容器操作

    一.列出主机上已经创建的容器 docker ps -a 二.创建交互式容器 命令: docker run -i -t ubuntu /bin/bash 其中-i -t 表示创建一个提供交互式shell ...

  3. docker containerd中的容器操作

    containerd的中的各种操作都是通过Task来进行的,因此对于容器的create, start, delete等等操作其实都是一个个的Task而已. Task的数据结构如下所示: type Ta ...

  4. 实例解析Docker数据卷+数据卷容器+flocker数据共享+DockerHub操作

    Docker内部数据管理和Docker之间的数据共享为数据卷和数据卷容器,实例解析1.将本地的文件作为容器的数据卷,2.数据卷flocker插件实现容器集群(或者Docker Swarm)的数据共享3 ...

  5. docker的安装和基本的docker命令、镜像和容器的操作

    1.yum 包更新到最新 yum update 2.安装需要的软件包, yum-util 提供yum-config-manager功能,另外两个是devicemapper驱动依赖的 yum insta ...

  6. centos7下安装docker(11容器操作总结)

    这段时间主要是学习了对容器的操作,包括:容器的状态:start,stop,restart,rename,pause,unpause,rm,attach,exec,kill,logs:还学习了对容器的资 ...

  7. 【Docker】容器操作(转)

    来自:https://www.cnblogs.com/zydev/p/5803461.html 列出主机上的容器 列出正在运行的容器:   docker ps 列出所有容器:  docker ps - ...

  8. docker 容器操作( 以 tomcat 为例 )

    一.容器操作 一个镜像可以启动多个容器.比如一个 tomcat 镜像,可以启动多个tomcat 容器,启动后的这些 tomcat 都是各自独立的 二.步骤 1.搜索镜像 [root@localhost ...

  9. Docker.[4].容器操作.

    Docker.[4].容器操作. 熟悉的指令: 启动容器 docker run 交互式启动容器 docker run -i -t IMAGENAME /bin/bash 停止容器 exit 停止容器 ...

随机推荐

  1. gene框架文档 - 概述

    欢迎使用Gene框架 最新版本:V1.2.2 开源地址:https://github.com/sasou/php-gene 作者:sasou 文档地址:http://php-gene.com/doc ...

  2. php学习笔记:对文件的增删查改等操作

    文件的创建: 采用touch()函数,当文件不存在会被创建 例如: <?php header("Content-type: text/html; charset=utf-8" ...

  3. Google OKR 目标管理体系学习

    OKR 全称是「目标和关键成果」(Objectives and Key Results).它是Google在公司创立不足一年的时候,从Intel公司引入的目标管理系统,也常被认为是一套组织测评系统. ...

  4. .net经验积累

    希望对.net编程者有所帮助 1.学会配置环境变量  1.我的电脑-属性-环境变量-双击下面的path-粘贴路径  2.ctrl+r 输入软件名字按回车 2.常用vs2010快捷键  代码格式化:ct ...

  5. 轻量级SaaS在线作图工具ProcessOn

    俗话说“一图胜千言”,在办公应用领域,流程图是一个非常好的表现企业业务流程或工作岗位规范等内容的展现形式,比如去给客户做调研,回来后都要描述出客户的关键业务流程,谁.什么时候.在什么地方.负责什么事情 ...

  6. 用css伪类实现提示框效果

    题目要求用css实现下图效果: 很明显难点就在那个多出去的三角形上,下面代码是用一个div来实现的,用到了伪类 : befor和 : after,使用这两个伪类活生生的在div之前和之后多出了&quo ...

  7. javascript宿主对象之window.screen、window.close()/open()、window.moveTo、window.resizeTo

    window.screen属性所提供的是浏览器以外的信息.这里只简单的概述一下: screen.availWidth - 可用的屏幕宽度 (除去操作系统菜单) screen.availHeight - ...

  8. 使用WebMatrix发布网站

    使用WebMatrix发布网站 WebMatrix 简介: Microsoft WebMatrix 是微软最新的 Web 开发工具,它包含了构建网站所需要的一切元素.您可以从开源 Web 项目或者内置 ...

  9. Hosts文件小结

    今天又遇到Hosts,小结一下: Hosts文件是什么? Hosts文件从表象上来看是一个没有扩展名的系统文件.其基本作用就是将一些常用的网址域名与其对应的IP地址建立一个关联"数据库&qu ...

  10. Q:解决每天第一次打开MSCRM系统展示慢的问题

    问题:第天第一次打开系统时,需要加载很长时间,基本为1分多钟,而第二次打开只需5秒. 解决方案:利用IIS中的Session. 一.打开IIS,选择打开服务器功能中“Session State”. 二 ...