Skip to content

Commit 65de933

Browse files
author
Michal Minář
committed
image-pruning: adding replica sets to the graph
Signed-off-by: Michal Minář <[email protected]>
1 parent 59ab4e3 commit 65de933

File tree

8 files changed

+164
-0
lines changed

8 files changed

+164
-0
lines changed

pkg/apps/graph/nodes/nodes.go

+27
Original file line numberDiff line numberDiff line change
@@ -92,3 +92,30 @@ func FindOrCreateSyntheticDeploymentConfigNode(g osgraph.MutableUniqueGraph, dc
9292
},
9393
).(*DeploymentConfigNode)
9494
}
95+
96+
// EnsureReplicaSetNode adds the provided replica set to the graph if it does not exist
97+
func EnsureReplicaSetNode(g osgraph.MutableUniqueGraph, rs *kapisext.ReplicaSet) *ReplicaSetNode {
98+
rsName := ReplicaSetNodeName(rs)
99+
rsNode := osgraph.EnsureUnique(
100+
g,
101+
rsName,
102+
func(node osgraph.Node) graph.Node {
103+
return &ReplicaSetNode{Node: node, ReplicaSet: rs, IsFound: true}
104+
},
105+
).(*ReplicaSetNode)
106+
107+
podTemplateSpecNode := kubegraph.EnsurePodTemplateSpecNode(g, &rs.Spec.Template, rs.Namespace, rsName)
108+
g.AddEdge(rsNode, podTemplateSpecNode, osgraph.ContainsEdgeKind)
109+
110+
return rsNode
111+
}
112+
113+
func FindOrCreateSyntheticReplicaSetNode(g osgraph.MutableUniqueGraph, rs *kapisext.ReplicaSet) *ReplicaSetNode {
114+
return osgraph.EnsureUnique(
115+
g,
116+
ReplicaSetNodeName(rs),
117+
func(node osgraph.Node) graph.Node {
118+
return &ReplicaSetNode{Node: node, ReplicaSet: rs, IsFound: false}
119+
},
120+
).(*ReplicaSetNode)
121+
}

pkg/apps/graph/nodes/types.go

+28
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ var (
1313
DaemonSetNodeKind = reflect.TypeOf(kapisext.DaemonSet{}).Name()
1414
DeploymentNodeKind = reflect.TypeOf(kapisext.Deployment{}).Name()
1515
DeploymentConfigNodeKind = reflect.TypeOf(deployapi.DeploymentConfig{}).Name()
16+
ReplicaSetNodeKind = reflect.TypeOf(kapisext.ReplicaSet{}).Name()
1617
)
1718

