@@ -240,28 +240,41 @@ func NewPruner(options PrunerOptions) (Pruner, error) {
240
240
}
241
241
algorithm .namespace = options .Namespace
242
242
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 ,
250
247
}
251
- if err := addBuildsToGraph (g , options .Builds ); err != nil {
248
+
249
+ if err := p .buildGraph (options ); err != nil {
252
250
return nil , err
253
251
}
254
- addDaemonSetsToGraph (g , options .DSs )
255
- addDeploymentsToGraph (g , options .Deployments )
256
- addDeploymentConfigsToGraph (g , options .DCs )
257
- addReplicaSetsToGraph (g , options .RSs )
258
252
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
265
278
}
266
279
267
280
func getValue (option interface {}) string {
@@ -275,24 +288,24 @@ func getValue(option interface{}) string {
275
288
// registries in the algorithm and are at least as old as the minimum age
276
289
// threshold as specified by the algorithm. It also adds all the images' layers
277
290
// to the graph.
278
- func addImagesToGraph ( g graph. Graph , images * imageapi.ImageList , algorithm pruneAlgorithm ) {
291
+ func ( p * pruner ) addImagesToGraph ( images * imageapi.ImageList ) {
279
292
for i := range images .Items {
280
293
image := & images .Items [i ]
281
294
282
295
glog .V (4 ).Infof ("Adding image %q to graph" , image .Name )
283
- imageNode := imagegraph .EnsureImageNode (g , image )
296
+ imageNode := imagegraph .EnsureImageNode (p . g , image )
284
297
285
298
if image .DockerImageManifestMediaType == schema2 .MediaTypeManifest && len (image .DockerImageMetadata .ID ) > 0 {
286
299
configName := image .DockerImageMetadata .ID
287
300
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 )
290
303
}
291
304
292
305
for _ , layer := range image .DockerImageLayers {
293
306
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 )
296
309
}
297
310
}
298
311
}
@@ -310,7 +323,7 @@ func addImagesToGraph(g graph.Graph, images *imageapi.ImageList, algorithm prune
310
323
//
311
324
// addImageStreamsToGraph also adds references from each stream to all the
312
325
// 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 ) {
314
327
for i := range streams .Items {
315
328
stream := & streams .Items [i ]
316
329
@@ -319,18 +332,20 @@ func addImageStreamsToGraph(g graph.Graph, streams *imageapi.ImageStreamList, li
319
332
// use a weak reference for old image revisions by default
320
333
oldImageRevisionReferenceKind := WeakReferencedImageEdgeKind
321
334
322
- if ! algorithm .pruneOverSizeLimit && stream .CreationTimestamp .Time .After (algorithm .keepYoungerThan ) {
335
+ if ! p . algorithm .pruneOverSizeLimit && stream .CreationTimestamp .Time .After (p . algorithm .keepYoungerThan ) {
323
336
// stream's age is below threshold - use a strong reference for old image revisions instead
324
337
oldImageRevisionReferenceKind = ReferencedImageEdgeKind
325
338
}
326
339
327
340
glog .V (4 ).Infof ("Adding ImageStream %s to graph" , getName (stream ))
328
- isNode := imagegraph .EnsureImageStreamNode (g , stream )
341
+ isNode := imagegraph .EnsureImageStreamNode (p . g , stream )
329
342
imageStreamNode := isNode .(* imagegraph.ImageStreamNode )
330
343
331
344
for tag , history := range stream .Status .Tags {
345
+ istNode := imagegraph .EnsureImageStreamTagNode (p .g , makeISTagWithStream (stream , tag ))
346
+
332
347
for i := range history .Items {
333
- n := imagegraph .FindImage (g , history .Items [i ].Image )
348
+ n := imagegraph .FindImage (p . g , history .Items [i ].Image )
334
349
if n == nil {
335
350
glog .V (2 ).Infof ("Unable to find image %q in graph (from tag=%q, revision=%d, dockerImageReference=%s) - skipping" ,
336
351
history .Items [i ].Image , tag , i , history .Items [i ].DockerImageReference )
@@ -339,40 +354,45 @@ func addImageStreamsToGraph(g graph.Graph, streams *imageapi.ImageStreamList, li
339
354
imageNode := n .(* imagegraph.ImageNode )
340
355
341
356
kind := oldImageRevisionReferenceKind
342
- if algorithm .pruneOverSizeLimit {
357
+ if p . algorithm .pruneOverSizeLimit {
343
358
if exceedsLimits (stream , imageNode .Image , limits ) {
344
359
kind = WeakReferencedImageEdgeKind
345
360
} else {
346
361
kind = ReferencedImageEdgeKind
347
362
}
348
363
} else {
349
- if i < algorithm .keepTagRevisions {
364
+ if i < p . algorithm .keepTagRevisions {
350
365
kind = ReferencedImageEdgeKind
351
366
}
352
367
}
353
368
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
+
354
374
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 ) {
356
376
glog .V (4 ).Infof ("Strong reference found" )
357
377
continue
358
378
}
359
379
360
380
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 )
362
382
363
383
glog .V (4 ).Infof ("Adding stream->(layer|config) references" )
364
384
// 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 ) {
366
386
cn , ok := s .(* imagegraph.ImageComponentNode )
367
387
if ! ok {
368
388
continue
369
389
}
370
390
371
391
glog .V (4 ).Infof ("Adding reference from stream %s to %s" , getName (stream ), cn .Describe ())
372
392
if cn .Type == imagegraph .ImageComponentTypeConfig {
373
- g .AddEdge (imageStreamNode , s , ReferencedImageConfigEdgeKind )
393
+ p . g .AddEdge (imageStreamNode , s , ReferencedImageConfigEdgeKind )
374
394
} else {
375
- g .AddEdge (imageStreamNode , s , ReferencedImageLayerEdgeKind )
395
+ p . g .AddEdge (imageStreamNode , s , ReferencedImageLayerEdgeKind )
376
396
}
377
397
}
378
398
}
@@ -416,7 +436,7 @@ func exceedsLimits(is *imageapi.ImageStream, image *imageapi.Image, limits map[s
416
436
//
417
437
// Edges are added to the graph from each pod to the images specified by that
418
438
// 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 ) {
420
440
for i := range pods .Items {
421
441
pod := & pods .Items [i ]
422
442
@@ -426,23 +446,23 @@ func addPodsToGraph(g graph.Graph, pods *kapi.PodList, algorithm pruneAlgorithm)
426
446
// pending or running. Additionally, it has to be at least as old as the minimum
427
447
// age threshold defined by the algorithm.
428
448
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 ) {
430
450
glog .V (4 ).Infof ("Pod %s is neither running nor pending and is too old" , getName (pod ))
431
451
continue
432
452
}
433
453
}
434
454
435
455
glog .V (4 ).Infof ("Adding pod %s to graph" , getName (pod ))
436
- podNode := kubegraph .EnsurePodNode (g , pod )
456
+ podNode := kubegraph .EnsurePodNode (p . g , pod )
437
457
438
- addPodSpecToGraph (g , & pod .Spec , podNode )
458
+ p . addPodSpecToGraph (& pod .Spec , podNode )
439
459
}
440
460
}
441
461
442
462
// Edges are added to the graph from each predecessor (pod or replication
443
463
// controller) to the images specified by the pod spec's list of containers, as
444
464
// 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 ) {
446
466
for j := range spec .Containers {
447
467
container := spec .Containers [j ]
448
468
@@ -455,18 +475,36 @@ func addPodSpecToGraph(g graph.Graph, spec *kapi.PodSpec, predecessor gonum.Node
455
475
}
456
476
457
477
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
+ }
458
483
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
+ }
459
497
continue
460
498
}
461
499
462
- imageNode := imagegraph .FindImage (g , ref .ID )
500
+ imageNode := imagegraph .FindImage (p . g , ref .ID )
463
501
if imageNode == nil {
464
502
glog .V (2 ).Infof ("Unable to find image %q in the graph - skipping" , ref .ID )
465
503
continue
466
504
}
467
505
468
506
glog .V (4 ).Infof ("Adding edge from pod to image" )
469
- g .AddEdge (predecessor , imageNode , ReferencedImageEdgeKind )
507
+ p . g .AddEdge (predecessor , imageNode , ReferencedImageEdgeKind )
470
508
}
471
509
}
472
510
@@ -475,38 +513,38 @@ func addPodSpecToGraph(g graph.Graph, spec *kapi.PodSpec, predecessor gonum.Node
475
513
// Edges are added to the graph from each replication controller to the images
476
514
// specified by its pod spec's list of containers, as long as the image is
477
515
// managed by OpenShift.
478
- func addReplicationControllersToGraph ( g graph. Graph , rcs * kapi.ReplicationControllerList ) {
516
+ func ( p * pruner ) addReplicationControllersToGraph ( rcs * kapi.ReplicationControllerList ) {
479
517
for i := range rcs .Items {
480
518
rc := & rcs .Items [i ]
481
519
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 )
484
522
}
485
523
}
486
524
487
525
// addDaemonSetsToGraph adds daemon set to the graph.
488
526
//
489
527
// Edges are added to the graph from each daemon set to the images specified by its pod spec's list of
490
528
// 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 ) {
492
530
for i := range dss .Items {
493
531
ds := & dss .Items [i ]
494
532
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 )
497
535
}
498
536
}
499
537
500
538
// addDeploymentsToGraph adds kube's deployments to the graph.
501
539
//
502
540
// Edges are added to the graph from each deployment to the images specified by its pod spec's list of
503
541
// 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 ) {
505
543
for i := range dmnts .Items {
506
544
d := & dmnts .Items [i ]
507
545
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 )
510
548
}
511
549
}
512
550
@@ -515,37 +553,37 @@ func addDeploymentsToGraph(g graph.Graph, dmnts *kapisext.DeploymentList) {
515
553
// Edges are added to the graph from each deployment config to the images
516
554
// specified by its pod spec's list of containers, as long as the image is
517
555
// managed by OpenShift.
518
- func addDeploymentConfigsToGraph ( g graph. Graph , dcs * deployapi.DeploymentConfigList ) {
556
+ func ( p * pruner ) addDeploymentConfigsToGraph ( dcs * deployapi.DeploymentConfigList ) {
519
557
for i := range dcs .Items {
520
558
dc := & dcs .Items [i ]
521
559
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 )
524
562
}
525
563
}
526
564
527
565
// addReplicaSetsToGraph adds replica set to the graph.
528
566
//
529
567
// Edges are added to the graph from each replica set to the images specified by its pod spec's list of
530
568
// 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 ) {
532
570
for i := range rss .Items {
533
571
rs := & rss .Items [i ]
534
572
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 )
537
575
}
538
576
}
539
577
540
578
// addBuildConfigsToGraph adds build configs to the graph.
541
579
//
542
580
// 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 {
544
582
for i := range bcs .Items {
545
583
bc := & bcs .Items [i ]
546
584
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 {
549
587
return fmt .Errorf ("unable to add BuildConfig %s to graph: %v" , getName (bc ), err )
550
588
}
551
589
}
@@ -555,12 +593,12 @@ func addBuildConfigsToGraph(g graph.Graph, bcs *buildapi.BuildConfigList) error
555
593
// addBuildsToGraph adds builds to the graph.
556
594
//
557
595
// 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 {
559
597
for i := range builds .Items {
560
598
build := & builds .Items [i ]
561
599
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 {
564
602
return fmt .Errorf ("unable to add Build %s to graph: %v" , getName (build ), err )
565
603
}
566
604
}
@@ -573,7 +611,7 @@ func addBuildsToGraph(g graph.Graph, builds *buildapi.BuildList) error {
573
611
// Edges are added to the graph from each predecessor (build or build config)
574
612
// to the image specified by strategy.from, as long as the image is managed by
575
613
// 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 {
577
615
from := buildapi .GetInputReference (strategy )
578
616
if from == nil {
579
617
glog .V (4 ).Infof ("Unable to determine 'from' reference - skipping" )
@@ -603,14 +641,14 @@ func addBuildStrategyImageReferencesToGraph(g graph.Graph, strategy buildapi.Bui
603
641
}
604
642
605
643
glog .V (4 ).Infof ("Looking for image %q in graph" , imageID )
606
- imageNode := imagegraph .FindImage (g , imageID )
644
+ imageNode := imagegraph .FindImage (p . g , imageID )
607
645
if imageNode == nil {
608
646
glog .V (4 ).Infof ("Unable to find image %q in graph - skipping" , imageID )
609
647
return nil
610
648
}
611
649
612
650
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 )
614
652
615
653
return nil
616
654
}
@@ -1166,3 +1204,21 @@ func getName(obj runtime.Object) string {
1166
1204
}
1167
1205
return fmt .Sprintf ("%s/%s" , accessor .GetNamespace (), accessor .GetName ())
1168
1206
}
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
+ }
0 commit comments