Skip to content

Commit

Permalink
Add importer into registraiton
Browse files Browse the repository at this point in the history
Signed-off-by: Jian Qiu <[email protected]>
  • Loading branch information
qiujian16 committed Dec 10, 2024
1 parent ddc5024 commit 362c27e
Show file tree
Hide file tree
Showing 4 changed files with 561 additions and 0 deletions.
291 changes: 291 additions & 0 deletions pkg/registration/hub/importer/importer.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,291 @@
package importer

import (
"context"
"fmt"

"github.com/openshift/api"
"github.com/openshift/library-go/pkg/controller/factory"
"github.com/openshift/library-go/pkg/operator/events"
"github.com/openshift/library-go/pkg/operator/resource/resourceapply"
"github.com/openshift/library-go/pkg/operator/resource/resourcemerge"
appsv1 "k8s.io/api/apps/v1"
apiextensionsv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
apiextensionsclient "k8s.io/apiextensions-apiserver/pkg/client/clientset/clientset"
"k8s.io/apimachinery/pkg/api/equality"
"k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/api/meta"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/runtime/serializer"
utilerrors "k8s.io/apimachinery/pkg/util/errors"
utilruntime "k8s.io/apimachinery/pkg/util/runtime"
"k8s.io/client-go/kubernetes"
"k8s.io/klog/v2"
"k8s.io/utils/pointer"

clusterclientset "open-cluster-management.io/api/client/cluster/clientset/versioned"
clusterinformerv1 "open-cluster-management.io/api/client/cluster/informers/externalversions/cluster/v1"
clusterlisterv1 "open-cluster-management.io/api/client/cluster/listers/cluster/v1"
operatorclient "open-cluster-management.io/api/client/operator/clientset/versioned"
v1 "open-cluster-management.io/api/cluster/v1"
operatorv1 "open-cluster-management.io/api/operator/v1"
"open-cluster-management.io/sdk-go/pkg/patcher"

"open-cluster-management.io/ocm/pkg/common/queue"
"open-cluster-management.io/ocm/pkg/operator/helpers/chart"
cloudproviders "open-cluster-management.io/ocm/pkg/registration/hub/importer/providers"
)

var (
genericScheme = runtime.NewScheme()
genericCodecs = serializer.NewCodecFactory(genericScheme)
genericCodec = genericCodecs.UniversalDeserializer()
)

func init() {
utilruntime.Must(api.InstallKube(genericScheme))
utilruntime.Must(apiextensionsv1.AddToScheme(genericScheme))
utilruntime.Must(operatorv1.Install(genericScheme))
}

const operatorNamesapce = "open-cluster-management"

// KlusterletConfigRenderer renders the config for klusterlet chart.
type KlusterletConfigRenderer func(
ctx context.Context, config *chart.KlusterletChartConfig) (*chart.KlusterletChartConfig, error)

type Importer struct {
providers []cloudproviders.Interface
clusterClient clusterclientset.Interface
clusterLister clusterlisterv1.ManagedClusterLister
renders []KlusterletConfigRenderer
patcher patcher.Patcher[*v1.ManagedCluster, v1.ManagedClusterSpec, v1.ManagedClusterStatus]
}

func NewImporter(
renders []KlusterletConfigRenderer,
clusterClient clusterclientset.Interface,
clusterInformer clusterinformerv1.ManagedClusterInformer,
providers []cloudproviders.Interface,
recorder events.Recorder) factory.Controller {
controllerName := "managed-cluster-importer"
syncCtx := factory.NewSyncContext(controllerName, recorder)

i := &Importer{
providers: providers,
clusterClient: clusterClient,
clusterLister: clusterInformer.Lister(),
renders: renders,
patcher: patcher.NewPatcher[
*v1.ManagedCluster, v1.ManagedClusterSpec, v1.ManagedClusterStatus](
clusterClient.ClusterV1().ManagedClusters()),
}

for _, provider := range providers {
provider.Register(syncCtx)
}

return factory.New().WithInformersQueueKeysFunc(queue.QueueKeyByMetaName, clusterInformer.Informer()).
WithSyncContext(syncCtx).WithSync(i.sync).ToController(controllerName, recorder)
}

func (i *Importer) sync(ctx context.Context, syncCtx factory.SyncContext) error {
clusterName := syncCtx.QueueKey()
logger := klog.FromContext(ctx)
logger.V(4).Info("Reconciling key", "clusterName", clusterName)

cluster, err := i.clusterLister.Get(clusterName)
switch {
case errors.IsNotFound(err):
return nil
case err != nil:
return err
}

// If the cluster is imported, skip the reconcile
if meta.IsStatusConditionTrue(cluster.Status.Conditions, "Imported") {
return nil
}

// get provider from the provider list
var provider cloudproviders.Interface
for _, p := range i.providers {
if p.IsManagedClusterOwner(cluster) {
provider = p
}
}
if provider == nil {
logger.Info("provider not found for cluster", "cluster", cluster.Name)
return nil
}

newCluster := cluster.DeepCopy()
newCluster, err = i.reconcile(ctx, logger, syncCtx.Recorder(), provider, newCluster)
updated, updatedErr := i.patcher.PatchStatus(ctx, newCluster, newCluster.Status, cluster.Status)
if updatedErr != nil {
return updatedErr
}
if err != nil {
return err
}
if updated {
syncCtx.Recorder().Eventf(
"ManagedClusterImported", "managed cluster %s is imported", clusterName)
}

return nil
}

