Skip to content

Commit d88a058

Browse files
committed
Add support for external references.
- Add ReadOnly flag to resource. True indicates an external resource we want to read (no create/update/delete) - Reuse template to describe the external resource (gvkn). Reason: we could templatize the name part of the external resource - Use as much of existing resource reconcile flow as possible. Reason: we can optionally support ReadyWhen and IncludeWhen rules for the external resource - Rename WantToCreateResource() to ReadyToProcessResource() to reflect the fact that resources will not be created in case readOnly is set to true - mark readonly resource as skipped on deletion
1 parent 7ea746c commit d88a058

File tree

9 files changed

+73
-14
lines changed

9 files changed

+73
-14
lines changed

api/v1alpha1/types.go

+3
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,9 @@ type Resource struct {
9090
ID string `json:"id,omitempty"`
9191
// +kubebuilder:validation:Required
9292
Template runtime.RawExtension `json:"template,omitempty"`
93+
// ReadOnly indicates an external resource we want to read and use in the Graph
94+
// +kubebuilder:validation:Optional
95+
ReadOnly bool `json:"readOnly,omitempty"`
9396
// +kubebuilder:validation:Optional
9497
ReadyWhen []string `json:"readyWhen,omitempty"`
9598
// +kubebuilder:validation:Optional

config/crd/bases/kro.run_resourcegraphdefinitions.yaml

+4
Original file line numberDiff line numberDiff line change
@@ -79,6 +79,10 @@ spec:
7979
items:
8080
type: string
8181
type: array
82+
readOnly:
83+
description: ReadOnly indicates an external resource we want
84+
to read and use in the Graph
85+
type: boolean
8286
readyWhen:
8387
items:
8488
type: string

helm/crds/kro.run_resourcegraphdefinitions.yaml

+4
Original file line numberDiff line numberDiff line change
@@ -79,6 +79,10 @@ spec:
7979
items:
8080
type: string
8181
type: array
82+
readOnly:
83+
description: ReadOnly indicates an external resource we want
84+
to read and use in the Graph
85+
type: boolean
8286
readyWhen:
8387
items:
8488
type: string

pkg/controller/instance/controller_reconcile.go

+21-3
Original file line numberDiff line numberDiff line change
@@ -142,9 +142,9 @@ func (igr *instanceGraphReconciler) reconcileResource(ctx context.Context, resou
142142
resourceState := &ResourceState{State: "IN_PROGRESS"}
143143
igr.state.ResourceStates[resourceID] = resourceState
144144

145-
// Check if resource should be created
146-
if want, err := igr.runtime.WantToCreateResource(resourceID); err != nil || !want {
147-
log.V(1).Info("Skipping resource creation", "reason", err)
145+
// Check if resource should be processed (create or get)
146+
if want, err := igr.runtime.ReadyToProcessResource(resourceID); err != nil || !want {
147+
log.V(1).Info("Skipping resource processing", "reason", err)
148148
resourceState.State = "SKIPPED"
149149
igr.runtime.IgnoreResource(resourceID)
150150
return nil
@@ -177,6 +177,12 @@ func (igr *instanceGraphReconciler) handleResourceReconciliation(
177177
observed, err := rc.Get(ctx, resource.GetName(), metav1.GetOptions{})
178178
if err != nil {
179179
if apierrors.IsNotFound(err) {
180+
// For read-only resources, we don't create
181+
if igr.runtime.ResourceDescriptor(resourceID).IsReadOnly() {
182+
resourceState.State = "WAITING_FOR_READONLY"
183+
resourceState.Err = fmt.Errorf("read-only resource not found: %w", err)
184+
return igr.delayedRequeue(resourceState.Err)
185+
}
180186
return igr.handleResourceCreation(ctx, rc, resource, resourceID, resourceState)
181187
}
182188
resourceState.State = "ERROR"
@@ -196,6 +202,12 @@ func (igr *instanceGraphReconciler) handleResourceReconciliation(
196202
}
197203

198204
resourceState.State = "SYNCED"
205+
206+
// For read-only resources, don't perform updates
207+
if igr.runtime.ResourceDescriptor(resourceID).IsReadOnly() {
208+
return nil
209+
}
210+
199211
return igr.updateResource(ctx, rc, resource, observed, resourceID, resourceState)
200212
}
201213

@@ -354,6 +366,12 @@ func (igr *instanceGraphReconciler) deleteResourcesInOrder(ctx context.Context)
354366
continue
355367
}
356368

369+
// Skip deletion for read-only resources
370+
if igr.runtime.ResourceDescriptor(resourceID).IsReadOnly() {
371+
igr.state.ResourceStates[resourceID].State = "SKIPPED"
372+
continue
373+
}
374+
357375
if err := igr.deleteResource(ctx, resourceID); err != nil {
358376
return err
359377
}

pkg/graph/builder.go

+1
Original file line numberDiff line numberDiff line change
@@ -340,6 +340,7 @@ func (b *Builder) buildRGResource(rgResource *v1alpha1.Resource, namespacedResou
340340
includeWhenExpressions: includeWhen,
341341
namespaced: isNamespaced,
342342
order: order,
343+
readOnly: rgResource.ReadOnly,
343344
}, nil
344345
}
345346

pkg/graph/resource.go

+8
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,8 @@ type Resource struct {
7373
// order reflects the original order in which the resources were specified,
7474
// and lets us keep the client-specified ordering where the dependencies allow.
7575
order int
76+
// readOnly indicates if the resource should only be read and not created/updated
77+
readOnly bool
7678
}
7779

7880
// GetDependencies returns the dependencies of the resource.
@@ -164,6 +166,11 @@ func (r *Resource) IsNamespaced() bool {
164166
return r.namespaced
165167
}
166168

169+
// IsReadOnly returns whether the resource is read-only
170+
func (r *Resource) IsReadOnly() bool {
171+
return r.readOnly
172+
}
173+
167174
// DeepCopy returns a deep copy of the resource.
168175
func (r *Resource) DeepCopy() *Resource {
169176
return &Resource{
@@ -177,5 +184,6 @@ func (r *Resource) DeepCopy() *Resource {
177184
readyWhenExpressions: slices.Clone(r.readyWhenExpressions),
178185
includeWhenExpressions: slices.Clone(r.includeWhenExpressions),
179186
namespaced: r.namespaced,
187+
readOnly: r.readOnly,
180188
}
181189
}

pkg/runtime/interfaces.go

+6-2
Original file line numberDiff line numberDiff line change
@@ -61,9 +61,9 @@ type Interface interface {
6161
// IsResourceReady returns true if the resource is ready, and false otherwise.
6262
IsResourceReady(resourceID string) (bool, string, error)
6363

64-
// WantToCreateResource returns true if all the condition expressions return true
64+
// ReadyToProcessResource returns true if all the condition expressions return true
6565
// if not it will add itself to the ignored resources
66-
WantToCreateResource(resourceID string) (bool, error)
66+
ReadyToProcessResource(resourceID string) (bool, error)
6767

6868
// IgnoreResource ignores resource that has a condition expressison that evaluated
6969
// to false
@@ -116,6 +116,10 @@ type ResourceDescriptor interface {
116116
// IsNamespaced returns true if the resource is namespaced, and false if it's
117117
// cluster-scoped.
118118
IsNamespaced() bool
119+
120+
// IsReadOnly returns true if the resource is marked as read only
121+
// This is used for external references
122+
IsReadOnly() bool
119123
}
120124

121125
// Resource extends `ResourceDescriptor` to include the actual resource data.

pkg/runtime/runtime.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -519,9 +519,9 @@ func (rt *ResourceGraphDefinitionRuntime) areDependenciesIgnored(resourceID stri
519519
return false
520520
}
521521

522-
// WantToCreateResource returns true if all the condition expressions return true
522+
// ReadyToProcessResource returns true if all the condition expressions return true
523523
// if not it will add itself to the ignored resources
524-
func (rt *ResourceGraphDefinitionRuntime) WantToCreateResource(resourceID string) (bool, error) {
524+
func (rt *ResourceGraphDefinitionRuntime) ReadyToProcessResource(resourceID string) (bool, error) {
525525
if rt.areDependenciesIgnored(resourceID) {
526526
return false, nil
527527
}

pkg/runtime/runtime_test.go

+24-7
Original file line numberDiff line numberDiff line change
@@ -2228,7 +2228,11 @@ func Test_IsResourceReady(t *testing.T) {
22282228
{
22292229
name: "multiple expressions all true",
22302230
resource: newTestResource(
2231-
withReadyExpressions([]string{"test.status.ready", "test.status.healthy && test.status.count > 10", "test.status.count > 5"}),
2231+
withReadyExpressions([]string{
2232+
"test.status.ready",
2233+
"test.status.healthy && test.status.count > 10",
2234+
"test.status.count > 5",
2235+
}),
22322236
),
22332237
resolvedObject: map[string]interface{}{
22342238
"status": map[string]interface{}{
@@ -2280,7 +2284,7 @@ func Test_IsResourceReady(t *testing.T) {
22802284
})
22812285
}
22822286
}
2283-
func Test_WantToCreateResource(t *testing.T) {
2287+
func Test_ReadyToProcessResource(t *testing.T) {
22842288
tests := []struct {
22852289
name string
22862290
resource Resource
@@ -2368,25 +2372,25 @@ func Test_WantToCreateResource(t *testing.T) {
23682372
},
23692373
}
23702374

2371-
got, err := rt.WantToCreateResource("test")
2375+
got, err := rt.ReadyToProcessResource("test")
23722376
if tt.wantErr {
23732377
if err == nil {
2374-
t.Error("WantToCreateResource() expected error, got none")
2378+
t.Error("ReadyToProcessResource() expected error, got none")
23752379
}
23762380
return
23772381
}
23782382
if tt.wantSkip {
23792383
if err == nil || !strings.Contains(err.Error(), "Skipping resource creation due to condition") {
2380-
t.Errorf("WantToCreateResource() expected skip message, got %v", err)
2384+
t.Errorf("ReadyToProcessResource() expected skip message, got %v", err)
23812385
}
23822386
return
23832387
}
23842388
if err != nil {
2385-
t.Errorf("WantToCreateResource() unexpected error = %v", err)
2389+
t.Errorf("ReadyToProcessResource() unexpected error = %v", err)
23862390
return
23872391
}
23882392
if got != tt.want {
2389-
t.Errorf("WantToCreateResource() = %v, want %v", got, tt.want)
2393+
t.Errorf("ReadyToProcessResource() = %v, want %v", got, tt.want)
23902394
}
23912395
})
23922396
}
@@ -2613,6 +2617,7 @@ type mockResource struct {
26132617
conditions []string
26142618
topLevelFields []string
26152619
namespaced bool
2620+
readOnly bool
26162621
obj *unstructured.Unstructured
26172622
}
26182623

@@ -2656,6 +2661,10 @@ func (m *mockResource) Unstructured() *unstructured.Unstructured {
26562661
return m.obj
26572662
}
26582663

2664+
func (m *mockResource) IsReadOnly() bool {
2665+
return m.readOnly
2666+
}
2667+
26592668
type mockResourceOption func(*mockResource)
26602669

26612670
/* func withGVR(group, version, resource string) mockResourceOption {
@@ -2686,6 +2695,14 @@ func withReadyExpressions(exprs []string) mockResourceOption {
26862695
}
26872696
}
26882697

2698+
/*
2699+
func withReadOnly(ro bool) mockResourceOption {
2700+
return func(m *mockResource) {
2701+
m.readOnly = ro
2702+
}
2703+
}
2704+
*/
2705+
26892706
func withConditions(conditions []string) mockResourceOption {
26902707
return func(m *mockResource) {
26912708
m.conditions = conditions

0 commit comments

Comments
 (0)