Skip to content

Commit fd9949c

Browse files
committed
feat: rewrite inventory client with basic CRUD
This change rewrites the inventory client interface using simple CRUD methods. Prior to this change the inventory client consisted of several tightly coupled interfaces which made the client more complicated and less flexible. The old client also performed a Get of the inventory many times throughout the applier/destroyer processes, including before every update. This approach discarded optimistic concurrency protection since the resourceVersion was never persisted. The new approach preserves an internal interface representation of the inventory so that the resourceVersion can be preserved across updates. The new client interface is simpler to implement and is intended to support implementations for sharded inventories, which may consist of multiple underlying KRM objects. The prior interfaces made assumptions of a single KRM object in several places, which made it incompatible with a sharded inventory.
1 parent d8c50d5 commit fd9949c

37 files changed

+1026
-1830
lines changed

cmd/status/cmdstatus.go

+11-4
Original file line numberDiff line numberDiff line change
@@ -152,7 +152,7 @@ func (r *Runner) preRunE(*cobra.Command, []string) error {
152152
// and get info from the cluster based on the local info
153153
// wrap it to be a map mapping from string to objectMetadataSet
154154
func (r *Runner) loadInvFromDisk(cmd *cobra.Command, args []string) (*printer.PrintData, error) {
155-
inv, err := r.loader.GetInvInfo(cmd, args)
155+
invInfo, err := r.loader.GetInvInfo(cmd, args)
156156
if err != nil {
157157
return nil, err
158158
}
@@ -164,11 +164,13 @@ func (r *Runner) loadInvFromDisk(cmd *cobra.Command, args []string) (*printer.Pr
164164

165165
// Based on the inventory template manifest we look up the inventory
166166
// from the live state using the inventory client.
167-
identifiers, err := invClient.GetClusterObjs(cmd.Context(), inv)
167+
inv, err := invClient.Get(cmd.Context(), invInfo, inventory.GetOptions{})
168168
if err != nil {
169169
return nil, err
170170
}
171171

172+
identifiers := inv.ObjectRefs()
173+
172174
printData := printer.PrintData{
173175
Identifiers: object.ObjMetadataSet{},
174176
InvNameMap: make(map[object.ObjMetadata]string),
@@ -179,7 +181,7 @@ func (r *Runner) loadInvFromDisk(cmd *cobra.Command, args []string) (*printer.Pr
179181
// check if the object is under one of the targeted namespaces
180182
if _, ok := r.namespaceSet[obj.Namespace]; ok || len(r.namespaceSet) == 0 {
181183
// add to the map for future reference
182-
printData.InvNameMap[obj] = inv.Name()
184+
printData.InvNameMap[obj] = invInfo.(inventory.SingleObjectInfo).Name()
183185
// append to identifiers
184186
printData.Identifiers = append(printData.Identifiers, obj)
185187
}
@@ -201,11 +203,16 @@ func (r *Runner) listInvFromCluster() (*printer.PrintData, error) {
201203
StatusSet: r.statusSet,
202204
}
203205

204-
identifiersMap, err := invClient.ListClusterInventoryObjs(r.ctx)
206+
inventories, err := invClient.List(r.ctx, inventory.ListOptions{})
205207
if err != nil {
206208
return nil, err
207209
}
208210

211+
identifiersMap := make(map[string]object.ObjMetadataSet)
212+
for _, inv := range inventories {
213+
identifiersMap[inv.Info().(inventory.SingleObjectInfo).Name()] = inv.ObjectRefs()
214+
}
215+
209216
for invName, identifiers := range identifiersMap {
210217
// Check if there are targeted inventory names and include the current inventory name
211218
if _, ok := r.inventoryNameSet[invName]; !ok && len(r.inventoryNameSet) != 0 {

cmd/status/cmdstatus_test.go

+22-22
Original file line numberDiff line numberDiff line change
@@ -31,8 +31,8 @@ kind: ConfigMap
3131
apiVersion: v1
3232
metadata:
3333
labels:
34-
cli-utils.sigs.k8s.io/inventory-id: test
35-
name: foo
34+
cli-utils.sigs.k8s.io/inventory-id: inventory-id
35+
name: inventory-name
3636
namespace: default
3737
`
3838
depObject = object.ObjMetadata{
@@ -121,8 +121,8 @@ func TestCommand(t *testing.T) {
121121
},
122122
},
123123
expectedOutput: `
124-
foo/deployment.apps/default/foo is InProgress: inProgress
125-
foo/statefulset.apps/default/bar is Current: current
124+
inventory-name/deployment.apps/default/foo is InProgress: inProgress
125+
inventory-name/statefulset.apps/default/bar is Current: current
126126
`,
127127
},
128128
"wait for all current": {
@@ -168,10 +168,10 @@ foo/statefulset.apps/default/bar is Current: current
168168
},
169169
},
170170
expectedOutput: `
171-
foo/deployment.apps/default/foo is InProgress: inProgress
172-
foo/statefulset.apps/default/bar is InProgress: inProgress
173-
foo/statefulset.apps/default/bar is Current: current
174-
foo/deployment.apps/default/foo is Current: current
171+
inventory-name/deployment.apps/default/foo is InProgress: inProgress
172+
inventory-name/statefulset.apps/default/bar is InProgress: inProgress
173+
inventory-name/statefulset.apps/default/bar is Current: current
174+
inventory-name/deployment.apps/default/foo is Current: current
175175
`,
176176
},
177177
"wait for all deleted": {
@@ -201,8 +201,8 @@ foo/deployment.apps/default/foo is Current: current
201201
},
202202
},
203203
expectedOutput: `
204-
foo/statefulset.apps/default/bar is NotFound: notFound
205-
foo/deployment.apps/default/foo is NotFound: notFound
204+
inventory-name/statefulset.apps/default/bar is NotFound: notFound
205+
inventory-name/deployment.apps/default/foo is NotFound: notFound
206206
`,
207207
},
208208
"forever with timeout": {
@@ -233,8 +233,8 @@ foo/deployment.apps/default/foo is NotFound: notFound
233233
},
234234
},
235235
expectedOutput: `
236-
foo/statefulset.apps/default/bar is InProgress: inProgress
237-
foo/deployment.apps/default/foo is InProgress: inProgress
236+
inventory-name/statefulset.apps/default/bar is InProgress: inProgress
237+
inventory-name/deployment.apps/default/foo is InProgress: inProgress
238238
`,
239239
},
240240
}
@@ -283,7 +283,7 @@ foo/deployment.apps/default/foo is InProgress: inProgress
283283
"name": "foo",
284284
"timestamp": "",
285285
"type": "status",
286-
"inventory-name": "foo",
286+
"inventory-name": "inventory-name",
287287
"status": "InProgress",
288288
"message": "inProgress",
289289
},
@@ -294,7 +294,7 @@ foo/deployment.apps/default/foo is InProgress: inProgress
294294
"name": "bar",
295295
"timestamp": "",
296296
"type": "status",
297-
"inventory-name": "foo",
297+
"inventory-name": "inventory-name",
298298
"status": "Current",
299299
"message": "current",
300300
},
@@ -350,7 +350,7 @@ foo/deployment.apps/default/foo is InProgress: inProgress
350350
"name": "foo",
351351
"timestamp": "",
352352
"type": "status",
353-
"inventory-name": "foo",
353+
"inventory-name": "inventory-name",
354354
"status": "InProgress",
355355
"message": "inProgress",
356356
},
@@ -361,7 +361,7 @@ foo/deployment.apps/default/foo is InProgress: inProgress
361361
"name": "bar",
362362
"timestamp": "",
363363
"type": "status",
364-
"inventory-name": "foo",
364+
"inventory-name": "inventory-name",
365365
"status": "InProgress",
366366
"message": "inProgress",
367367
},
@@ -372,7 +372,7 @@ foo/deployment.apps/default/foo is InProgress: inProgress
372372
"name": "bar",
373373
"timestamp": "",
374374
"type": "status",
375-
"inventory-name": "foo",
375+
"inventory-name": "inventory-name",
376376
"status": "Current",
377377
"message": "current",
378378
},
@@ -383,7 +383,7 @@ foo/deployment.apps/default/foo is InProgress: inProgress
383383
"name": "foo",
384384
"timestamp": "",
385385
"type": "status",
386-
"inventory-name": "foo",
386+
"inventory-name": "inventory-name",
387387
"status": "Current",
388388
"message": "current",
389389
},
@@ -423,7 +423,7 @@ foo/deployment.apps/default/foo is InProgress: inProgress
423423
"name": "bar",
424424
"timestamp": "",
425425
"type": "status",
426-
"inventory-name": "foo",
426+
"inventory-name": "inventory-name",
427427
"status": "NotFound",
428428
"message": "notFound",
429429
},
@@ -434,7 +434,7 @@ foo/deployment.apps/default/foo is InProgress: inProgress
434434
"name": "foo",
435435
"timestamp": "",
436436
"type": "status",
437-
"inventory-name": "foo",
437+
"inventory-name": "inventory-name",
438438
"status": "NotFound",
439439
"message": "notFound",
440440
},
@@ -475,7 +475,7 @@ foo/deployment.apps/default/foo is InProgress: inProgress
475475
"name": "bar",
476476
"timestamp": "",
477477
"type": "status",
478-
"inventory-name": "foo",
478+
"inventory-name": "inventory-name",
479479
"status": "InProgress",
480480
"message": "inProgress",
481481
},
@@ -486,7 +486,7 @@ foo/deployment.apps/default/foo is InProgress: inProgress
486486
"name": "foo",
487487
"timestamp": "",
488488
"type": "status",
489-
"inventory-name": "foo",
489+
"inventory-name": "inventory-name",
490490
"status": "InProgress",
491491
"message": "inProgress",
492492
},

pkg/apis/actuation/types.go

-22
Original file line numberDiff line numberDiff line change
@@ -4,31 +4,9 @@
44
package actuation
55

66
import (
7-
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
87
"k8s.io/apimachinery/pkg/types"
98
)
109

11-
// Inventory represents the inventory object in memory.
12-
// Inventory is currently only used for in-memory storage and not serialized to
13-
// disk or to the API server.
14-
type Inventory struct {
15-
metav1.TypeMeta `json:",inline"`
16-
metav1.ObjectMeta `json:"metadata,omitempty"`
17-
18-
Spec InventorySpec `json:"spec,omitempty"`
19-
Status InventoryStatus `json:"status,omitempty"`
20-
}
21-
22-
// InventorySpec is the specification of the desired/expected inventory state.
23-
type InventorySpec struct {
24-
Objects []ObjectReference `json:"objects,omitempty"`
25-
}
26-
27-
// InventoryStatus is the status of the current/last-known inventory state.
28-
type InventoryStatus struct {
29-
Objects []ObjectStatus `json:"objects,omitempty"`
30-
}
31-
3210
// ObjectReference is a reference to a KRM resource by name and kind.
3311
//
3412
// Kubernetes only stores one API Version for each Kind at any given time,

pkg/apis/actuation/zz_generated.deepcopy.go

+1-62
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

pkg/apply/applier.go

+21-6
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import (
88
"fmt"
99
"time"
1010

11+
apierrors "k8s.io/apimachinery/pkg/api/errors"
1112
"k8s.io/apimachinery/pkg/api/meta"
1213
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
1314
"k8s.io/apimachinery/pkg/util/sets"
@@ -52,19 +53,19 @@ type Applier struct {
5253

5354
// prepareObjects returns the set of objects to apply and to prune or
5455
// an error if one occurred.
55-
func (a *Applier) prepareObjects(ctx context.Context, localInv inventory.Info, localObjs object.UnstructuredSet,
56+
func (a *Applier) prepareObjects(ctx context.Context, inv inventory.Inventory, localObjs object.UnstructuredSet,
5657
o ApplierOptions) (object.UnstructuredSet, object.UnstructuredSet, error) {
57-
if localInv == nil {
58+
if inv == nil {
5859
return nil, nil, fmt.Errorf("the local inventory can't be nil")
5960
}
6061
if err := inventory.ValidateNoInventory(localObjs); err != nil {
6162
return nil, nil, err
6263
}
6364
// Add the inventory annotation to the resources being applied.
6465
for _, localObj := range localObjs {
65-
inventory.AddInventoryIDAnnotation(localObj, localInv)
66+
inventory.AddInventoryIDAnnotation(localObj, inv.Info().ID())
6667
}
67-
pruneObjs, err := a.pruner.GetPruneObjs(ctx, localInv, localObjs, prune.Options{
68+
pruneObjs, err := a.pruner.GetPruneObjs(ctx, inv, localObjs, prune.Options{
6869
DryRunStrategy: o.DryRunStrategy,
6970
})
7071
if err != nil {
@@ -96,8 +97,22 @@ func (a *Applier) Run(ctx context.Context, invInfo inventory.Info, objects objec
9697
}
9798
validator.Validate(objects)
9899

100+
inv, err := a.invClient.Get(ctx, invInfo, inventory.GetOptions{})
101+
if apierrors.IsNotFound(err) {
102+
inv, err = a.invClient.NewInventory(invInfo)
103+
}
104+
if err != nil {
105+
handleError(eventChannel, err)
106+
return
107+
}
108+
if inv.Info().ID() != invInfo.ID() {
109+
handleError(eventChannel, fmt.Errorf("expected inventory object to have inventory-id %q but got %q",
110+
invInfo.ID(), inv.Info().ID()))
111+
return
112+
}
113+
99114
// Decide which objects to apply and which to prune
100-
applyObjs, pruneObjs, err := a.prepareObjects(ctx, invInfo, objects, options)
115+
applyObjs, pruneObjs, err := a.prepareObjects(ctx, inv, objects, options)
101116
if err != nil {
102117
handleError(eventChannel, err)
103118
return
@@ -154,6 +169,7 @@ func (a *Applier) Run(ctx context.Context, invInfo inventory.Info, objects objec
154169
OpenAPIGetter: a.openAPIGetter,
155170
InfoHelper: a.infoHelper,
156171
Mapper: a.mapper,
172+
Inventory: inv,
157173
InvClient: a.invClient,
158174
Collector: vCollector,
159175
ApplyFilters: applyFilters,
@@ -175,7 +191,6 @@ func (a *Applier) Run(ctx context.Context, invInfo inventory.Info, objects objec
175191
taskQueue := taskBuilder.
176192
WithApplyObjects(applyObjs).
177193
WithPruneObjects(pruneObjs).
178-
WithInventory(invInfo).
179194
Build(taskContext, opts)
180195

181196
klog.V(4).Infof("validation errors: %d", len(vCollector.Errors))

0 commit comments

Comments
 (0)