Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 99e3193

Browse files
author
Michal Minář
committedOct 16, 2017
image-pruning: derefence imagestreamtags
Create strong references to images for each pod/bc/dc/etc that uses <host>/<repo>:tag reference. Resolves [bz#1498604](https://bugzilla.redhat.com/show_bug.cgi?id=1498604) Signed-off-by: Michal Minář <[email protected]>
1 parent 969bdd4 commit 99e3193

File tree

2 files changed

+182
-70
lines changed

2 files changed

+182
-70
lines changed
 

‎pkg/image/prune/prune.go

+122-66
Original file line numberDiff line numberDiff line change
@@ -240,28 +240,41 @@ func NewPruner(options PrunerOptions) (Pruner, error) {
240240
}
241241
algorithm.namespace = options.Namespace
242242

243-
g := graph.New()
244-
addImagesToGraph(g, options.Images, algorithm)
245-
addImageStreamsToGraph(g, options.Streams, options.LimitRanges, algorithm)
246-
addPodsToGraph(g, options.Pods, algorithm)
247-
addReplicationControllersToGraph(g, options.RCs)
248-
if err := addBuildConfigsToGraph(g, options.BCs); err != nil {
249-
return nil, err
243+
p := &pruner{
244+
algorithm: algorithm,
245+
registryClient: options.RegistryClient,
246+
registryURL: options.RegistryURL,
250247
}
251-
if err := addBuildsToGraph(g, options.Builds); err != nil {
248+
249+
if err := p.buildGraph(options); err != nil {
252250
return nil, err
253251
}
254-
addDaemonSetsToGraph(g, options.DSs)
255-
addDeploymentsToGraph(g, options.Deployments)
256-
addDeploymentConfigsToGraph(g, options.DCs)
257-
addReplicaSetsToGraph(g, options.RSs)
258252

259-
return &pruner{
260-
g: g,
261-
algorithm: algorithm,
262-
registryClient: options.RegistryClient,
263-
registryURL: options.RegistryURL,
264-
}, nil
253+
return p, nil
254+
}
255+
256+
// buildGraph builds a graph
257+
func (p *pruner) buildGraph(options PrunerOptions) error {
258+
p.g = graph.New()
259+
260+
p.addImagesToGraph(options.Images)
261+
p.addImageStreamsToGraph(options.Streams, options.LimitRanges)
262+
p.addPodsToGraph(options.Pods)
263+
p.addReplicationControllersToGraph(options.RCs)
264+
if err := p.addBuildConfigsToGraph(options.BCs); err != nil {
265+
// TODO: either make the other methods error out on image reference parsing or ignore this error
266+
return err
267+
}
268+
if err := p.addBuildsToGraph(options.Builds); err != nil {
269+
// TODO: either make the other methods error out on image reference parsing or ignore this error
270+
return err
271+
}
272+
p.addDaemonSetsToGraph(options.DSs)
273+
p.addDeploymentsToGraph(options.Deployments)
274+
p.addDeploymentConfigsToGraph(options.DCs)
275+
p.addReplicaSetsToGraph(options.RSs)
276+
277+
return nil
265278
}
266279

267280
func getValue(option interface{}) string {
@@ -275,24 +288,24 @@ func getValue(option interface{}) string {
275288
// registries in the algorithm and are at least as old as the minimum age
276289
// threshold as specified by the algorithm. It also adds all the images' layers
277290
// to the graph.
278-
func addImagesToGraph(g graph.Graph, images *imageapi.ImageList, algorithm pruneAlgorithm) {
291+
func (p *pruner) addImagesToGraph(images *imageapi.ImageList) {
279292
for i := range images.Items {
280293
image := &images.Items[i]
281294

282295
glog.V(4).Infof("Adding image %q to graph", image.Name)
283-
imageNode := imagegraph.EnsureImageNode(g, image)
296+
imageNode := imagegraph.EnsureImageNode(p.g, image)
284297

285298
if image.DockerImageManifestMediaType == schema2.MediaTypeManifest && len(image.DockerImageMetadata.ID) > 0 {
286299
configName := image.DockerImageMetadata.ID
287300
glog.V(4).Infof("Adding image config %q to graph", configName)
288-
configNode := imagegraph.EnsureImageComponentConfigNode(g, configName)
289-
g.AddEdge(imageNode, configNode, ReferencedImageConfigEdgeKind)
301+
configNode := imagegraph.EnsureImageComponentConfigNode(p.g, configName)
302+
p.g.AddEdge(imageNode, configNode, ReferencedImageConfigEdgeKind)
290303
}
291304

292305
for _, layer := range image.DockerImageLayers {
293306
glog.V(4).Infof("Adding image layer %q to graph", layer.Name)
294-
layerNode := imagegraph.EnsureImageComponentLayerNode(g, layer.Name)
295-
g.AddEdge(imageNode, layerNode, ReferencedImageLayerEdgeKind)
307+
layerNode := imagegraph.EnsureImageComponentLayerNode(p.g, layer.Name)
308+
p.g.AddEdge(imageNode, layerNode, ReferencedImageLayerEdgeKind)
296309
}
297310
}
298311
}
@@ -310,7 +323,7 @@ func addImagesToGraph(g graph.Graph, images *imageapi.ImageList, algorithm prune
310323
//
311324
// addImageStreamsToGraph also adds references from each stream to all the
312325
// layers it references (via each image a stream references).
313-
func addImageStreamsToGraph(g graph.Graph, streams *imageapi.ImageStreamList, limits map[string][]*kapi.LimitRange, algorithm pruneAlgorithm) {
326+
func (p *pruner) addImageStreamsToGraph(streams *imageapi.ImageStreamList, limits map[string][]*kapi.LimitRange) {
314327
for i := range streams.Items {
315328
stream := &streams.Items[i]
316329

@@ -319,18 +332,20 @@ func addImageStreamsToGraph(g graph.Graph, streams *imageapi.ImageStreamList, li
319332
// use a weak reference for old image revisions by default
320333
oldImageRevisionReferenceKind := WeakReferencedImageEdgeKind
321334

322-
if !algorithm.pruneOverSizeLimit && stream.CreationTimestamp.Time.After(algorithm.keepYoungerThan) {
335+
if !p.algorithm.pruneOverSizeLimit && stream.CreationTimestamp.Time.After(p.algorithm.keepYoungerThan) {
323336
// stream's age is below threshold - use a strong reference for old image revisions instead
324337
oldImageRevisionReferenceKind = ReferencedImageEdgeKind
325338
}
326339

327340
glog.V(4).Infof("Adding ImageStream %s to graph", getName(stream))
328-
isNode := imagegraph.EnsureImageStreamNode(g, stream)
341+
isNode := imagegraph.EnsureImageStreamNode(p.g, stream)
329342
imageStreamNode := isNode.(*imagegraph.ImageStreamNode)
330343

331344
for tag, history := range stream.Status.Tags {
345+
istNode := imagegraph.EnsureImageStreamTagNode(p.g, makeISTagWithStream(stream, tag))
346+
332347
for i := range history.Items {
333-
n := imagegraph.FindImage(g, history.Items[i].Image)
348+
n := imagegraph.FindImage(p.g, history.Items[i].Image)
334349
if n == nil {
335350
glog.V(2).Infof("Unable to find image %q in graph (from tag=%q, revision=%d, dockerImageReference=%s) - skipping",
336351
history.Items[i].Image, tag, i, history.Items[i].DockerImageReference)
@@ -339,40 +354,45 @@ func addImageStreamsToGraph(g graph.Graph, streams *imageapi.ImageStreamList, li
339354
imageNode := n.(*imagegraph.ImageNode)
340355

341356
kind := oldImageRevisionReferenceKind
342-
if algorithm.pruneOverSizeLimit {
357+
if p.algorithm.pruneOverSizeLimit {
343358
if exceedsLimits(stream, imageNode.Image, limits) {
344359
kind = WeakReferencedImageEdgeKind
345360
} else {
346361
kind = ReferencedImageEdgeKind
347362
}
348363
} else {
349-
if i < algorithm.keepTagRevisions {
364+
if i < p.algorithm.keepTagRevisions {
350365
kind = ReferencedImageEdgeKind
351366
}
352367
}
353368

369+
if i == 0 {
370+
glog.V(4).Infof("Adding edge (kind=%s) from %q to %q", kind, istNode.UniqueName(), imageNode.UniqueName())
371+
p.g.AddEdge(istNode, imageNode, kind)
372+
}
373+
354374
glog.V(4).Infof("Checking for existing strong reference from stream %s to image %s", getName(stream), imageNode.Image.Name)
355-
if edge := g.Edge(imageStreamNode, imageNode); edge != nil && g.EdgeKinds(edge).Has(ReferencedImageEdgeKind) {
375+
if edge := p.g.Edge(imageStreamNode, imageNode); edge != nil && p.g.EdgeKinds(edge).Has(ReferencedImageEdgeKind) {
356376
glog.V(4).Infof("Strong reference found")
357377
continue
358378
}
359379

360380
glog.V(4).Infof("Adding edge (kind=%s) from %q to %q", kind, imageStreamNode.UniqueName(), imageNode.UniqueName())
361-
g.AddEdge(imageStreamNode, imageNode, kind)
381+
p.g.AddEdge(imageStreamNode, imageNode, kind)
362382

363383
glog.V(4).Infof("Adding stream->(layer|config) references")
364384
// add stream -> layer references so we can prune them later
365-
for _, s := range g.From(imageNode) {
385+
for _, s := range p.g.From(imageNode) {
366386
cn, ok := s.(*imagegraph.ImageComponentNode)
367387
if !ok {
368388
continue
369389
}
370390

371391
glog.V(4).Infof("Adding reference from stream %s to %s", getName(stream), cn.Describe())
372392
if cn.Type == imagegraph.ImageComponentTypeConfig {
373-
g.AddEdge(imageStreamNode, s, ReferencedImageConfigEdgeKind)
393+
p.g.AddEdge(imageStreamNode, s, ReferencedImageConfigEdgeKind)
374394
} else {
375-
g.AddEdge(imageStreamNode, s, ReferencedImageLayerEdgeKind)
395+
p.g.AddEdge(imageStreamNode, s, ReferencedImageLayerEdgeKind)
376396
}
377397
}
378398
}
@@ -416,7 +436,7 @@ func exceedsLimits(is *imageapi.ImageStream, image *imageapi.Image, limits map[s
416436
//
417437
// Edges are added to the graph from each pod to the images specified by that
418438
// pod's list of containers, as long as the image is managed by OpenShift.
419-
func addPodsToGraph(g graph.Graph, pods *kapi.PodList, algorithm pruneAlgorithm) {
439+
func (p *pruner) addPodsToGraph(pods *kapi.PodList) {
420440
for i := range pods.Items {
421441
pod := &pods.Items[i]
422442

@@ -426,23 +446,23 @@ func addPodsToGraph(g graph.Graph, pods *kapi.PodList, algorithm pruneAlgorithm)
426446
// pending or running. Additionally, it has to be at least as old as the minimum
427447
// age threshold defined by the algorithm.
428448
if pod.Status.Phase != kapi.PodRunning && pod.Status.Phase != kapi.PodPending {
429-
if !pod.CreationTimestamp.Time.After(algorithm.keepYoungerThan) {
449+
if !pod.CreationTimestamp.Time.After(p.algorithm.keepYoungerThan) {
430450
glog.V(4).Infof("Pod %s is neither running nor pending and is too old", getName(pod))
431451
continue
432452
}
433453
}
434454

435455
glog.V(4).Infof("Adding pod %s to graph", getName(pod))
436-
podNode := kubegraph.EnsurePodNode(g, pod)
456+
podNode := kubegraph.EnsurePodNode(p.g, pod)
437457

438-
addPodSpecToGraph(g, &pod.Spec, podNode)
458+
p.addPodSpecToGraph(&pod.Spec, podNode)
439459
}
440460
}
441461

442462
// Edges are added to the graph from each predecessor (pod or replication
443463
// controller) to the images specified by the pod spec's list of containers, as
444464
// long as the image is managed by OpenShift.
445-
func addPodSpecToGraph(g graph.Graph, spec *kapi.PodSpec, predecessor gonum.Node) {
465+
func (p *pruner) addPodSpecToGraph(spec *kapi.PodSpec, predecessor gonum.Node) {
446466
for j := range spec.Containers {
447467
container := spec.Containers[j]
448468

@@ -455,18 +475,36 @@ func addPodSpecToGraph(g graph.Graph, spec *kapi.PodSpec, predecessor gonum.Node
455475
}
456476

457477
if len(ref.ID) == 0 {
478+
ref = ref.DockerClientDefaults()
479+
if p.registryURL.Host != ref.RegistryURL().Host {
480+
glog.V(4).Infof("%q has third party registry host name (%q != %q) - skipping", container.Image, ref.RegistryURL().Host, p.registryURL.Host)
481+
continue
482+
}
458483
glog.V(4).Infof("%q has no image ID", container.Image)
484+
node := p.g.Find(imagegraph.ImageStreamTagNodeName(makeISTagWithReference(&ref)))
485+
if node == nil {
486+
glog.V(4).Infof("No image stream tag found for %q - skipping", container.Image)
487+
continue
488+
}
489+
for _, n := range p.g.From(node) {
490+
imgNode, ok := n.(*imagegraph.ImageNode)
491+
if !ok {
492+
continue
493+
}
494+
glog.V(4).Infof("Adding edge from pod to image %q referenced by %s:%s", imgNode.Image.Name, ref.RepositoryName(), ref.Tag)
495+
p.g.AddEdge(predecessor, imgNode, ReferencedImageEdgeKind)
496+
}
459497
continue
460498
}
461499

462-
imageNode := imagegraph.FindImage(g, ref.ID)
500+
imageNode := imagegraph.FindImage(p.g, ref.ID)
463501
if imageNode == nil {
464502
glog.V(2).Infof("Unable to find image %q in the graph - skipping", ref.ID)
465503
continue
466504
}
467505

468506
glog.V(4).Infof("Adding edge from pod to image")
469-
g.AddEdge(predecessor, imageNode, ReferencedImageEdgeKind)
507+
p.g.AddEdge(predecessor, imageNode, ReferencedImageEdgeKind)
470508
}
471509
}
472510

@@ -475,38 +513,38 @@ func addPodSpecToGraph(g graph.Graph, spec *kapi.PodSpec, predecessor gonum.Node
475513
// Edges are added to the graph from each replication controller to the images
476514
// specified by its pod spec's list of containers, as long as the image is
477515
// managed by OpenShift.
478-
func addReplicationControllersToGraph(g graph.Graph, rcs *kapi.ReplicationControllerList) {
516+
func (p *pruner) addReplicationControllersToGraph(rcs *kapi.ReplicationControllerList) {
479517
for i := range rcs.Items {
480518
rc := &rcs.Items[i]
481519
glog.V(4).Infof("Examining replication controller %s", getName(rc))
482-
rcNode := kubegraph.EnsureReplicationControllerNode(g, rc)
483-
addPodSpecToGraph(g, &rc.Spec.Template.Spec, rcNode)
520+
rcNode := kubegraph.EnsureReplicationControllerNode(p.g, rc)
521+
p.addPodSpecToGraph(&rc.Spec.Template.Spec, rcNode)
484522
}
485523
}
486524

487525
// addDaemonSetsToGraph adds daemon set to the graph.
488526
//
489527
// Edges are added to the graph from each daemon set to the images specified by its pod spec's list of
490528
// containers, as long as the image is managed by OpenShift.
491-
func addDaemonSetsToGraph(g graph.Graph, dss *kapisext.DaemonSetList) {
529+
func (p *pruner) addDaemonSetsToGraph(dss *kapisext.DaemonSetList) {
492530
for i := range dss.Items {
493531
ds := &dss.Items[i]
494532
glog.V(4).Infof("Examining DaemonSet %s", getName(ds))
495-
dsNode := deploygraph.EnsureDaemonSetNode(g, ds)
496-
addPodSpecToGraph(g, &ds.Spec.Template.Spec, dsNode)
533+
dsNode := deploygraph.EnsureDaemonSetNode(p.g, ds)
534+
p.addPodSpecToGraph(&ds.Spec.Template.Spec, dsNode)
497535
}
498536
}
499537

500538
// addDeploymentsToGraph adds kube's deployments to the graph.
501539
//
502540
// Edges are added to the graph from each deployment to the images specified by its pod spec's list of
503541
// containers, as long as the image is managed by OpenShift.
504-
func addDeploymentsToGraph(g graph.Graph, dmnts *kapisext.DeploymentList) {
542+
func (p *pruner) addDeploymentsToGraph(dmnts *kapisext.DeploymentList) {
505543
for i := range dmnts.Items {
506544
d := &dmnts.Items[i]
507545
glog.V(4).Infof("Examining Deployment %s", getName(d))
508-
dNode := deploygraph.EnsureDeploymentNode(g, d)
509-
addPodSpecToGraph(g, &d.Spec.Template.Spec, dNode)
546+
dNode := deploygraph.EnsureDeploymentNode(p.g, d)
547+
p.addPodSpecToGraph(&d.Spec.Template.Spec, dNode)
510548
}
511549
}
512550

@@ -515,37 +553,37 @@ func addDeploymentsToGraph(g graph.Graph, dmnts *kapisext.DeploymentList) {
515553
// Edges are added to the graph from each deployment config to the images
516554
// specified by its pod spec's list of containers, as long as the image is
517555
// managed by OpenShift.
518-
func addDeploymentConfigsToGraph(g graph.Graph, dcs *deployapi.DeploymentConfigList) {
556+
func (p *pruner) addDeploymentConfigsToGraph(dcs *deployapi.DeploymentConfigList) {
519557
for i := range dcs.Items {
520558
dc := &dcs.Items[i]
521559
glog.V(4).Infof("Examining DeploymentConfig %s", getName(dc))
522-
dcNode := deploygraph.EnsureDeploymentConfigNode(g, dc)
523-
addPodSpecToGraph(g, &dc.Spec.Template.Spec, dcNode)
560+
dcNode := deploygraph.EnsureDeploymentConfigNode(p.g, dc)
561+
p.addPodSpecToGraph(&dc.Spec.Template.Spec, dcNode)
524562
}
525563
}
526564

527565
// addReplicaSetsToGraph adds replica set to the graph.
528566
//
529567
// Edges are added to the graph from each replica set to the images specified by its pod spec's list of
530568
// containers, as long as the image is managed by OpenShift.
531-
func addReplicaSetsToGraph(g graph.Graph, rss *kapisext.ReplicaSetList) {
569+
func (p *pruner) addReplicaSetsToGraph(rss *kapisext.ReplicaSetList) {
532570
for i := range rss.Items {
533571
rs := &rss.Items[i]
534572
glog.V(4).Infof("Examining ReplicaSet %s", getName(rs))
535-
rsNode := deploygraph.EnsureReplicaSetNode(g, rs)
536-
addPodSpecToGraph(g, &rs.Spec.Template.Spec, rsNode)
573+
rsNode := deploygraph.EnsureReplicaSetNode(p.g, rs)
574+
p.addPodSpecToGraph(&rs.Spec.Template.Spec, rsNode)
537575
}
538576
}
539577

540578
// addBuildConfigsToGraph adds build configs to the graph.
541579
//
542580
// Edges are added to the graph from each build config to the image specified by its strategy.from.
543-
func addBuildConfigsToGraph(g graph.Graph, bcs *buildapi.BuildConfigList) error {
581+
func (p *pruner) addBuildConfigsToGraph(bcs *buildapi.BuildConfigList) error {
544582
for i := range bcs.Items {
545583
bc := &bcs.Items[i]
546584
glog.V(4).Infof("Examining BuildConfig %s", getName(bc))
547-
bcNode := buildgraph.EnsureBuildConfigNode(g, bc)
548-
if err := addBuildStrategyImageReferencesToGraph(g, bc.Spec.Strategy, bcNode); err != nil {
585+
bcNode := buildgraph.EnsureBuildConfigNode(p.g, bc)
586+
if err := p.addBuildStrategyImageReferencesToGraph(bc.Spec.Strategy, bcNode); err != nil {
549587
return fmt.Errorf("unable to add BuildConfig %s to graph: %v", getName(bc), err)
550588
}
551589
}
@@ -555,12 +593,12 @@ func addBuildConfigsToGraph(g graph.Graph, bcs *buildapi.BuildConfigList) error
555593
// addBuildsToGraph adds builds to the graph.
556594
//
557595
// Edges are added to the graph from each build to the image specified by its strategy.from.
558-
func addBuildsToGraph(g graph.Graph, builds *buildapi.BuildList) error {
596+
func (p *pruner) addBuildsToGraph(builds *buildapi.BuildList) error {
559597
for i := range builds.Items {
560598
build := &builds.Items[i]
561599
glog.V(4).Infof("Examining build %s", getName(build))
562-
buildNode := buildgraph.EnsureBuildNode(g, build)
563-
if err := addBuildStrategyImageReferencesToGraph(g, build.Spec.Strategy, buildNode); err != nil {
600+
buildNode := buildgraph.EnsureBuildNode(p.g, build)
601+
if err := p.addBuildStrategyImageReferencesToGraph(build.Spec.Strategy, buildNode); err != nil {
564602
return fmt.Errorf("unable to add Build %s to graph: %v", getName(build), err)
565603
}
566604
}
@@ -573,7 +611,7 @@ func addBuildsToGraph(g graph.Graph, builds *buildapi.BuildList) error {
573611
// Edges are added to the graph from each predecessor (build or build config)
574612
// to the image specified by strategy.from, as long as the image is managed by
575613
// OpenShift.
576-
func addBuildStrategyImageReferencesToGraph(g graph.Graph, strategy buildapi.BuildStrategy, predecessor gonum.Node) error {
614+
func (p *pruner) addBuildStrategyImageReferencesToGraph(strategy buildapi.BuildStrategy, predecessor gonum.Node) error {
577615
from := buildapi.GetInputReference(strategy)
578616
if from == nil {
579617
glog.V(4).Infof("Unable to determine 'from' reference - skipping")
@@ -603,14 +641,14 @@ func addBuildStrategyImageReferencesToGraph(g graph.Graph, strategy buildapi.Bui
603641
}
604642

605643
glog.V(4).Infof("Looking for image %q in graph", imageID)
606-
imageNode := imagegraph.FindImage(g, imageID)
644+
imageNode := imagegraph.FindImage(p.g, imageID)
607645
if imageNode == nil {
608646
glog.V(4).Infof("Unable to find image %q in graph - skipping", imageID)
609647
return nil
610648
}
611649

612650
glog.V(4).Infof("Adding edge from %v to %v", predecessor, imageNode)
613-
g.AddEdge(predecessor, imageNode, ReferencedImageEdgeKind)
651+
p.g.AddEdge(predecessor, imageNode, ReferencedImageEdgeKind)
614652

615653
return nil
616654
}
@@ -1166,3 +1204,21 @@ func getName(obj runtime.Object) string {
11661204
}
11671205
return fmt.Sprintf("%s/%s", accessor.GetNamespace(), accessor.GetName())
11681206
}
1207+
1208+
func makeISTag(namespace, name, tag string) *imageapi.ImageStreamTag {
1209+
return &imageapi.ImageStreamTag{
1210+
ObjectMeta: metav1.ObjectMeta{
1211+
Namespace: namespace,
1212+
Name: imageapi.JoinImageStreamTag(name, tag),
1213+
},
1214+
}
1215+
}
1216+
1217+
func makeISTagWithStream(is *imageapi.ImageStream, tag string) *imageapi.ImageStreamTag {
1218+
return makeISTag(is.Namespace, is.Name, tag)
1219+
}
1220+
1221+
func makeISTagWithReference(ref *imageapi.DockerImageReference) *imageapi.ImageStreamTag {
1222+
defaultedRef := ref.DockerClientDefaults()
1223+
return makeISTag(defaultedRef.Namespace, defaultedRef.Name, defaultedRef.Tag)
1224+
}

‎pkg/image/prune/prune_test.go

+60-4
Original file line numberDiff line numberDiff line change
@@ -533,6 +533,7 @@ func TestImagePruning(t *testing.T) {
533533
tests := map[string]struct {
534534
pruneOverSizeLimit *bool
535535
allImages *bool
536+
keepTagRevisions *int
536537
namespace string
537538
images imageapi.ImageList
538539
pods kapi.PodList
@@ -911,6 +912,54 @@ func TestImagePruning(t *testing.T) {
911912
expectedImageDeletions: []string{"sha256:0000000000000000000000000000000000000000000000000000000000000002"},
912913
},
913914

915+
"images referenced by istag - keep": {
916+
keepTagRevisions: keepTagRevisions(0),
917+
images: imageList(
918+
image("sha256:0000000000000000000000000000000000000000000000000000000000000001", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000001"),
919+
image("sha256:0000000000000000000000000000000000000000000000000000000000000002", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000002"),
920+
image("sha256:0000000000000000000000000000000000000000000000000000000000000003", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000003"),
921+
image("sha256:0000000000000000000000000000000000000000000000000000000000000004", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000004"),
922+
image("sha256:0000000000000000000000000000000000000000000000000000000000000005", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000005"),
923+
),
924+
streams: streamList(
925+
stream(registryHost, "foo", "bar", tags(
926+
tag("latest",
927+
tagEvent("sha256:0000000000000000000000000000000000000000000000000000000000000000", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000000"),
928+
tagEvent("sha256:0000000000000000000000000000000000000000000000000000000000000001", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000001"),
929+
tagEvent("sha256:0000000000000000000000000000000000000000000000000000000000000002", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000002"),
930+
tagEvent("sha256:0000000000000000000000000000000000000000000000000000000000000003", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000003"),
931+
tagEvent("sha256:0000000000000000000000000000000000000000000000000000000000000004", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000004"),
932+
tagEvent("sha256:0000000000000000000000000000000000000000000000000000000000000005", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000005"),
933+
),
934+
tag("dummy",
935+
tagEvent("sha256:0000000000000000000000000000000000000000000000000000000000000005", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000005"),
936+
),
937+
)),
938+
stream(registryHost, "foo", "baz", tags(
939+
tag("late",
940+
tagEvent("sha256:0000000000000000000000000000000000000000000000000000000000000002", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000002"),
941+
),
942+
)),
943+
),
944+
dss: dsList(ds("nm", "dsfoo", fmt.Sprintf("%s/%s/%s:%s", registryHost, "foo", "bar", "latest"))),
945+
dcs: dcList(dc("nm", "dcfoo", fmt.Sprintf("%s/%s/%s:%s", "remote.registry:5000", "foo", "bar", "dummy"))),
946+
rss: rsList(rs("nm", "rsfoo", fmt.Sprintf("%s/%s/%s:%s", registryHost, "foo", "baz", "late"))),
947+
expectedImageDeletions: []string{
948+
"sha256:0000000000000000000000000000000000000000000000000000000000000001",
949+
"sha256:0000000000000000000000000000000000000000000000000000000000000003",
950+
"sha256:0000000000000000000000000000000000000000000000000000000000000004",
951+
"sha256:0000000000000000000000000000000000000000000000000000000000000005",
952+
},
953+
expectedStreamUpdates: []string{
954+
"foo/bar:dummy",
955+
"foo/bar|sha256:0000000000000000000000000000000000000000000000000000000000000000",
956+
"foo/bar|sha256:0000000000000000000000000000000000000000000000000000000000000001",
957+
"foo/bar|sha256:0000000000000000000000000000000000000000000000000000000000000003",
958+
"foo/bar|sha256:0000000000000000000000000000000000000000000000000000000000000004",
959+
"foo/bar|sha256:0000000000000000000000000000000000000000000000000000000000000005",
960+
},
961+
},
962+
914963
"multiple resources pointing to image - don't prune": {
915964
images: imageList(
916965
image("sha256:0000000000000000000000000000000000000000000000000000000000000000", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000000"),
@@ -1244,10 +1293,13 @@ func TestImagePruning(t *testing.T) {
12441293
if test.pruneOverSizeLimit != nil {
12451294
options.PruneOverSizeLimit = test.pruneOverSizeLimit
12461295
} else {
1247-
keepYoungerThan := 60 * time.Minute
1248-
keepTagRevisions := 3
1249-
options.KeepYoungerThan = &keepYoungerThan
1250-
options.KeepTagRevisions = &keepTagRevisions
1296+
youngerThan := 60 * time.Minute
1297+
tagRevisions := 3
1298+
if test.keepTagRevisions != nil {
1299+
tagRevisions = *test.keepTagRevisions
1300+
}
1301+
options.KeepYoungerThan = &youngerThan
1302+
options.KeepTagRevisions = &tagRevisions
12511303
}
12521304
p, err := NewPruner(options)
12531305
if err != nil {
@@ -1608,3 +1660,7 @@ func TestImageIsPrunable(t *testing.T) {
16081660
t.Fatalf("Image is prunable although it should not")
16091661
}
16101662
}
1663+
1664+
func keepTagRevisions(n int) *int {
1665+
return &n
1666+
}

0 commit comments

Comments
 (0)
Please sign in to comment.