k3s/vendor/github.com/kubernetes-sigs/cri-tools/cmd/crictl/image.go
2020-08-28 17:18:31 -07:00

612 lines
16 KiB
Go

/*
Copyright 2017 The Kubernetes Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package crictl
import (
"fmt"
"sort"
"strings"
"github.com/docker/go-units"
"github.com/pkg/errors"
"github.com/sirupsen/logrus"
"github.com/urfave/cli/v2"
"golang.org/x/net/context"
pb "k8s.io/cri-api/pkg/apis/runtime/v1alpha2"
)
type imageByRef []*pb.Image
func (a imageByRef) Len() int { return len(a) }
func (a imageByRef) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
func (a imageByRef) Less(i, j int) bool {
if len(a[i].RepoTags) > 0 && len(a[j].RepoTags) > 0 {
return a[i].RepoTags[0] < a[j].RepoTags[0]
}
if len(a[i].RepoDigests) > 0 && len(a[j].RepoDigests) > 0 {
return a[i].RepoDigests[0] < a[j].RepoDigests[0]
}
return a[i].Id < a[j].Id
}
var pullImageCommand = &cli.Command{
Name: "pull",
Usage: "Pull an image from a registry",
UseShortOptionHandling: true,
Flags: []cli.Flag{
&cli.StringFlag{
Name: "creds",
Value: "",
Usage: "Use `USERNAME[:PASSWORD]` for accessing the registry",
},
&cli.StringFlag{
Name: "auth",
Value: "",
Usage: "Use `AUTH_STRING` for accessing the registry. AUTH_STRING is a base64 encoded 'USERNAME[:PASSWORD]'",
},
&cli.StringFlag{
Name: "pod-config",
Value: "",
Usage: "Use `pod-config.[json|yaml]` to override the the pull context",
TakesFile: true,
},
},
ArgsUsage: "NAME[:TAG|@DIGEST]",
Action: func(context *cli.Context) error {
imageName := context.Args().First()
if imageName == "" {
return cli.ShowSubcommandHelp(context)
}
imageClient, conn, err := getImageClient(context)
if err != nil {
return err
}
defer closeConnection(context, conn)
auth, err := getAuth(context.String("creds"), context.String("auth"))
if err != nil {
return err
}
var sandbox *pb.PodSandboxConfig
if context.IsSet("pod-config") {
sandbox, err = loadPodSandboxConfig(context.String("pod-config"))
if err != nil {
return errors.Wrap(err, "load podSandboxConfig")
}
}
r, err := PullImageWithSandbox(imageClient, imageName, auth, sandbox)
if err != nil {
return errors.Wrap(err, "pulling image")
}
fmt.Printf("Image is up to date for %s\n", r.ImageRef)
return nil
},
}
var listImageCommand = &cli.Command{
Name: "images",
Aliases: []string{"image", "img"},
Usage: "List images",
ArgsUsage: "[REPOSITORY[:TAG]]",
UseShortOptionHandling: true,
Flags: []cli.Flag{
&cli.BoolFlag{
Name: "verbose",
Aliases: []string{"v"},
Usage: "Show verbose info for images",
},
&cli.BoolFlag{
Name: "quiet",
Aliases: []string{"q"},
Usage: "Only show image IDs",
},
&cli.StringFlag{
Name: "output",
Aliases: []string{"o"},
Usage: "Output format, One of: json|yaml|table",
},
&cli.BoolFlag{
Name: "digests",
Usage: "Show digests",
},
&cli.BoolFlag{
Name: "no-trunc",
Usage: "Show output without truncating the ID",
},
},
Action: func(context *cli.Context) error {
imageClient, conn, err := getImageClient(context)
if err != nil {
return err
}
defer closeConnection(context, conn)
r, err := ListImages(imageClient, context.Args().First())
if err != nil {
return errors.Wrap(err, "listing images")
}
sort.Sort(imageByRef(r.Images))
switch context.String("output") {
case "json":
return outputProtobufObjAsJSON(r)
case "yaml":
return outputProtobufObjAsYAML(r)
}
// output in table format by default.
display := newTableDisplay(20, 1, 3, ' ', 0)
verbose := context.Bool("verbose")
showDigest := context.Bool("digests")
quiet := context.Bool("quiet")
noTrunc := context.Bool("no-trunc")
if !verbose && !quiet {
if showDigest {
display.AddRow([]string{columnImage, columnTag, columnDigest, columnImageID, columnSize})
} else {
display.AddRow([]string{columnImage, columnTag, columnImageID, columnSize})
}
}
for _, image := range r.Images {
if quiet {
fmt.Printf("%s\n", image.Id)
continue
}
if !verbose {
imageName, repoDigest := normalizeRepoDigest(image.RepoDigests)
repoTagPairs := normalizeRepoTagPair(image.RepoTags, imageName)
size := units.HumanSizeWithPrecision(float64(image.GetSize_()), 3)
id := image.Id
if !noTrunc {
id = getTruncatedID(id, "sha256:")
repoDigest = getTruncatedID(repoDigest, "sha256:")
}
for _, repoTagPair := range repoTagPairs {
if showDigest {
display.AddRow([]string{repoTagPair[0], repoTagPair[1], repoDigest, id, size})
} else {
display.AddRow([]string{repoTagPair[0], repoTagPair[1], id, size})
}
}
continue
}
fmt.Printf("ID: %s\n", image.Id)
for _, tag := range image.RepoTags {
fmt.Printf("RepoTags: %s\n", tag)
}
for _, digest := range image.RepoDigests {
fmt.Printf("RepoDigests: %s\n", digest)
}
if image.Size_ != 0 {
fmt.Printf("Size: %d\n", image.Size_)
}
if image.Uid != nil {
fmt.Printf("Uid: %v\n", image.Uid)
}
if image.Username != "" {
fmt.Printf("Username: %v\n", image.Username)
}
fmt.Printf("\n")
}
display.Flush()
return nil
},
}
var imageStatusCommand = &cli.Command{
Name: "inspecti",
Usage: "Return the status of one or more images",
ArgsUsage: "IMAGE-ID [IMAGE-ID...]",
UseShortOptionHandling: true,
Flags: []cli.Flag{
&cli.StringFlag{
Name: "output",
Aliases: []string{"o"},
Usage: "Output format, One of: json|yaml|go-template|table",
},
&cli.BoolFlag{
Name: "quiet",
Aliases: []string{"q"},
Usage: "Do not show verbose information",
},
&cli.StringFlag{
Name: "template",
Usage: "The template string is only used when output is go-template; The Template format is golang template",
},
},
Action: func(context *cli.Context) error {
if context.NArg() == 0 {
return cli.ShowSubcommandHelp(context)
}
imageClient, conn, err := getImageClient(context)
if err != nil {
return err
}
defer closeConnection(context, conn)
verbose := !(context.Bool("quiet"))
output := context.String("output")
if output == "" { // default to json output
output = "json"
}
tmplStr := context.String("template")
for i := 0; i < context.NArg(); i++ {
id := context.Args().Get(i)
r, err := ImageStatus(imageClient, id, verbose)
if err != nil {
return errors.Wrapf(err, "image status for %q request", id)
}
image := r.Image
if image == nil {
return fmt.Errorf("no such image %q present", id)
}
status, err := protobufObjectToJSON(r.Image)
if err != nil {
return errors.Wrapf(err, "marshal status to json for %q", id)
}
switch output {
case "json", "yaml", "go-template":
if err := outputStatusInfo(status, r.Info, output, tmplStr); err != nil {
return errors.Wrapf(err, "output status for %q", id)
}
continue
case "table": // table output is after this switch block
default:
return fmt.Errorf("output option cannot be %s", output)
}
// otherwise output in table format
fmt.Printf("ID: %s\n", image.Id)
for _, tag := range image.RepoTags {
fmt.Printf("Tag: %s\n", tag)
}
for _, digest := range image.RepoDigests {
fmt.Printf("Digest: %s\n", digest)
}
size := units.HumanSizeWithPrecision(float64(image.GetSize_()), 3)
fmt.Printf("Size: %s\n", size)
if verbose {
fmt.Printf("Info: %v\n", r.GetInfo())
}
}
return nil
},
}
var removeImageCommand = &cli.Command{
Name: "rmi",
Usage: "Remove one or more images",
ArgsUsage: "IMAGE-ID [IMAGE-ID...]",
UseShortOptionHandling: true,
Flags: []cli.Flag{
&cli.BoolFlag{
Name: "all",
Aliases: []string{"a"},
Usage: "Remove all images",
},
&cli.BoolFlag{
Name: "prune",
Aliases: []string{"q"},
Usage: "Remove all unused images",
},
},
Action: func(ctx *cli.Context) error {
imageClient, conn, err := getImageClient(ctx)
if err != nil {
return err
}
defer closeConnection(ctx, conn)
ids := map[string]bool{}
for _, id := range ctx.Args().Slice() {
logrus.Debugf("User specified image to be removed: %v", id)
ids[id] = true
}
all := ctx.Bool("all")
prune := ctx.Bool("prune")
// Add all available images to the ID selector
if all || prune {
r, err := imageClient.ListImages(context.Background(),
&pb.ListImagesRequest{})
if err != nil {
return err
}
for _, img := range r.GetImages() {
logrus.Debugf("Adding image to be removed: %v", img.GetId())
ids[img.GetId()] = true
}
}
// On prune, remove images which are in use from the ID selector
if prune {
runtimeClient, conn, err := getRuntimeClient(ctx)
if err != nil {
return err
}
defer closeConnection(ctx, conn)
// Container images
c, err := runtimeClient.ListContainers(
context.Background(), &pb.ListContainersRequest{},
)
if err != nil {
return err
}
for _, container := range c.GetContainers() {
img := container.GetImage().Image
imageStatus, err := ImageStatus(imageClient, img, false)
if err != nil {
logrus.Errorf(
"image status request for %q failed: %v",
img, err,
)
continue
}
id := imageStatus.GetImage().GetId()
logrus.Debugf("Excluding in use container image: %v", id)
ids[id] = false
}
}
if len(ids) == 0 {
logrus.Info("No images to remove")
return nil
}
errored := false
for id, remove := range ids {
if !remove {
continue
}
status, err := ImageStatus(imageClient, id, false)
if err != nil {
logrus.Errorf("image status request for %q failed: %v", id, err)
errored = true
continue
}
if status.Image == nil {
logrus.Errorf("no such image %s", id)
errored = true
continue
}
_, err = RemoveImage(imageClient, id)
if err != nil {
// We ignore further errors on prune because there might be
// races
if !prune {
logrus.Errorf("error of removing image %q: %v", id, err)
errored = true
}
continue
}
for _, repoTag := range status.Image.RepoTags {
fmt.Printf("Deleted: %s\n", repoTag)
}
}
if errored {
return fmt.Errorf("unable to remove the image(s)")
}
return nil
},
}
var imageFsInfoCommand = &cli.Command{
Name: "imagefsinfo",
Usage: "Return image filesystem info",
UseShortOptionHandling: true,
Flags: []cli.Flag{
&cli.StringFlag{
Name: "output",
Aliases: []string{"o"},
Usage: "Output format, One of: json|yaml|go-template|table",
},
&cli.StringFlag{
Name: "template",
Usage: "The template string is only used when output is go-template; The Template format is golang template",
},
},
Action: func(context *cli.Context) error {
imageClient, conn, err := getImageClient(context)
if err != nil {
return err
}
defer closeConnection(context, conn)
output := context.String("output")
if output == "" { // default to json output
output = "json"
}
tmplStr := context.String("template")
r, err := ImageFsInfo(imageClient)
if err != nil {
return errors.Wrap(err, "image filesystem info request")
}
for _, info := range r.ImageFilesystems {
status, err := protobufObjectToJSON(info)
if err != nil {
return errors.Wrap(err, "marshal image filesystem info to json")
}
switch output {
case "json", "yaml", "go-template":
if err := outputStatusInfo(status, nil, output, tmplStr); err != nil {
return errors.Wrap(err, "output image filesystem info")
}
continue
case "table": // table output is after this switch block
default:
return fmt.Errorf("output option cannot be %s", output)
}
// otherwise output in table format
fmt.Printf("TimeStamp: %d\n", info.Timestamp)
fmt.Printf("UsedBytes: %s\n", info.UsedBytes)
fmt.Printf("Mountpoint: %s\n", info.FsId.Mountpoint)
}
return nil
},
}
func parseCreds(creds string) (string, string, error) {
if creds == "" {
return "", "", errors.New("credentials can't be empty")
}
up := strings.SplitN(creds, ":", 2)
if len(up) == 1 {
return up[0], "", nil
}
if up[0] == "" {
return "", "", errors.New("username can't be empty")
}
return up[0], up[1], nil
}
func getAuth(creds string, auth string) (*pb.AuthConfig, error) {
if creds != "" && auth != "" {
return nil, errors.New("both `--creds` and `--auth` are specified")
}
if creds != "" {
username, password, err := parseCreds(creds)
if err != nil {
return nil, err
}
return &pb.AuthConfig{
Username: username,
Password: password,
}, nil
}
if auth != "" {
return &pb.AuthConfig{
Auth: auth,
}, nil
}
return nil, nil
}
// Ideally repo tag should always be image:tag.
// The repoTags is nil when pulling image by repoDigest,Then we will show image name instead.
func normalizeRepoTagPair(repoTags []string, imageName string) (repoTagPairs [][]string) {
const none = "<none>"
if len(repoTags) == 0 {
repoTagPairs = append(repoTagPairs, []string{imageName, none})
return
}
for _, repoTag := range repoTags {
idx := strings.LastIndex(repoTag, ":")
if idx == -1 {
repoTagPairs = append(repoTagPairs, []string{"errorRepoTag", "errorRepoTag"})
continue
}
name := repoTag[:idx]
if name == none {
name = imageName
}
repoTagPairs = append(repoTagPairs, []string{name, repoTag[idx+1:]})
}
return
}
func normalizeRepoDigest(repoDigests []string) (string, string) {
if len(repoDigests) == 0 {
return "<none>", "<none>"
}
repoDigestPair := strings.Split(repoDigests[0], "@")
if len(repoDigestPair) != 2 {
return "errorName", "errorRepoDigest"
}
return repoDigestPair[0], repoDigestPair[1]
}
// PullImageWithSandbox sends a PullImageRequest to the server, and parses
// the returned PullImageResponse.
func PullImageWithSandbox(client pb.ImageServiceClient, image string, auth *pb.AuthConfig, sandbox *pb.PodSandboxConfig) (resp *pb.PullImageResponse, err error) {
request := &pb.PullImageRequest{
Image: &pb.ImageSpec{
Image: image,
},
}
if auth != nil {
request.Auth = auth
}
if sandbox != nil {
request.SandboxConfig = sandbox
}
logrus.Debugf("PullImageRequest: %v", request)
resp, err = client.PullImage(context.Background(), request)
logrus.Debugf("PullImageResponse: %v", resp)
return
}
// ListImages sends a ListImagesRequest to the server, and parses
// the returned ListImagesResponse.
func ListImages(client pb.ImageServiceClient, image string) (resp *pb.ListImagesResponse, err error) {
request := &pb.ListImagesRequest{Filter: &pb.ImageFilter{Image: &pb.ImageSpec{Image: image}}}
logrus.Debugf("ListImagesRequest: %v", request)
resp, err = client.ListImages(context.Background(), request)
logrus.Debugf("ListImagesResponse: %v", resp)
return
}
// ImageStatus sends an ImageStatusRequest to the server, and parses
// the returned ImageStatusResponse.
func ImageStatus(client pb.ImageServiceClient, image string, verbose bool) (resp *pb.ImageStatusResponse, err error) {
request := &pb.ImageStatusRequest{
Image: &pb.ImageSpec{Image: image},
Verbose: verbose,
}
logrus.Debugf("ImageStatusRequest: %v", request)
resp, err = client.ImageStatus(context.Background(), request)
logrus.Debugf("ImageStatusResponse: %v", resp)
return
}
// RemoveImage sends a RemoveImageRequest to the server, and parses
// the returned RemoveImageResponse.
func RemoveImage(client pb.ImageServiceClient, image string) (resp *pb.RemoveImageResponse, err error) {
if image == "" {
return nil, fmt.Errorf("ImageID cannot be empty")
}
request := &pb.RemoveImageRequest{Image: &pb.ImageSpec{Image: image}}
logrus.Debugf("RemoveImageRequest: %v", request)
resp, err = client.RemoveImage(context.Background(), request)
logrus.Debugf("RemoveImageResponse: %v", resp)
return
}
// ImageFsInfo sends an ImageStatusRequest to the server, and parses
// the returned ImageFsInfoResponse.
func ImageFsInfo(client pb.ImageServiceClient) (resp *pb.ImageFsInfoResponse, err error) {
request := &pb.ImageFsInfoRequest{}
logrus.Debugf("ImageFsInfoRequest: %v", request)
resp, err = client.ImageFsInfo(context.Background(), request)
logrus.Debugf("ImageFsInfoResponse: %v", resp)
return
}