1819
func DaemonSetNodeName(o *kapisext.DaemonSet) osgraph.UniqueName {
@@ -95,3 +96,30 @@ func (n DeploymentConfigNode) String() string {
9596
func (*DeploymentConfigNode) Kind() string {
9697
return DeploymentConfigNodeKind
9798
}
99+
100+
func ReplicaSetNodeName(o *kapisext.ReplicaSet) osgraph.UniqueName {
101+
return osgraph.GetUniqueRuntimeObjectNodeName(ReplicaSetNodeKind, o)
102+
}
103+
104+
type ReplicaSetNode struct {
105+
osgraph.Node
106+
ReplicaSet *kapisext.ReplicaSet
107+
108+
IsFound bool
109+
}
110+
111+
func (n ReplicaSetNode) Found() bool {
112+
return n.IsFound
113+
}
114+
115+
func (n ReplicaSetNode) Object() interface{} {
116+
return n.ReplicaSet
117+
}
118+
119+
func (n ReplicaSetNode) String() string {
120+
return string(ReplicaSetNodeName(n.ReplicaSet))
121+
}
122+
123+
func (*ReplicaSetNode) Kind() string {
124+
return ReplicaSetNodeKind
125+
}

pkg/cmd/server/bootstrappolicy/policy.go

+1
Original file line numberDiff line numberDiff line change
@@ -553,6 +553,7 @@ func GetOpenshiftBootstrapClusterRoles() []rbac.ClusterRole {
553553
rbac.NewRule("get", "list").Groups(deployGroup, legacyDeployGroup).Resources("deploymentconfigs").RuleOrDie(),
554554
rbac.NewRule("get", "list").Groups(extensionsGroup).Resources("daemonsets").RuleOrDie(),
555555
rbac.NewRule("get", "list").Groups(extensionsGroup).Resources("deployments").RuleOrDie(),
556+
rbac.NewRule("get", "list").Groups(extensionsGroup).Resources("replicasets").RuleOrDie(),
556557

557558
rbac.NewRule("delete").Groups(imageGroup, legacyImageGroup).Resources("images").RuleOrDie(),
558559
rbac.NewRule("get", "list").Groups(imageGroup, legacyImageGroup).Resources("images", "imagestreams").RuleOrDie(),

pkg/image/prune/prune.go

+17
Original file line numberDiff line numberDiff line change
@@ -141,6 +141,8 @@ type PrunerOptions struct {
141141
Deployments *kapisext.DeploymentList
142142
// DCs is the entire list of deployment configs across all namespaces in the cluster.
143143
DCs *deployapi.DeploymentConfigList
144+
// RSs is the entire list of replica sets across all namespaces in the cluster.
145+
RSs *kapisext.ReplicaSetList
144146
// LimitRanges is a map of LimitRanges across namespaces, being keys in this map.
145147
LimitRanges map[string][]*kapi.LimitRange
146148
// DryRun indicates that no changes will be made to the cluster and nothing
@@ -205,6 +207,7 @@ var _ Pruner = &pruner{}
205207
// - any daemonsets
206208
// - any kube deployments
207209
// - any deployment configs
210+
// - any replica sets
208211
// - any build configs
209212
// - any builds
210213
// - the n most recent tag revisions in an image stream's status.tags
@@ -251,6 +254,7 @@ func NewPruner(options PrunerOptions) (Pruner, error) {
251254
addDaemonSetsToGraph(g, options.DSs)
252255
addDeploymentsToGraph(g, options.Deployments)
253256
addDeploymentConfigsToGraph(g, options.DCs)
257+
addReplicaSetsToGraph(g, options.RSs)
254258

255259
return &pruner{
256260
g: g,
@@ -520,6 +524,19 @@ func addDeploymentConfigsToGraph(g graph.Graph, dcs *deployapi.DeploymentConfigL
520524
}
521525
}
522526

527+
// addReplicaSetsToGraph adds replica set to the graph.
528+
//
529+
// Edges are added to the graph from each replica set to the images specified by its pod spec's list of
530+
// containers, as long as the image is managed by OpenShift.
531+
func addReplicaSetsToGraph(g graph.Graph, rss *kapisext.ReplicaSetList) {
532+
for i := range rss.Items {
533+
rs := &rss.Items[i]
534+
glog.V(4).Infof("Examining ReplicaSet %s", getName(rs))
535+
rsNode := deploygraph.EnsureReplicaSetNode(g, rs)
536+
addPodSpecToGraph(g, &rs.Spec.Template.Spec, rsNode)
537+
}
538+
}
539+
523540
// addBuildConfigsToGraph adds build configs to the graph.
524541
//
525542
// Edges are added to the graph from each build config to the image specified by its strategy.from.

pkg/image/prune/prune_test.go

+34
Original file line numberDiff line numberDiff line change
@@ -305,6 +305,26 @@ func dc(namespace, name string, containerImages ...string) deployapi.DeploymentC
305305
}
306306
}
307307

308+
func rsList(rss ...kapisext.ReplicaSet) kapisext.ReplicaSetList {
309+
return kapisext.ReplicaSetList{
310+
Items: rss,
311+
}
312+
}
313+
314+
func rs(namespace, name string, containerImages ...string) kapisext.ReplicaSet {
315+
return kapisext.ReplicaSet{
316+
ObjectMeta: metav1.ObjectMeta{
317+
Namespace: namespace,
318+
Name: name,
319+
},
320+
Spec: kapisext.ReplicaSetSpec{
321+
Template: kapi.PodTemplateSpec{
322+
Spec: podSpec(containerImages...),
323+
},
324+
},
325+
}
326+
}
327+
308328
func bcList(bcs ...buildapi.BuildConfig) buildapi.BuildConfigList {
309329
return buildapi.BuildConfigList{
310330
Items: bcs,
@@ -523,6 +543,7 @@ func TestImagePruning(t *testing.T) {
523543
dss kapisext.DaemonSetList
524544
deployments kapisext.DeploymentList
525545
dcs deployapi.DeploymentConfigList
546+
rss kapisext.ReplicaSetList
526547
limits map[string][]*kapi.LimitRange
527548
expectedImageDeletions []string
528549
expectedStreamUpdates []string
@@ -686,6 +707,15 @@ func TestImagePruning(t *testing.T) {
686707
expectedImageDeletions: []string{"sha256:0000000000000000000000000000000000000000000000000000000000000001"},
687708
},
688709

710+
"referenced by replicaset - don't prune": {
711+
images: imageList(
712+
image("sha256:0000000000000000000000000000000000000000000000000000000000000000", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000000"),
713+
image("sha256:0000000000000000000000000000000000000000000000000000000000000001", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000001"),
714+
),
715+
rss: rsList(rs("foo", "rc1", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000000")),
716+
expectedImageDeletions: []string{"sha256:0000000000000000000000000000000000000000000000000000000000000001"},
717+
},
718+
689719
"referenced by upstream deployment - don't prune": {
690720
images: imageList(
691721
image("sha256:0000000000000000000000000000000000000000000000000000000000000000", registryHost+"/foo/bar@sha256:0000000000000000000000000000000000000000000000000000000000000000"),
@@ -1207,6 +1237,7 @@ func TestImagePruning(t *testing.T) {
12071237
DSs: &test.dss,
12081238
Deployments: &test.deployments,
12091239
DCs: &test.dcs,
1240+
RSs: &test.rss,
12101241
LimitRanges: test.limits,
12111242
RegistryURL: &url.URL{Scheme: "https", Host: registryHost},
12121243
}
@@ -1455,6 +1486,7 @@ func TestRegistryPruning(t *testing.T) {
14551486
DSs: &kapisext.DaemonSetList{},
14561487
Deployments: &kapisext.DeploymentList{},
14571488
DCs: &deployapi.DeploymentConfigList{},
1489+
RSs: &kapisext.ReplicaSetList{},
14581490
RegistryURL: &url.URL{Scheme: "https", Host: "registry1.io"},
14591491
}
14601492
p, err := NewPruner(options)
@@ -1515,6 +1547,7 @@ func TestImageWithStrongAndWeakRefsIsNotPruned(t *testing.T) {
15151547
dss := dsList()
15161548
deployments := deploymentList()
15171549
dcs := dcList()
1550+
rss := rsList()
15181551

15191552
options := PrunerOptions{
15201553
Images: &images,
@@ -1526,6 +1559,7 @@ func TestImageWithStrongAndWeakRefsIsNotPruned(t *testing.T) {
15261559
DSs: &dss,
15271560
Deployments: &deployments,
15281561
DCs: &dcs,
1562+
RSs: &rss,
15291563
}
15301564
keepYoungerThan := 24 * time.Hour
15311565
keepTagRevisions := 2

pkg/oc/admin/prune/images.go

+12
Original file line numberDiff line numberDiff line change
@@ -264,6 +264,7 @@ func (o PruneImagesOptions) Run() error {
264264

265265
allDSs, err := o.KubeClient.Extensions().DaemonSets(o.Namespace).List(metav1.ListOptions{})
266266
if err != nil {
267+
// TODO: remove in future (3.9) release
267268
if !kerrors.IsForbidden(err) {
268269
return err
269270
}
@@ -272,6 +273,7 @@ func (o PruneImagesOptions) Run() error {
272273

273274
allDeployments, err := o.KubeClient.Extensions().Deployments(o.Namespace).List(metav1.ListOptions{})
274275
if err != nil {
276+
// TODO: remove in future (3.9) release
275277
if !kerrors.IsForbidden(err) {
276278
return err
277279
}
@@ -283,6 +285,15 @@ func (o PruneImagesOptions) Run() error {
283285
return err
284286
}
285287

288+
allRSs, err := o.KubeClient.Extensions().ReplicaSets(o.Namespace).List(metav1.ListOptions{})
289+
if err != nil {
290+
// TODO: remove in future (3.9) release
291+
if !kerrors.IsForbidden(err) {
292+
return err
293+
}
294+
fmt.Fprintf(os.Stderr, "Failed to list replicasets: %v\n - * Make sure to update clusterRoleBindings.\n", err)
295+
}
296+
286297
limitRangesList, err := o.KubeClient.Core().LimitRanges(o.Namespace).List(metav1.ListOptions{})
287298
if err != nil {
288299
return err
@@ -352,6 +363,7 @@ func (o PruneImagesOptions) Run() error {
352363
DSs: allDSs,
353364
Deployments: allDeployments,
354365
DCs: allDCs,
366+
RSs: allRSs,
355367
LimitRanges: limitRangesMap,
356368
DryRun: o.Confirm == false,
357369
RegistryClient: registryClient,

test/testdata/bootstrappolicy/bootstrap_cluster_roles.yaml

+21
Original file line numberDiff line numberDiff line change
@@ -1871,6 +1871,27 @@ items:
18711871
verbs:
18721872
- get
18731873
- list
1874+
- apiGroups:
1875+
- extensions
1876+
resources:
1877+
- daemonsets
1878+
verbs:
1879+
- get
1880+
- list
1881+
- apiGroups:
1882+
- extensions
1883+
resources:
1884+
- deployments
1885+
verbs:
1886+
- get
1887+
- list
1888+
- apiGroups:
1889+
- extensions
1890+
resources:
1891+
- replicasets
1892+
verbs:
1893+
- get
1894+
- list
18741895
- apiGroups:
18751896
- ""
18761897
- image.openshift.io

test/testdata/bootstrappolicy/bootstrap_policy_file.yaml

+24
Original file line numberDiff line numberDiff line change
@@ -2041,6 +2041,30 @@ items:
20412041
verbs:
20422042
- get
20432043
- list
2044+
- apiGroups:
2045+
- extensions
2046+
attributeRestrictions: null
2047+
resources:
2048+
- daemonsets
2049+
verbs:
2050+
- get
2051+
- list
2052+
- apiGroups:
2053+
- extensions
2054+
attributeRestrictions: null
2055+
resources:
2056+
- deployments
2057+
verbs:
2058+
- get
2059+
- list
2060+
- apiGroups:
2061+
- extensions
2062+
attributeRestrictions: null
2063+
resources:
2064+
- replicasets
2065+
verbs:
2066+
- get
2067+
- list
20442068
- apiGroups:
20452069
- ""
20462070
- image.openshift.io

0 commit comments

Comments
 (0)