Skip to content

Commit aaa5c94

Browse files
authored
Merge pull request #188 from mortent/SimplifyInfoHelper
Simplify the InfoHelper and ApplyTask
2 parents 54e401f + ae77d32 commit aaa5c94

File tree

9 files changed

+164
-62
lines changed

9 files changed

+164
-62
lines changed

pkg/apply/applier.go

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -292,11 +292,18 @@ func (a *Applier) Run(ctx context.Context, objects []*resource.Info, options Opt
292292
return
293293
}
294294

295+
mapper, err := a.factory.ToRESTMapper()
296+
if err != nil {
297+
handleError(eventChannel, err)
298+
return
299+
}
300+
295301
// Fetch the queue (channel) of tasks that should be executed.
296302
taskQueue := (&solver.TaskQueueSolver{
297303
ApplyOptions: a.ApplyOptions,
298304
PruneOptions: a.PruneOptions,
299305
InfoHelper: a.infoHelperFactoryFunc(),
306+
Mapper: mapper,
300307
}).BuildTaskQueue(resourceObjects, solver.Options{
301308
ReconcileTimeout: options.ReconcileTimeout,
302309
Prune: !options.NoPrune,

pkg/apply/applier_test.go

Lines changed: 0 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -801,11 +801,3 @@ func (f *fakeInfoHelper) getClient(gv schema.GroupVersion) (resource.RESTClient,
801801
}
802802
return f.factory.Client, nil
803803
}
804-
805-
func (f *fakeInfoHelper) ResetRESTMapper() error {
806-
return nil
807-
}
808-
809-
func (f *fakeInfoHelper) ToRESTMapper() (meta.RESTMapper, error) {
810-
return f.factory.ToRESTMapper()
811-
}

pkg/apply/info/info_helper.go

Lines changed: 0 additions & 25 deletions
Original file line numberDiff line numberDiff line change
@@ -4,14 +4,10 @@
44
package info
55

66
import (
7-
"fmt"
8-
"reflect"
9-
107
"k8s.io/apimachinery/pkg/api/meta"
118
"k8s.io/apimachinery/pkg/runtime/schema"
129
"k8s.io/cli-runtime/pkg/resource"
1310
"k8s.io/client-go/rest"
14-
"k8s.io/client-go/restmapper"
1511
"k8s.io/kubectl/pkg/cmd/util"
1612
)
1713

@@ -21,13 +17,6 @@ type InfoHelper interface {
2117
// objects. This must be called at a time when all needed resource
2218
// types are available in the RESTMapper.
2319
UpdateInfos(infos []*resource.Info) error
24-
25-
// ResetRESTMapper resets the state of the RESTMapper so any
26-
// added resource types in the cluster will be picked up.
27-
ResetRESTMapper() error
28-
29-
// ToRESTMapper returns a RESTMapper
30-
ToRESTMapper() (meta.RESTMapper, error)
3120
}
3221

3322
func NewInfoHelper(factory util.Factory, namespace string) *infoHelper {
@@ -68,20 +57,6 @@ func (ih *infoHelper) ToRESTMapper() (meta.RESTMapper, error) {
6857
return ih.factory.ToRESTMapper()
6958
}
7059

71-
func (ih *infoHelper) ResetRESTMapper() error {
72-
mapper, err := ih.factory.ToRESTMapper()
73-
if err != nil {
74-
return err
75-
}
76-
fv := reflect.ValueOf(mapper).FieldByName("RESTMapper")
77-
ddRESTMapper, ok := fv.Interface().(*restmapper.DeferredDiscoveryRESTMapper)
78-
if !ok {
79-
return fmt.Errorf("unexpected RESTMapper type")
80-
}
81-
ddRESTMapper.Reset()
82-
return nil
83-
}
84-
8560
func (ih *infoHelper) getClient(gv schema.GroupVersion) (*rest.RESTClient, error) {
8661
cfg, err := ih.factory.ToRESTConfig()
8762
if err != nil {

pkg/apply/solver/solver.go

Lines changed: 7 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ import (
1919

2020
v1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
2121
"k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1beta1"
22+
"k8s.io/apimachinery/pkg/api/meta"
2223
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
2324
"k8s.io/apimachinery/pkg/runtime/schema"
2425
"k8s.io/cli-runtime/pkg/resource"
@@ -36,6 +37,7 @@ type TaskQueueSolver struct {
3637
ApplyOptions *apply.ApplyOptions
3738
PruneOptions *prune.PruneOptions
3839
InfoHelper info.InfoHelper
40+
Mapper meta.RESTMapper
3941
}
4042

4143
type Options struct {
@@ -68,13 +70,16 @@ func (t *TaskQueueSolver) BuildTaskQueue(ro resourceObjects,
6870
ApplyOptions: t.ApplyOptions,
6971
DryRun: o.DryRun,
7072
InfoHelper: t.InfoHelper,
73+
Mapper: t.Mapper,
7174
})
7275
if !o.DryRun {
7376
tasks = append(tasks, taskrunner.NewWaitTask(
7477
object.InfosToObjMetas(crdSplitRes.crds),
7578
taskrunner.AllCurrent,
7679
1*time.Minute),
77-
)
80+
&task.ResetRESTMapperTask{
81+
Mapper: t.Mapper,
82+
})
7883
}
7984
remainingInfos = crdSplitRes.after
8085
}
@@ -86,6 +91,7 @@ func (t *TaskQueueSolver) BuildTaskQueue(ro resourceObjects,
8691
ApplyOptions: t.ApplyOptions,
8792
DryRun: o.DryRun,
8893
InfoHelper: t.InfoHelper,
94+
Mapper: t.Mapper,
8995
},
9096
&task.SendEventTask{
9197
Event: event.Event{

pkg/apply/solver/solver_test.go

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ import (
1616
"sigs.k8s.io/cli-utils/pkg/apply/task"
1717
"sigs.k8s.io/cli-utils/pkg/apply/taskrunner"
1818
"sigs.k8s.io/cli-utils/pkg/object"
19+
"sigs.k8s.io/cli-utils/pkg/testutil"
1920
)
2021

2122
var (
@@ -149,6 +150,7 @@ func TestTaskQueueSolver_BuildTaskQueue(t *testing.T) {
149150
object.InfoToObjMeta(crdInfo),
150151
},
151152
taskrunner.AllCurrent, 1*time.Second),
153+
&task.ResetRESTMapperTask{},
152154
&task.ApplyTask{
153155
Objects: []*resource.Info{
154156
depInfo,
@@ -194,6 +196,7 @@ func TestTaskQueueSolver_BuildTaskQueue(t *testing.T) {
194196
tqs := TaskQueueSolver{
195197
ApplyOptions: applyOptions,
196198
PruneOptions: pruneOptions,
199+
Mapper: testutil.NewFakeRESTMapper(),
197200
}
198201

199202
tq := tqs.BuildTaskQueue(&fakeResourceObjects{

pkg/apply/task/apply_task.go

Lines changed: 2 additions & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ import (
2020
type ApplyTask struct {
2121
ApplyOptions applyOptions
2222
InfoHelper info.InfoHelper
23+
Mapper meta.RESTMapper
2324
Objects []*resource.Info
2425
CRDs []*resource.Info
2526
DryRun bool
@@ -113,11 +114,6 @@ func (a *ApplyTask) Start(taskContext *taskrunner.TaskContext) {
113114
gen := acc.GetGeneration()
114115
taskContext.ResourceApplied(id, gen)
115116
}
116-
err = a.InfoHelper.ResetRESTMapper()
117-
if err != nil {
118-
a.sendTaskResult(taskContext, err)
119-
return
120-
}
121117
a.sendTaskResult(taskContext, nil)
122118
}()
123119
}
@@ -149,18 +145,13 @@ func (a *ApplyTask) filterCRsWithCRDInSet(objects []*resource.Info) ([]*resource
149145
var objs []*resource.Info
150146
var objsWithCRD []*resource.Info
151147

152-
mapper, err := a.InfoHelper.ToRESTMapper()
153-
if err != nil {
154-
return objs, objsWithCRD, err
155-
}
156-
157148
crdsInfo := buildCRDsInfo(a.CRDs)
158149
for _, obj := range objects {
159150
gvk := obj.Object.GetObjectKind().GroupVersionKind()
160151

161152
// First check if we find the type in the RESTMapper.
162153
//TODO: Maybe we do care if there is a new version of the CRD?
163-
_, err := mapper.RESTMapping(gvk.GroupKind())
154+
_, err := a.Mapper.RESTMapping(gvk.GroupKind())
164155
if err != nil && !meta.IsNoMatchError(err) {
165156
return objs, objsWithCRD, err
166157
}

pkg/apply/task/apply_task_test.go

Lines changed: 5 additions & 17 deletions
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,6 @@ import (
88
"testing"
99

1010
"gotest.tools/assert"
11-
"k8s.io/apimachinery/pkg/api/meta"
1211
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
1312
"k8s.io/apimachinery/pkg/runtime/schema"
1413
"k8s.io/cli-runtime/pkg/resource"
@@ -231,11 +230,10 @@ func TestApplyTask_DryRun(t *testing.T) {
231230
applyTask := &ApplyTask{
232231
ApplyOptions: applyOptions,
233232
Objects: tc.infos,
234-
InfoHelper: &fakeInfoHelper{
235-
restMapper: restMapper,
236-
},
237-
DryRun: true,
238-
CRDs: tc.crds,
233+
InfoHelper: &fakeInfoHelper{},
234+
Mapper: restMapper,
235+
DryRun: true,
236+
CRDs: tc.crds,
239237
}
240238

241239
var events []event.Event
@@ -307,18 +305,8 @@ func (f *fakeApplyOptions) SetObjects(objects []*resource.Info) {
307305
f.objects = objects
308306
}
309307

310-
type fakeInfoHelper struct {
311-
restMapper meta.RESTMapper
312-
}
308+
type fakeInfoHelper struct{}
313309

314310
func (f *fakeInfoHelper) UpdateInfos([]*resource.Info) error {
315311
return nil
316312
}
317-
318-
func (f *fakeInfoHelper) ResetRESTMapper() error {
319-
return nil
320-
}
321-
322-
func (f *fakeInfoHelper) ToRESTMapper() (meta.RESTMapper, error) {
323-
return f.restMapper, nil
324-
}

pkg/apply/task/resetmapper_task.go

Lines changed: 61 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
// Copyright 2020 The Kubernetes Authors.
2+
// SPDX-License-Identifier: Apache-2.0
3+
4+
package task
5+
6+
import (
7+
"fmt"
8+
"reflect"
9+
10+
"k8s.io/apimachinery/pkg/api/meta"
11+
"k8s.io/client-go/restmapper"
12+
"sigs.k8s.io/cli-utils/pkg/apply/taskrunner"
13+
)
14+
15+
// ResetRESTMapperTask resets the provided RESTMapper.
16+
type ResetRESTMapperTask struct {
17+
Mapper meta.RESTMapper
18+
}
19+
20+
// Start creates a new goroutine that will unwrap the provided RESTMapper
21+
// to get the underlying DeferredDiscoveryRESTMapper and then reset it. It
22+
// will send a TaskResult on the taskChannel to signal that the task has
23+
// been completed.
24+
func (r *ResetRESTMapperTask) Start(taskContext *taskrunner.TaskContext) {
25+
go func() {
26+
ddRESTMapper, err := extractDeferredDiscoveryRESTMapper(r.Mapper)
27+
if err != nil {
28+
r.sendTaskResult(taskContext, err)
29+
return
30+
}
31+
ddRESTMapper.Reset()
32+
r.sendTaskResult(taskContext, nil)
33+
}()
34+
}
35+
36+
// extractDeferredDiscoveryRESTMapper unwraps the provided RESTMapper
37+
// interface to get access to the underlying DeferredDiscoveryRESTMapper
38+
// that can be reset.
39+
func extractDeferredDiscoveryRESTMapper(mapper meta.RESTMapper) (*restmapper.DeferredDiscoveryRESTMapper,
40+
error) {
41+
val := reflect.ValueOf(mapper)
42+
if val.Type().Kind() != reflect.Struct {
43+
return nil, fmt.Errorf("unexpected RESTMapper type: %s", val.Type().String())
44+
}
45+
fv := val.FieldByName("RESTMapper")
46+
ddRESTMapper, ok := fv.Interface().(*restmapper.DeferredDiscoveryRESTMapper)
47+
if !ok {
48+
return nil, fmt.Errorf("unexpected RESTMapper type")
49+
}
50+
return ddRESTMapper, nil
51+
}
52+
53+
func (r *ResetRESTMapperTask) sendTaskResult(taskContext *taskrunner.TaskContext, err error) {
54+
taskContext.TaskChannel() <- taskrunner.TaskResult{
55+
Err: err,
56+
}
57+
}
58+
59+
// ClearTimeout doesn't do anything as ResetRESTMapperTask doesn't support
60+
// timeouts.
61+
func (r *ResetRESTMapperTask) ClearTimeout() {}
Lines changed: 79 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
// Copyright 2020 The Kubernetes Authors.
2+
// SPDX-License-Identifier: Apache-2.0
3+
4+
package task
5+
6+
import (
7+
"testing"
8+
9+
"github.com/stretchr/testify/assert"
10+
"k8s.io/apimachinery/pkg/api/meta"
11+
"k8s.io/client-go/discovery"
12+
"k8s.io/client-go/restmapper"
13+
"sigs.k8s.io/cli-utils/pkg/apply/event"
14+
"sigs.k8s.io/cli-utils/pkg/apply/taskrunner"
15+
"sigs.k8s.io/cli-utils/pkg/testutil"
16+
)
17+
18+
func TestResetRESTMapperTask(t *testing.T) {
19+
testCases := map[string]struct {
20+
toRESTMapper func() (meta.RESTMapper, *fakeCachedDiscoveryClient)
21+
expectErr bool
22+
expectedErrMessage string
23+
}{
24+
"correct wrapped RESTMapper": {
25+
toRESTMapper: func() (meta.RESTMapper, *fakeCachedDiscoveryClient) {
26+
discoveryClient := &fakeCachedDiscoveryClient{}
27+
ddRESTMapper := restmapper.NewDeferredDiscoveryRESTMapper(discoveryClient)
28+
return restmapper.NewShortcutExpander(ddRESTMapper, discoveryClient), discoveryClient
29+
},
30+
expectErr: false,
31+
},
32+
"incorrect wrapped RESTMapper": {
33+
toRESTMapper: func() (meta.RESTMapper, *fakeCachedDiscoveryClient) {
34+
return testutil.NewFakeRESTMapper(), nil
35+
},
36+
expectErr: true,
37+
expectedErrMessage: "unexpected RESTMapper type",
38+
},
39+
}
40+
41+
for tn, tc := range testCases {
42+
t.Run(tn, func(t *testing.T) {
43+
eventChannel := make(chan event.Event)
44+
defer close(eventChannel)
45+
taskContext := taskrunner.NewTaskContext(eventChannel)
46+
47+
mapper, discoveryClient := tc.toRESTMapper()
48+
49+
resetRESTMapperTask := &ResetRESTMapperTask{
50+
Mapper: mapper,
51+
}
52+
53+
resetRESTMapperTask.Start(taskContext)
54+
55+
result := <-taskContext.TaskChannel()
56+
57+
if tc.expectErr {
58+
assert.Error(t, result.Err)
59+
assert.Contains(t, result.Err.Error(), tc.expectedErrMessage)
60+
return
61+
}
62+
63+
assert.True(t, discoveryClient.invalidated)
64+
})
65+
}
66+
}
67+
68+
type fakeCachedDiscoveryClient struct {
69+
discovery.DiscoveryInterface
70+
invalidated bool
71+
}
72+
73+
func (d *fakeCachedDiscoveryClient) Fresh() bool {
74+
return true
75+
}
76+
77+
func (d *fakeCachedDiscoveryClient) Invalidate() {
78+
d.invalidated = true
79+
}

0 commit comments

Comments
 (0)