func (i *Importer) reconcile(
ctx context.Context,
logger klog.Logger,
recorder events.Recorder,
provider cloudproviders.Interface,
cluster *v1.ManagedCluster) (*v1.ManagedCluster, error) {
config, err := provider.KubeConfig(cluster)
if err != nil {
meta.SetStatusCondition(&cluster.Status.Conditions, metav1.Condition{
Type: "Imported",
Status: metav1.ConditionFalse,
Reason: "KubeConfigGetFailed",
Message: fmt.Sprintf("failed to get kubeconfig. See errors:\n%s",
err.Error()),
})
return cluster, err
}

if config == nil {
meta.SetStatusCondition(&cluster.Status.Conditions, metav1.Condition{
Type: "Imported",
Status: metav1.ConditionFalse,
Reason: "KubeConfigNotFound",
Message: "Secret for kubeconfig is not found.",
})
return cluster, nil
}

// render the klsuterlet chart config
klusterletChartConfig := &chart.KlusterletChartConfig{
CreateNamespace: true,
Klusterlet: chart.KlusterletConfig{
ClusterName: cluster.Name,
},
}
for _, renderer := range i.renders {
klusterletChartConfig, err = renderer(ctx, klusterletChartConfig)
if err != nil {
meta.SetStatusCondition(&cluster.Status.Conditions, metav1.Condition{
Type: "Imported",
Status: metav1.ConditionFalse,
Reason: "ConfigRendererFailed",
Message: fmt.Sprintf("failed to render config. See errors:\n%s",
err.Error()),
})
return cluster, err
}
}
rawManifests, err := chart.RenderKlusterletChart(klusterletChartConfig, operatorNamesapce)
if err != nil {
return cluster, err
}

// build related client
kubeClient, err := kubernetes.NewForConfig(config)
if err != nil {
meta.SetStatusCondition(&cluster.Status.Conditions, metav1.Condition{
Type: "Imported",
Status: metav1.ConditionFalse,
Reason: "ClientBuildErr",
Message: fmt.Sprintf("failed to import the klusterlet. See errors: %s",
err.Error()),
})
return cluster, err
}
hubApiExtensionClient, err := apiextensionsclient.NewForConfig(config)
if err != nil {
return cluster, err
}
operatorClient, err := operatorclient.NewForConfig(config)
if err != nil {
return cluster, err
}

clientHolder := resourceapply.NewKubeClientHolder(kubeClient).
WithAPIExtensionsClient(hubApiExtensionClient)
cache := resourceapply.NewResourceCache()
var results []resourceapply.ApplyResult
for _, manifest := range rawManifests {
requiredObj, _, err := genericCodec.Decode(manifest, nil, nil)
if err != nil {
logger.Error(err, "failed to decode manifest", "manifest", manifest)
return cluster, err
}
result := resourceapply.ApplyResult{}
switch t := requiredObj.(type) {
case *appsv1.Deployment:
result.Result, result.Changed, result.Error = resourceapply.ApplyDeployment(
ctx, kubeClient.AppsV1(), recorder, t, 0)
results = append(results, result)
case *operatorv1.Klusterlet:
result.Result, result.Changed, result.Error = ApplyKlusterlet(
ctx, operatorClient, t)
results = append(results, result)
default:
tempResults := resourceapply.ApplyDirectly(ctx, clientHolder, recorder, cache,
func(name string) ([]byte, error) {
return manifest, nil
},
"manifest")
results = append(results, tempResults...)
}
}

var errs []error
for _, result := range results {
if result.Error != nil {
errs = append(errs, result.Error)
}
}
if len(errs) > 0 {
meta.SetStatusCondition(&cluster.Status.Conditions, metav1.Condition{
Type: "Imported",
Status: metav1.ConditionFalse,
Reason: "ImportFailed",
Message: fmt.Sprintf("failed to import the klusterlet. See errors:\n%s",
err.Error()),
})
} else {
meta.SetStatusCondition(&cluster.Status.Conditions, metav1.Condition{
Type: "Imported",
Status: metav1.ConditionTrue,
Reason: "ImportSucceed",
})
}

return cluster, utilerrors.NewAggregate(errs)
}

func ApplyKlusterlet(ctx context.Context, client operatorclient.Interface, required *operatorv1.Klusterlet) (*operatorv1.Klusterlet, bool, error) {
existing, err := client.OperatorV1().Klusterlets().Get(ctx, required.Name, metav1.GetOptions{})
if errors.IsNotFound(err) {
requiredCopy := required.DeepCopy()
actual, err := client.OperatorV1().Klusterlets().Create(ctx, requiredCopy, metav1.CreateOptions{})
return actual, true, err
}
if err != nil {
return nil, false, err
}

modified := pointer.Bool(false)
existingCopy := existing.DeepCopy()
resourcemerge.EnsureObjectMeta(modified, &existingCopy.ObjectMeta, required.ObjectMeta)

if !*modified && equality.Semantic.DeepEqual(existingCopy.Spec, required.Spec) {
return existingCopy, false, nil
}

existingCopy.Spec = required.Spec
actual, err := client.OperatorV1().Klusterlets().Update(ctx, existingCopy, metav1.UpdateOptions{})
return actual, true, err
}
Loading

0 comments on commit 362c27e

Please sign in to comment.