8957156c41
Parse the set of image names as tagged references, canonical references, or repository names to which we add the default tag, and return them in libkpod.ImageData reports. Signed-off-by: Nalin Dahyabhai <nalin@redhat.com>
195 lines
4.7 KiB
Go
195 lines
4.7 KiB
Go
package image
|
|
|
|
import (
|
|
"encoding/json"
|
|
"time"
|
|
|
|
"github.com/containers/image/docker/reference"
|
|
"github.com/containers/storage"
|
|
"github.com/kubernetes-incubator/cri-o/libkpod/driver"
|
|
digest "github.com/opencontainers/go-digest"
|
|
ociv1 "github.com/opencontainers/image-spec/specs-go/v1"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
// ImageData handles the data used when inspecting a container
|
|
// nolint
|
|
type ImageData struct {
|
|
ID string
|
|
Tags []string
|
|
Digests []string
|
|
Parent string
|
|
Comment string
|
|
Created *time.Time
|
|
Container string
|
|
ContainerConfig containerConfig
|
|
Author string
|
|
Config ociv1.ImageConfig
|
|
Architecture string
|
|
OS string
|
|
Size uint
|
|
VirtualSize uint
|
|
GraphDriver driver.Data
|
|
RootFS ociv1.RootFS
|
|
}
|
|
|
|
type containerConfig struct {
|
|
Hostname string
|
|
Domainname string
|
|
User string
|
|
AttachStdin bool
|
|
AttachStdout bool
|
|
AttachStderr bool
|
|
Tty bool
|
|
OpenStdin bool
|
|
StdinOnce bool
|
|
Env []string
|
|
Cmd []string
|
|
ArgsEscaped bool
|
|
Image digest.Digest
|
|
Volumes map[string]interface{}
|
|
WorkingDir string
|
|
Entrypoint []string
|
|
Labels interface{}
|
|
OnBuild []string
|
|
}
|
|
|
|
type rootFS struct {
|
|
Type string
|
|
Layers []string
|
|
}
|
|
|
|
// ParseImageNames parses the names we've stored with an image into a list of
|
|
// tagged references and a list of references which contain digests.
|
|
func ParseImageNames(names []string) (tags, digests []string, err error) {
|
|
for _, name := range names {
|
|
if named, err := reference.ParseNamed(name); err == nil {
|
|
if digested, ok := named.(reference.Digested); ok {
|
|
canonical, err := reference.WithDigest(named, digested.Digest())
|
|
if err == nil {
|
|
digests = append(digests, canonical.String())
|
|
}
|
|
} else {
|
|
if reference.IsNameOnly(named) {
|
|
named = reference.TagNameOnly(named)
|
|
}
|
|
if tagged, ok := named.(reference.Tagged); ok {
|
|
namedTagged, err := reference.WithTag(named, tagged.Tag())
|
|
if err == nil {
|
|
tags = append(tags, namedTagged.String())
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return tags, digests, nil
|
|
}
|
|
|
|
// GetImageData gets the ImageData for a container with the given name in the given store.
|
|
func GetImageData(store storage.Store, name string) (*ImageData, error) {
|
|
img, err := FindImage(store, name)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "error reading image %q", name)
|
|
}
|
|
|
|
cid, err := GetImageCopyData(store, name)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "error reading image %q", name)
|
|
}
|
|
blobDigests, err := getDigests(*img)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var bigData interface{}
|
|
ctrConfig := containerConfig{}
|
|
container := ""
|
|
if len(blobDigests) > 0 {
|
|
bd, err := store.ImageBigData(img.ID, string(blobDigests[len(blobDigests)-1]))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
err = json.Unmarshal(bd, &bigData)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
container = (bigData.(map[string]interface{})["container"]).(string)
|
|
cc, err := json.MarshalIndent((bigData.(map[string]interface{})["container_config"]).(map[string]interface{}), "", " ")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
err = json.Unmarshal(cc, &ctrConfig)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
tags, digests, err := ParseImageNames(img.Names)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "error parsing image names for %q", name)
|
|
}
|
|
|
|
driverName, err := driver.GetDriverName(store)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
topLayerID, err := GetTopLayerID(*img)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
driverMetadata, err := driver.GetDriverMetadata(store, topLayerID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
layer, err := store.Layer(topLayerID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
size, err := store.DiffSize(layer.Parent, layer.ID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
virtualSize, err := Size(store, *img)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &ImageData{
|
|
ID: img.ID,
|
|
Tags: tags,
|
|
Digests: digests,
|
|
Parent: string(cid.Docker.Parent),
|
|
Comment: cid.OCIv1.History[0].Comment,
|
|
Created: cid.OCIv1.Created,
|
|
Container: container,
|
|
ContainerConfig: ctrConfig,
|
|
Author: cid.OCIv1.Author,
|
|
Config: cid.OCIv1.Config,
|
|
Architecture: cid.OCIv1.Architecture,
|
|
OS: cid.OCIv1.OS,
|
|
Size: uint(size),
|
|
VirtualSize: uint(virtualSize),
|
|
GraphDriver: driver.Data{
|
|
Name: driverName,
|
|
Data: driverMetadata,
|
|
},
|
|
RootFS: cid.OCIv1.RootFS,
|
|
}, nil
|
|
}
|
|
|
|
func getDigests(img storage.Image) ([]digest.Digest, error) {
|
|
metadata, err := ParseMetadata(img)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
digests := []digest.Digest{}
|
|
for _, blob := range metadata.Blobs {
|
|
digests = append(digests, blob.Digest)
|
|
}
|
|
return digests, nil
|
|
}
|