mirror of
https://github.com/moby/moby.git
synced 2022-11-09 12:21:53 -05:00
Merge pull request #12058 from duglin/RemoveTable2
Remove engine.Table from docker history and docker rmi
This commit is contained in:
commit
326a5545d4
5 changed files with 76 additions and 57 deletions
|
@ -1,11 +1,12 @@
|
||||||
package client
|
package client
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"text/tabwriter"
|
"text/tabwriter"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/docker/docker/engine"
|
"github.com/docker/docker/api/types"
|
||||||
flag "github.com/docker/docker/pkg/mflag"
|
flag "github.com/docker/docker/pkg/mflag"
|
||||||
"github.com/docker/docker/pkg/stringid"
|
"github.com/docker/docker/pkg/stringid"
|
||||||
"github.com/docker/docker/pkg/units"
|
"github.com/docker/docker/pkg/units"
|
||||||
|
@ -20,16 +21,16 @@ func (cli *DockerCli) CmdHistory(args ...string) error {
|
||||||
quiet := cmd.Bool([]string{"q", "-quiet"}, false, "Only show numeric IDs")
|
quiet := cmd.Bool([]string{"q", "-quiet"}, false, "Only show numeric IDs")
|
||||||
noTrunc := cmd.Bool([]string{"#notrunc", "-no-trunc"}, false, "Don't truncate output")
|
noTrunc := cmd.Bool([]string{"#notrunc", "-no-trunc"}, false, "Don't truncate output")
|
||||||
cmd.Require(flag.Exact, 1)
|
cmd.Require(flag.Exact, 1)
|
||||||
|
|
||||||
cmd.ParseFlags(args, true)
|
cmd.ParseFlags(args, true)
|
||||||
|
|
||||||
body, _, err := readBody(cli.call("GET", "/images/"+cmd.Arg(0)+"/history", nil, nil))
|
rdr, _, err := cli.call("GET", "/images/"+cmd.Arg(0)+"/history", nil, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
outs := engine.NewTable("Created", 0)
|
history := []types.ImageHistory{}
|
||||||
if _, err := outs.ReadListFrom(body); err != nil {
|
err = json.NewDecoder(rdr).Decode(&history)
|
||||||
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -38,30 +39,23 @@ func (cli *DockerCli) CmdHistory(args ...string) error {
|
||||||
fmt.Fprintln(w, "IMAGE\tCREATED\tCREATED BY\tSIZE")
|
fmt.Fprintln(w, "IMAGE\tCREATED\tCREATED BY\tSIZE")
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, out := range outs.Data {
|
for _, entry := range history {
|
||||||
outID := out.Get("Id")
|
if *noTrunc {
|
||||||
if !*quiet {
|
fmt.Fprintf(w, entry.ID)
|
||||||
if *noTrunc {
|
|
||||||
fmt.Fprintf(w, "%s\t", outID)
|
|
||||||
} else {
|
|
||||||
fmt.Fprintf(w, "%s\t", stringid.TruncateID(outID))
|
|
||||||
}
|
|
||||||
|
|
||||||
fmt.Fprintf(w, "%s ago\t", units.HumanDuration(time.Now().UTC().Sub(time.Unix(out.GetInt64("Created"), 0))))
|
|
||||||
|
|
||||||
if *noTrunc {
|
|
||||||
fmt.Fprintf(w, "%s\t", out.Get("CreatedBy"))
|
|
||||||
} else {
|
|
||||||
fmt.Fprintf(w, "%s\t", utils.Trunc(out.Get("CreatedBy"), 45))
|
|
||||||
}
|
|
||||||
fmt.Fprintf(w, "%s\n", units.HumanSize(float64(out.GetInt64("Size"))))
|
|
||||||
} else {
|
} else {
|
||||||
if *noTrunc {
|
fmt.Fprintf(w, stringid.TruncateID(entry.ID))
|
||||||
fmt.Fprintln(w, outID)
|
|
||||||
} else {
|
|
||||||
fmt.Fprintln(w, stringid.TruncateID(outID))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
if !*quiet {
|
||||||
|
fmt.Fprintf(w, "\t%s ago\t", units.HumanDuration(time.Now().UTC().Sub(time.Unix(entry.Created, 0))))
|
||||||
|
|
||||||
|
if *noTrunc {
|
||||||
|
fmt.Fprintf(w, "%s\t", entry.CreatedBy)
|
||||||
|
} else {
|
||||||
|
fmt.Fprintf(w, "%s\t", utils.Trunc(entry.CreatedBy, 45))
|
||||||
|
}
|
||||||
|
fmt.Fprintf(w, "%s", units.HumanSize(float64(entry.Size)))
|
||||||
|
}
|
||||||
|
fmt.Fprintf(w, "\n")
|
||||||
}
|
}
|
||||||
w.Flush()
|
w.Flush()
|
||||||
return nil
|
return nil
|
||||||
|
|
|
@ -1,10 +1,11 @@
|
||||||
package client
|
package client
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
|
||||||
"github.com/docker/docker/engine"
|
"github.com/docker/docker/api/types"
|
||||||
flag "github.com/docker/docker/pkg/mflag"
|
flag "github.com/docker/docker/pkg/mflag"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -18,7 +19,6 @@ func (cli *DockerCli) CmdRmi(args ...string) error {
|
||||||
noprune = cmd.Bool([]string{"-no-prune"}, false, "Do not delete untagged parents")
|
noprune = cmd.Bool([]string{"-no-prune"}, false, "Do not delete untagged parents")
|
||||||
)
|
)
|
||||||
cmd.Require(flag.Min, 1)
|
cmd.Require(flag.Min, 1)
|
||||||
|
|
||||||
cmd.ParseFlags(args, true)
|
cmd.ParseFlags(args, true)
|
||||||
|
|
||||||
v := url.Values{}
|
v := url.Values{}
|
||||||
|
@ -31,22 +31,24 @@ func (cli *DockerCli) CmdRmi(args ...string) error {
|
||||||
|
|
||||||
var encounteredError error
|
var encounteredError error
|
||||||
for _, name := range cmd.Args() {
|
for _, name := range cmd.Args() {
|
||||||
body, _, err := readBody(cli.call("DELETE", "/images/"+name+"?"+v.Encode(), nil, nil))
|
rdr, _, err := cli.call("DELETE", "/images/"+name+"?"+v.Encode(), nil, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Fprintf(cli.err, "%s\n", err)
|
fmt.Fprintf(cli.err, "%s\n", err)
|
||||||
encounteredError = fmt.Errorf("Error: failed to remove one or more images")
|
encounteredError = fmt.Errorf("Error: failed to remove one or more images")
|
||||||
} else {
|
} else {
|
||||||
outs := engine.NewTable("Created", 0)
|
dels := []types.ImageDelete{}
|
||||||
if _, err := outs.ReadListFrom(body); err != nil {
|
err = json.NewDecoder(rdr).Decode(&dels)
|
||||||
|
if err != nil {
|
||||||
fmt.Fprintf(cli.err, "%s\n", err)
|
fmt.Fprintf(cli.err, "%s\n", err)
|
||||||
encounteredError = fmt.Errorf("Error: failed to remove one or more images")
|
encounteredError = fmt.Errorf("Error: failed to remove one or more images")
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
for _, out := range outs.Data {
|
|
||||||
if out.Get("Deleted") != "" {
|
for _, del := range dels {
|
||||||
fmt.Fprintf(cli.out, "Deleted: %s\n", out.Get("Deleted"))
|
if del.Deleted != "" {
|
||||||
|
fmt.Fprintf(cli.out, "Deleted: %s\n", del.Deleted)
|
||||||
} else {
|
} else {
|
||||||
fmt.Fprintf(cli.out, "Untagged: %s\n", out.Get("Untagged"))
|
fmt.Fprintf(cli.out, "Untagged: %s\n", del.Untagged)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -41,3 +41,18 @@ type ContainerChange struct {
|
||||||
Kind int
|
Kind int
|
||||||
Path string
|
Path string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GET "/images/{name:.*}/history"
|
||||||
|
type ImageHistory struct {
|
||||||
|
ID string `json:"Id"`
|
||||||
|
Created int64
|
||||||
|
CreatedBy string
|
||||||
|
Tags []string
|
||||||
|
Size int64
|
||||||
|
}
|
||||||
|
|
||||||
|
// DELETE "/images/{name:.*}"
|
||||||
|
type ImageDelete struct {
|
||||||
|
Untagged string `json:",omitempty"`
|
||||||
|
Deleted string `json:",omitempty"`
|
||||||
|
}
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
package daemon
|
package daemon
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
"github.com/docker/docker/engine"
|
"github.com/docker/docker/engine"
|
||||||
"github.com/docker/docker/graph"
|
"github.com/docker/docker/graph"
|
||||||
"github.com/docker/docker/image"
|
"github.com/docker/docker/image"
|
||||||
|
@ -16,21 +18,22 @@ func (daemon *Daemon) ImageDelete(job *engine.Job) error {
|
||||||
if n := len(job.Args); n != 1 {
|
if n := len(job.Args); n != 1 {
|
||||||
return fmt.Errorf("Usage: %s IMAGE", job.Name)
|
return fmt.Errorf("Usage: %s IMAGE", job.Name)
|
||||||
}
|
}
|
||||||
imgs := engine.NewTable("", 0)
|
|
||||||
if err := daemon.DeleteImage(job.Eng, job.Args[0], imgs, true, job.GetenvBool("force"), job.GetenvBool("noprune")); err != nil {
|
list := []types.ImageDelete{}
|
||||||
|
if err := daemon.DeleteImage(job.Eng, job.Args[0], &list, true, job.GetenvBool("force"), job.GetenvBool("noprune")); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if len(imgs.Data) == 0 {
|
if len(list) == 0 {
|
||||||
return fmt.Errorf("Conflict, %s wasn't deleted", job.Args[0])
|
return fmt.Errorf("Conflict, %s wasn't deleted", job.Args[0])
|
||||||
}
|
}
|
||||||
if _, err := imgs.WriteListTo(job.Stdout); err != nil {
|
if err := json.NewEncoder(job.Stdout).Encode(list); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: make this private and use the job instead
|
// FIXME: make this private and use the job instead
|
||||||
func (daemon *Daemon) DeleteImage(eng *engine.Engine, name string, imgs *engine.Table, first, force, noprune bool) error {
|
func (daemon *Daemon) DeleteImage(eng *engine.Engine, name string, list *[]types.ImageDelete, first, force, noprune bool) error {
|
||||||
var (
|
var (
|
||||||
repoName, tag string
|
repoName, tag string
|
||||||
tags = []string{}
|
tags = []string{}
|
||||||
|
@ -102,9 +105,9 @@ func (daemon *Daemon) DeleteImage(eng *engine.Engine, name string, imgs *engine.
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if tagDeleted {
|
if tagDeleted {
|
||||||
out := &engine.Env{}
|
*list = append(*list, types.ImageDelete{
|
||||||
out.Set("Untagged", utils.ImageReference(repoName, tag))
|
Untagged: utils.ImageReference(repoName, tag),
|
||||||
imgs.Add(out)
|
})
|
||||||
eng.Job("log", "untag", img.ID, "").Run()
|
eng.Job("log", "untag", img.ID, "").Run()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -117,12 +120,12 @@ func (daemon *Daemon) DeleteImage(eng *engine.Engine, name string, imgs *engine.
|
||||||
if err := daemon.Graph().Delete(img.ID); err != nil {
|
if err := daemon.Graph().Delete(img.ID); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
out := &engine.Env{}
|
*list = append(*list, types.ImageDelete{
|
||||||
out.SetJson("Deleted", img.ID)
|
Deleted: img.ID,
|
||||||
imgs.Add(out)
|
})
|
||||||
eng.Job("log", "delete", img.ID, "").Run()
|
eng.Job("log", "delete", img.ID, "").Run()
|
||||||
if img.Parent != "" && !noprune {
|
if img.Parent != "" && !noprune {
|
||||||
err := daemon.DeleteImage(eng, img.Parent, imgs, false, force, noprune)
|
err := daemon.DeleteImage(eng, img.Parent, list, false, force, noprune)
|
||||||
if first {
|
if first {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
package graph
|
package graph
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
"github.com/docker/docker/engine"
|
"github.com/docker/docker/engine"
|
||||||
"github.com/docker/docker/image"
|
"github.com/docker/docker/image"
|
||||||
"github.com/docker/docker/utils"
|
"github.com/docker/docker/utils"
|
||||||
|
@ -30,19 +32,22 @@ func (s *TagStore) CmdHistory(job *engine.Job) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
outs := engine.NewTable("Created", 0)
|
history := []types.ImageHistory{}
|
||||||
|
|
||||||
err = foundImage.WalkHistory(func(img *image.Image) error {
|
err = foundImage.WalkHistory(func(img *image.Image) error {
|
||||||
out := &engine.Env{}
|
history = append(history, types.ImageHistory{
|
||||||
out.SetJson("Id", img.ID)
|
ID: img.ID,
|
||||||
out.SetInt64("Created", img.Created.Unix())
|
Created: img.Created.Unix(),
|
||||||
out.Set("CreatedBy", strings.Join(img.ContainerConfig.Cmd, " "))
|
CreatedBy: strings.Join(img.ContainerConfig.Cmd, " "),
|
||||||
out.SetList("Tags", lookupMap[img.ID])
|
Tags: lookupMap[img.ID],
|
||||||
out.SetInt64("Size", img.Size)
|
Size: img.Size,
|
||||||
outs.Add(out)
|
})
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
if _, err := outs.WriteListTo(job.Stdout); err != nil {
|
|
||||||
|
if err = json.NewEncoder(job.Stdout).Encode(history); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue