sdoghds88888 发表于 2015-10-13 10:54:09

docker命令之rmi

1 man docker-rmi
  NAME

       docker-rmi - Remove one or more images.



SYNOPSIS

       docker rmi [-f|--force[=false] IMAGE



DESCRIPTION

       Thiswillremoveoneormoreimages from the host node.This does not remove images from a registry.You cannot

       remove an image of a running container unless you use the -f option.To see all images on a host use the docker images

       command.



OPTIONS
  -f, --force=true|false When set to true, force the removal of the image.The default is false.

2 代码分析

docker rmi
CmdRmi(api/client/command.go) ---> deleteImages(api/server/server.go) --->srv.ImageDelete (server/server.go)


CmdRmi和deleteImages省略,主要分析srv.ImageDelete部分
在srv.ImageDelete中调用server的方法srv.DeleteImage完成镜像的删除工作。


创建一张表,用于存储删除镜像的信息,将这些信息显示在标准输出上
imgs := engine.NewTable("", 0)


支持具体删除任务的方法
if err := srv.DeleteImage(job.Args, imgs, true, job.GetenvBool("force"), job.GetenvBool("noprune")); err != nil {
return job.Error(err)
}

将删除镜像的信息显示在标准输出上
if _, err := imgs.WriteListTo(job.Stdout); err != nil {
return job.Error(err)
}

srv.DeleteImage(server/server.go)分析


参数:
job.Args:删除镜像名
imgs:删除结果信息
true(first):
job.GetenvBool("force"):参数 -f|--force
job.GetenvBool("noprune"):参数


本地变量初始化
var (
repoName, tag string
tags          = []string{}
tagDeleted    bool
)


从镜像名name中解析出repo名和tag信息,如果name中没有tag信息,使用默认的latest tag
repoName, tag = utils.ParseRepositoryTag(name)
if tag == "" {
tag = graph.DEFAULTTAG
}

在graph中查找此name的镜像是否存在,如果没有查到,说明 此name的镜像不存在。
img, err := srv.daemon.Repositories().LookupImage(name)
if err != nil {
if r, _ := srv.daemon.Repositories().Get(repoName); r != nil {
return fmt.Errorf("No such image: %s:%s", repoName, tag)
}
return fmt.Errorf("No such image: %s", name)
}

如果镜像的ID中包含了name,说明是按镜像的ID删除,就不是用repoName和tag删除,所以将repoName和tag清空。
if strings.Contains(img.ID, name) {
repoName = ""
tag = ""
}



找到此镜像的所有父镜像
byParents, err := srv.daemon.Graph().ByParent()
if err != nil {
return err
}

如果是安装镜像ID删除,要对镜像的镜像树中的进行是否只属于此name的镜像做判断,如果镜像树中的镜像不只属于此镜像,
并且不是强制删除,报错。如果repoName不空,将此镜像所有tag加入到tags列表中。
if repoName == "" {
for _, repoAndTag := range srv.daemon.Repositories().ByID() {
parsedRepo, parsedTag := utils.ParseRepositoryTag(repoAndTag)
if repoName == "" || repoName == parsedRepo {
repoName = parsedRepo
if parsedTag != "" {
tags = append(tags, parsedTag)
}
} else if repoName != parsedRepo && !force {
// the id belongs to multiple repos, like base:latest and user:test,
// in that case return conflict
return fmt.Errorf("Conflict, cannot delete image %s because it is tagged in multiple repositories, use -f to force", name)
}
}
} else {
tags = append(tags, tag)
}

????????
if !first && len(tags) > 0 {
return nil
}

删除tags列表中镜像的tag
for _, tag := range tags {
tagDeleted, err = srv.daemon.Repositories().Delete(repoName, tag)
if err != nil {
return err
}
if tagDeleted {
out := &engine.Env{}
out.Set("Untagged", repoName+":"+tag)
imgs.Add(out)
srv.LogEvent("untag", img.ID, "")
}
}



如果有Container在使用镜像,则不能删除,否则可以删除。由于记录镜像信息分布在registry和graph中,需要在这两部分都做删除相应的信息。
如果镜像有父镜像,对父镜像做删除操作。
if (len(tags) <= 1 && repoName == &quot;&quot;) || len(tags) == 0 {
if len(byParents) == 0 {
if err := srv.canDeleteImage(img.ID, force, tagDeleted); err != nil {
return err
}
if err := srv.daemon.Repositories().DeleteAll(img.ID); err != nil {
return err
}
if err := srv.daemon.Graph().Delete(img.ID); err != nil {
return err
}
out := &engine.Env{}
out.Set(&quot;Deleted&quot;, img.ID)
imgs.Add(out)
srv.LogEvent(&quot;delete&quot;, img.ID, &quot;&quot;)
if img.Parent != &quot;&quot; && !noprune {
err := srv.DeleteImage(img.Parent, imgs, false, force, noprune)
if first {
return err
}
}
}
}


/var/lib/docker/
├── aufs                                       # Storage area for AUFS driver
│   ├── diff                                 # Branch directory of layer
│   ├── layers                                 # Infomation about docker layer
│   └── mnt                                    # Mount point of aufs, root of containers
├── containers                                 # Container configurations
│                                                (both LXC and Docker-specific)
├── graph                                    # Storage for the images
├── init
│   └── dockerinit-0.7.3                     # Used as /sbin/init in containers
├── linkgraph.db                               # SQLite database storing links
│                                                and names.
├── lxc-start-unconfined -> /usr/bin/lxc-start # When starting a privileged
│                                                container, this is used in
│                                                lieu of lxc-start, to evade
│                                                AppArmor confinement (which
│                                                matches by exact path).
├── repositories-aufs                        # repository infomation
└── volumes                                    # Storage for &quot;anonymous&quot; volumes
(those which are not bind-mounts)


























版权声明:本文为博主原创文章,未经博主允许不得转载。
页: [1]
查看完整版本: docker命令之rmi