Skip to content

Commit

Permalink
merge everything
Browse files Browse the repository at this point in the history
  • Loading branch information
sbueringer committed Jun 13, 2022
1 parent 39d5a32 commit 42305d7
Show file tree
Hide file tree
Showing 10 changed files with 34 additions and 34 deletions.
5 changes: 3 additions & 2 deletions api/v1beta1/clusterclass_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -342,7 +342,7 @@ type ClusterClassPatch struct {
Definitions []PatchDefinition `json:"definitions,omitempty"`

// External defines an external patch.
// FIXME(sbueringer): internal or external only => validate
// FIXME(sbueringer): implement validation that exactly one of Definitions/External is set
// +optional
External *ExternalPatchDefinition `json:"external,omitempty"`
}
Expand Down Expand Up @@ -449,8 +449,9 @@ type JSONPatchValue struct {

// ExternalPatchDefinition defines an external patch.
type ExternalPatchDefinition struct {
// FIXME(sbueringer): implement validation that at least one of them is set if parent exists
// GenerateExtension references an extension which is called to generate patches.
// FIXME(sbueringer): Q: Fine that GenerateExtension is optional as well?
// TBD here: https://vmware.slack.com/archives/C02TJ7SNQU8/p1654863214046059
// +optional
GenerateExtension *string `json:"generateExtension,omitempty"`

Expand Down
4 changes: 2 additions & 2 deletions api/v1beta1/zz_generated.openapi.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

10 changes: 6 additions & 4 deletions config/crd/bases/cluster.x-k8s.io_clusterclasses.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -796,12 +796,14 @@ spec:
type: string
external:
description: 'External defines an external patch. FIXME(sbueringer):
internal or external only => validate'
implement validation that exactly one of Definitions/External
is set'
properties:
generateExtension:
description: 'GenerateExtension references an extension
which is called to generate patches. FIXME(sbueringer):
Q: Fine that GenerateExtension is optional as well?'
description: 'FIXME(sbueringer): implement validation that
at least one of them is set if parent exists GenerateExtension
references an extension which is called to generate patches.
TBD here: https://vmware.slack.com/archives/C02TJ7SNQU8/p1654863214046059'
type: string
validateExtension:
description: GenerateExtension references an extension which
Expand Down
2 changes: 1 addition & 1 deletion controllers/alias.go
Original file line number Diff line number Diff line change
Expand Up @@ -148,8 +148,8 @@ func (r *ClusterTopologyReconciler) SetupWithManager(ctx context.Context, mgr ct
return (&clustertopologycontroller.Reconciler{
Client: r.Client,
APIReader: r.APIReader,
UnstructuredCachingClient: r.UnstructuredCachingClient,
RuntimeClient: r.RuntimeClient,
UnstructuredCachingClient: r.UnstructuredCachingClient,
WatchFilterValue: r.WatchFilterValue,
}).SetupWithManager(ctx, mgr, options)
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,6 @@ func (r *Reconciler) SetupWithManager(ctx context.Context, mgr ctrl.Manager, opt

// SetupForDryRun prepares the Reconciler for a dry run execution.
func (r *Reconciler) SetupForDryRun(recorder record.EventRecorder) {
// FIXME(sbueringer) this won't work without a real registry.
r.patchEngine = patches.NewEngine(r.RuntimeClient)
r.recorder = recorder
r.patchHelperFactory = dryRunPatchHelperFactory(r.Client)
Expand Down
25 changes: 11 additions & 14 deletions internal/controllers/topology/cluster/patches/engine.go
Original file line number Diff line number Diff line change
Expand Up @@ -82,17 +82,11 @@ func (e *engine) Apply(ctx context.Context, blueprint *scope.ClusterBlueprint, d
log.V(5).Infof("Applying patch to templates")

// Create patch generator for the current patch.
generator, err := createPatchGenerator(e.runtimeClient, &clusterClassPatch)
generator, err := createPatchGenerator(e.runtimeClient, desired.Cluster, &clusterClassPatch)
if err != nil {
return err
}

// Continue if no generator has been created. This can happen if an external config
// does not contain a generate extension.
if generator == nil {
continue
}

// Generate patches.
// NOTE: All the partial patches accumulate on top of the request, so the
// patch generator in the next iteration of the loop will get the modified
Expand Down Expand Up @@ -124,11 +118,11 @@ func (e *engine) Apply(ctx context.Context, blueprint *scope.ClusterBlueprint, d

log.V(5).Infof("Validating topology")

validator := external.NewValidator(e.runtimeClient, &clusterClassPatch)
validator := external.NewValidator(e.runtimeClient, desired.Cluster, &clusterClassPatch)

_, err := validator.Validate(ctx, validationRequest)
if err != nil {
return errors.Wrapf(err, "template validation %q failed", clusterClassPatch.Name)
return errors.Wrapf(err, "validation of patch %q failed", clusterClassPatch.Name)
}
}

Expand Down Expand Up @@ -263,17 +257,20 @@ func lookupMDTopology(topology *clusterv1.Topology, mdTopologyName string) (*clu
// createPatchGenerator creates a patch generator for the given patch.
// NOTE: Currently only inline JSON patches are supported; in the future we will add
// external patches as well.
func createPatchGenerator(runtimeClient runtimeclient.Client, patch *clusterv1.ClusterClassPatch) (api.Generator, error) {
func createPatchGenerator(runtimeClient runtimeclient.Client, cluster *clusterv1.Cluster, patch *clusterv1.ClusterClassPatch) (api.Generator, error) {
// Return a jsonPatchGenerator if there are PatchDefinitions in the patch.
if len(patch.Definitions) > 0 {
return inline.New(patch), nil
}
// Return an externalPatchGenerator if there is an external configuration in the patch.
if patch.External != nil {
if patch.External.GenerateExtension == nil {
return nil, nil
if patch.External != nil && patch.External.GenerateExtension != nil {
// FIXME(sbueringer): let's check for the RuntimeSDK feature flag:
// 1. in the validating webhook
// 2. here
if runtimeClient == nil {
return nil, errors.Errorf("failed to create patch generator for patch %q: runtimeClient is not set up", patch.Name)
}
return external.New(runtimeClient, patch), nil
return external.New(runtimeClient, cluster, patch), nil
}

return nil, errors.Errorf("failed to create patch generator for patch %q", patch.Name)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -30,20 +30,21 @@ import (
type externalPatchGenerator struct {
runtimeClient runtimeclient.Client
patch *clusterv1.ClusterClassPatch
cluster *clusterv1.Cluster
}

// New returns a new external Generator from a given ClusterClassPatch object.
func New(runtimeClient runtimeclient.Client, patch *clusterv1.ClusterClassPatch) api.Generator {
func New(runtimeClient runtimeclient.Client, cluster *clusterv1.Cluster, patch *clusterv1.ClusterClassPatch) api.Generator {
return &externalPatchGenerator{
runtimeClient: runtimeClient,
cluster: cluster,
patch: patch,
}
}

func (e externalPatchGenerator) Generate(ctx context.Context, req *runtimehooksv1.GeneratePatchesRequest) (*runtimehooksv1.GeneratePatchesResponse, error) {
resp := &runtimehooksv1.GeneratePatchesResponse{}
// FIXME(sbueringer): CallExtension and CallAllExtensions should check if registry is ready internally.
err := e.runtimeClient.CallExtension(ctx, runtimehooksv1.GeneratePatches, *e.patch.External.GenerateExtension, req, resp)
err := e.runtimeClient.CallExtension(ctx, runtimehooksv1.GeneratePatches, e.cluster, *e.patch.External.GenerateExtension, req, resp)
if err != nil {
return nil, err
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -30,19 +30,21 @@ import (
type externalValidator struct {
runtimeClient runtimeclient.Client
patch *clusterv1.ClusterClassPatch
cluster *clusterv1.Cluster
}

// NewValidator returns a new external Validator from a given ClusterClassPatch object.
func NewValidator(runtimeClient runtimeclient.Client, patch *clusterv1.ClusterClassPatch) api.Validator {
func NewValidator(runtimeClient runtimeclient.Client, cluster *clusterv1.Cluster, patch *clusterv1.ClusterClassPatch) api.Validator {
return &externalValidator{
runtimeClient: runtimeClient,
cluster: cluster,
patch: patch,
}
}

func (e externalValidator) Validate(ctx context.Context, req *runtimehooksv1.ValidateTopologyRequest) (*runtimehooksv1.ValidateTopologyResponse, error) {
resp := &runtimehooksv1.ValidateTopologyResponse{}
err := e.runtimeClient.CallExtension(ctx, runtimehooksv1.ValidateTopology, *e.patch.External.ValidateExtension, req, resp)
err := e.runtimeClient.CallExtension(ctx, runtimehooksv1.ValidateTopology, e.cluster, *e.patch.External.ValidateExtension, req, resp)
if err != nil {
return nil, err
}
Expand Down
6 changes: 2 additions & 4 deletions main.go
Original file line number Diff line number Diff line change
Expand Up @@ -313,14 +313,12 @@ func setupReconcilers(ctx context.Context, mgr ctrl.Manager) {
os.Exit(1)
}

var runtimeRegistry runtimeregistry.ExtensionRegistry
var runtimeClient runtimeclient.Client
if feature.Gates.Enabled(feature.RuntimeSDK) {
// This is the creation of the single RuntimeExtension registry for the controller.
runtimeRegistry = runtimeregistry.New()
runtimeClient = runtimeclient.New(runtimeclient.Options{
Catalog: catalog,
Registry: runtimeRegistry,
Registry: runtimeregistry.New(),
Client: mgr.GetClient(),
})
}
Expand Down Expand Up @@ -354,8 +352,8 @@ func setupReconcilers(ctx context.Context, mgr ctrl.Manager) {
if err := (&controllers.ClusterTopologyReconciler{
Client: mgr.GetClient(),
APIReader: mgr.GetAPIReader(),
UnstructuredCachingClient: unstructuredCachingClient,
RuntimeClient: runtimeClient,
UnstructuredCachingClient: unstructuredCachingClient,
WatchFilterValue: watchFilterValue,
}).SetupWithManager(ctx, mgr, concurrency(clusterTopologyConcurrency)); err != nil {
setupLog.Error(err, "unable to create controller", "controller", "ClusterTopology")
Expand Down
2 changes: 1 addition & 1 deletion poc/local/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ k apply -f ./poc/test/local/secure-infra.yaml

# fetch certificate
mkdir -p /tmp/k8s-webhook-server/serving-certs
for f in $(kubectl get secret my-local-extension-cert -o json | jq '.data | keys | .[]' -r); do
for f in $(kubectl get secret webhook-service-cert -o json | jq '.data | keys | .[]' -r); do
kubectl get secret my-local-extension-cert -o json | jq '.data["'$f'"]' -r | base64 -d > "/tmp/k8s-webhook-server/serving-certs/$f"
done

Expand Down

0 comments on commit 42305d7

Please sign in to comment.