|
@@ -19,7 +19,6 @@ import (
|
|
|
"github.com/docker/docker/archive"
|
|
|
"github.com/docker/docker/builder"
|
|
|
"github.com/docker/docker/engine"
|
|
|
- "github.com/docker/docker/graph"
|
|
|
"github.com/docker/docker/image"
|
|
|
"github.com/docker/docker/pkg/parsers"
|
|
|
"github.com/docker/docker/registry"
|
|
@@ -640,148 +639,6 @@ func (srv *Server) ImagePush(job *engine.Job) engine.Status {
|
|
|
return engine.StatusOK
|
|
|
}
|
|
|
|
|
|
-func (srv *Server) DeleteImage(name string, imgs *engine.Table, first, force, noprune bool) error {
|
|
|
- var (
|
|
|
- repoName, tag string
|
|
|
- tags = []string{}
|
|
|
- tagDeleted bool
|
|
|
- )
|
|
|
-
|
|
|
- repoName, tag = parsers.ParseRepositoryTag(name)
|
|
|
- if tag == "" {
|
|
|
- tag = graph.DEFAULTTAG
|
|
|
- }
|
|
|
-
|
|
|
- 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)
|
|
|
- }
|
|
|
-
|
|
|
- if strings.Contains(img.ID, name) {
|
|
|
- repoName = ""
|
|
|
- tag = ""
|
|
|
- }
|
|
|
-
|
|
|
- byParents, err := srv.daemon.Graph().ByParent()
|
|
|
- if err != nil {
|
|
|
- return err
|
|
|
- }
|
|
|
-
|
|
|
- //If delete by id, see if the id belong only to one repository
|
|
|
- if repoName == "" {
|
|
|
- for _, repoAndTag := range srv.daemon.Repositories().ByID()[img.ID] {
|
|
|
- parsedRepo, parsedTag := parsers.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
|
|
|
- }
|
|
|
-
|
|
|
- //Untag the current image
|
|
|
- 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, "")
|
|
|
- }
|
|
|
- }
|
|
|
- tags = srv.daemon.Repositories().ByID()[img.ID]
|
|
|
- if (len(tags) <= 1 && repoName == "") || len(tags) == 0 {
|
|
|
- if len(byParents[img.ID]) == 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("Deleted", img.ID)
|
|
|
- imgs.Add(out)
|
|
|
- srv.LogEvent("delete", img.ID, "")
|
|
|
- if img.Parent != "" && !noprune {
|
|
|
- err := srv.DeleteImage(img.Parent, imgs, false, force, noprune)
|
|
|
- if first {
|
|
|
- return err
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
- }
|
|
|
- return nil
|
|
|
-}
|
|
|
-
|
|
|
-func (srv *Server) ImageDelete(job *engine.Job) engine.Status {
|
|
|
- if n := len(job.Args); n != 1 {
|
|
|
- return job.Errorf("Usage: %s IMAGE", job.Name)
|
|
|
- }
|
|
|
- imgs := engine.NewTable("", 0)
|
|
|
- if err := srv.DeleteImage(job.Args[0], imgs, true, job.GetenvBool("force"), job.GetenvBool("noprune")); err != nil {
|
|
|
- return job.Error(err)
|
|
|
- }
|
|
|
- if len(imgs.Data) == 0 {
|
|
|
- return job.Errorf("Conflict, %s wasn't deleted", job.Args[0])
|
|
|
- }
|
|
|
- if _, err := imgs.WriteListTo(job.Stdout); err != nil {
|
|
|
- return job.Error(err)
|
|
|
- }
|
|
|
- return engine.StatusOK
|
|
|
-}
|
|
|
-
|
|
|
-func (srv *Server) canDeleteImage(imgID string, force, untagged bool) error {
|
|
|
- var message string
|
|
|
- if untagged {
|
|
|
- message = " (docker untagged the image)"
|
|
|
- }
|
|
|
- for _, container := range srv.daemon.List() {
|
|
|
- parent, err := srv.daemon.Repositories().LookupImage(container.Image)
|
|
|
- if err != nil {
|
|
|
- return err
|
|
|
- }
|
|
|
-
|
|
|
- if err := parent.WalkHistory(func(p *image.Image) error {
|
|
|
- if imgID == p.ID {
|
|
|
- if container.State.IsRunning() {
|
|
|
- if force {
|
|
|
- return fmt.Errorf("Conflict, cannot force delete %s because the running container %s is using it%s, stop it and retry", utils.TruncateID(imgID), utils.TruncateID(container.ID), message)
|
|
|
- }
|
|
|
- return fmt.Errorf("Conflict, cannot delete %s because the running container %s is using it%s, stop it and use -f to force", utils.TruncateID(imgID), utils.TruncateID(container.ID), message)
|
|
|
- } else if !force {
|
|
|
- return fmt.Errorf("Conflict, cannot delete %s because the container %s is using it%s, use -f to force", utils.TruncateID(imgID), utils.TruncateID(container.ID), message)
|
|
|
- }
|
|
|
- }
|
|
|
- return nil
|
|
|
- }); err != nil {
|
|
|
- return err
|
|
|
- }
|
|
|
- }
|
|
|
- return nil
|
|
|
-}
|
|
|
-
|
|
|
func (srv *Server) poolAdd(kind, key string) (chan struct{}, error) {
|
|
|
srv.Lock()
|
|
|
defer srv.Unlock()
|