0
0
Fork 0
mirror of https://github.com/crazy-max/diun.git synced 2025-04-05 03:45:25 +00:00

Handle digest based image reference ()

Co-authored-by: CrazyMax <crazy-max@users.noreply.github.com>
This commit is contained in:
CrazyMax 2021-04-25 00:28:20 +02:00 committed by GitHub
parent b7d9265df4
commit 9b4e62474b
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 160 additions and 37 deletions
internal/provider
pkg/docker

View file

@ -6,12 +6,10 @@ import (
"strings"
"github.com/crazy-max/diun/v4/internal/model"
"github.com/rs/zerolog/log"
)
// ValidateContainerImage returns a standard image through Docker labels
func ValidateContainerImage(image string, labels map[string]string, watchByDef bool) (img model.Image, err error) {
log.Debug().Msgf("Validating %s", image)
// ValidateImage returns a standard image through Docker labels
func ValidateImage(image string, labels map[string]string, watchByDef bool) (img model.Image, err error) {
if i := strings.Index(image, "@sha256:"); i > 0 {
image = image[:i]
}

View file

@ -36,28 +36,73 @@ func (c *Client) listContainerImage() []model.Image {
var list []model.Image
for _, ctn := range ctns {
imageRaw, err := cli.RawImage(ctn.Image)
imageName := ctn.Image
imageRaw, err := cli.ImageInspectWithRaw(imageName)
if err != nil {
c.logger.Error().Err(err).Msgf("Cannot inspect image from container %s", ctn.ID)
continue
}
if local := cli.IsLocalImage(imageRaw); local {
c.logger.Debug().Msgf("Skip locally built image from container %s", ctn.ID)
continue
}
if dangling := cli.IsDanglingImage(imageRaw); dangling {
c.logger.Debug().Msgf("Skip dangling image from container %s", ctn.ID)
c.logger.Error().Err(err).
Str("ctn_id", ctn.ID).
Str("ctn_image", imageName).
Msg("Cannot inspect image")
continue
}
image, err := provider.ValidateContainerImage(ctn.Image, ctn.Labels, *c.config.WatchByDefault)
if err != nil {
c.logger.Error().Err(err).Msgf("Cannot get image from container %s", ctn.ID)
continue
} else if reflect.DeepEqual(image, model.Image{}) {
c.logger.Debug().Msgf("Watch disabled for container %s", ctn.ID)
if local := cli.IsLocalImage(imageRaw); local {
c.logger.Debug().
Str("ctn_id", ctn.ID).
Str("ctn_image", imageName).
Msg("Skip locally built image")
continue
}
if dangling := cli.IsDanglingImage(imageRaw); dangling {
c.logger.Debug().
Str("ctn_id", ctn.ID).
Str("ctn_image", imageName).
Msg("Skip dangling image")
continue
}
if cli.IsDigest(imageName) {
if len(imageRaw.RepoDigests) > 0 {
c.logger.Debug().
Str("ctn_id", ctn.ID).
Str("ctn_image", imageName).
Strs("img_repodigests", imageRaw.RepoDigests).
Msg("Using first image repo digest available as image name")
imageName = imageRaw.RepoDigests[0]
} else {
c.logger.Debug().
Str("ctn_id", ctn.ID).
Str("ctn_image", imageName).
Strs("img_repodigests", imageRaw.RepoDigests).
Msg("Skip unknown image digest ref")
continue
}
}
c.logger.Debug().
Str("ctn_id", ctn.ID).
Str("ctn_image", imageName).
Interface("ctn_labels", ctn.Labels).
Msg("Validate image")
image, err := provider.ValidateImage(imageName, ctn.Labels, *c.config.WatchByDefault)
if err != nil {
c.logger.Error().Err(err).
Str("ctn_id", ctn.ID).
Str("ctn_image", imageName).
Interface("ctn_labels", ctn.Labels).
Msg("Invalid image")
continue
} else if reflect.DeepEqual(image, model.Image{}) {
c.logger.Debug().
Str("ctn_id", ctn.ID).
Str("ctn_image", imageName).
Interface("ctn_labels", ctn.Labels).
Msg("Watch disabled")
continue
}
list = append(list, image)
}

View file

@ -32,14 +32,32 @@ func (c *Client) listPodImage() []model.Image {
var list []model.Image
for _, pod := range pods {
for _, ctn := range pod.Spec.Containers {
image, err := provider.ValidateContainerImage(ctn.Image, pod.Annotations, *c.config.WatchByDefault)
c.logger.Debug().
Str("pod_name", pod.Name).
Interface("pod_annot", pod.Annotations).
Str("ctn_name", ctn.Name).
Str("ctn_image", ctn.Image).
Msg("Validate image")
image, err := provider.ValidateImage(ctn.Image, pod.Annotations, *c.config.WatchByDefault)
if err != nil {
c.logger.Error().Err(err).Msgf("Cannot get image from container %s (pod %s)", ctn.Name, pod.Name)
c.logger.Error().Err(err).
Str("pod_name", pod.Name).
Interface("pod_annot", pod.Annotations).
Str("ctn_name", ctn.Name).
Str("ctn_image", ctn.Image).
Msg("Invalid image")
continue
} else if reflect.DeepEqual(image, model.Image{}) {
c.logger.Debug().Msgf("Watch disabled for container %s (pod %s)", ctn.Name, pod.Name)
c.logger.Debug().
Str("pod_name", pod.Name).
Interface("pod_annot", pod.Annotations).
Str("ctn_name", ctn.Name).
Str("ctn_image", ctn.Image).
Msg("Watch disabled")
continue
}
list = append(list, image)
}
}

View file

@ -29,25 +29,73 @@ func (c *Client) listServiceImage() []model.Image {
var list []model.Image
for _, svc := range svcs {
if imageRaw, err := cli.RawImage(svc.Spec.TaskTemplate.ContainerSpec.Image); err == nil {
if local := cli.IsLocalImage(imageRaw); local {
c.logger.Debug().Msgf("Skip locally built image for service %s", svc.Spec.Name)
continue
}
if dangling := cli.IsDanglingImage(imageRaw); dangling {
c.logger.Debug().Msgf("Skip dangling image for service %s", svc.Spec.Name)
imageName := svc.Spec.TaskTemplate.ContainerSpec.Image
imageRaw, err := cli.ImageInspectWithRaw(svc.Spec.TaskTemplate.ContainerSpec.Image)
if err != nil {
c.logger.Error().Err(err).
Str("svc_name", svc.Spec.Name).
Str("ctn_image", imageName).
Msg("Cannot inspect image")
continue
}
if local := cli.IsLocalImage(imageRaw); local {
c.logger.Debug().
Str("svc_name", svc.Spec.Name).
Str("ctn_image", imageName).
Msg("Skip locally built image")
continue
}
if dangling := cli.IsDanglingImage(imageRaw); dangling {
c.logger.Debug().
Str("svc_name", svc.Spec.Name).
Str("ctn_image", imageName).
Msg("Skip dangling image")
continue
}
if cli.IsDigest(imageName) {
if len(imageRaw.RepoDigests) > 0 {
c.logger.Debug().
Str("svc_name", svc.Spec.Name).
Str("ctn_image", imageName).
Strs("img_repodigests", imageRaw.RepoDigests).
Msg("Using first image repo digest available as image name")
imageName = imageRaw.RepoDigests[0]
} else {
c.logger.Debug().
Str("svc_name", svc.Spec.Name).
Str("ctn_image", imageName).
Strs("img_repodigests", imageRaw.RepoDigests).
Msg("Skip unknown image digest ref")
continue
}
}
image, err := provider.ValidateContainerImage(svc.Spec.TaskTemplate.ContainerSpec.Image, svc.Spec.Labels, *c.config.WatchByDefault)
c.logger.Debug().
Str("svc_name", svc.Spec.Name).
Interface("svc_labels", svc.Spec.Labels).
Str("ctn_image", imageName).
Msg("Validate image")
image, err := provider.ValidateImage(imageName, svc.Spec.Labels, *c.config.WatchByDefault)
if err != nil {
c.logger.Error().Err(err).Msgf("Cannot get image from service %s", svc.Spec.Name)
c.logger.Error().Err(err).
Str("svc_name", svc.Spec.Name).
Interface("svc_labels", svc.Spec.Labels).
Str("ctn_image", svc.Spec.TaskTemplate.ContainerSpec.Image).
Msg("Invalid image")
continue
} else if reflect.DeepEqual(image, model.Image{}) {
c.logger.Debug().Msgf("Watch disabled for service %s", svc.Spec.Name)
c.logger.Debug().
Str("svc_name", svc.Spec.Name).
Interface("svc_labels", svc.Spec.Labels).
Str("ctn_image", svc.Spec.TaskTemplate.ContainerSpec.Image).
Msg("Watch disabled")
continue
}
list = append(list, image)
}

View file

@ -1,10 +1,24 @@
package docker
import "github.com/docker/docker/api/types"
import (
"regexp"
// RawImage returns the image information and its raw representation
func (c *Client) RawImage(image string) (types.ImageInspect, error) {
imageRaw, _, err := c.API.ImageInspectWithRaw(c.ctx, image)
"github.com/docker/docker/api/types"
)
// ContainerInspect returns the container information.
func (c *Client) ContainerInspect(containerID string) (types.ContainerJSON, error) {
return c.API.ContainerInspect(c.ctx, containerID)
}
// IsDigest determines image it looks like a digest based image reference.
func (c *Client) IsDigest(imageID string) bool {
return regexp.MustCompile(`^(@|sha256:|@sha256:)([0-9a-f]{64})$`).MatchString(imageID)
}
// ImageInspectWithRaw returns the image information and its raw representation.
func (c *Client) ImageInspectWithRaw(imageID string) (types.ImageInspect, error) {
imageRaw, _, err := c.API.ImageInspectWithRaw(c.ctx, imageID)
return imageRaw, err
}