package user import ( "context" "fmt" "time" virtuslabv1alpha1 "github.com/VirtusLab/jenkins-operator/pkg/apis/virtuslab/v1alpha1" "github.com/VirtusLab/jenkins-operator/pkg/controller/jenkins/backup" jenkinsclient "github.com/VirtusLab/jenkins-operator/pkg/controller/jenkins/client" "github.com/VirtusLab/jenkins-operator/pkg/controller/jenkins/configuration/base/resources" "github.com/VirtusLab/jenkins-operator/pkg/controller/jenkins/configuration/user/seedjobs" "github.com/VirtusLab/jenkins-operator/pkg/controller/jenkins/constants" "github.com/VirtusLab/jenkins-operator/pkg/controller/jenkins/groovy" "github.com/VirtusLab/jenkins-operator/pkg/controller/jenkins/jobs" "github.com/go-logr/logr" corev1 "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/types" k8s "sigs.k8s.io/controller-runtime/pkg/client" "sigs.k8s.io/controller-runtime/pkg/reconcile" ) // ReconcileUserConfiguration defines values required for Jenkins user configuration type ReconcileUserConfiguration struct { k8sClient k8s.Client jenkinsClient jenkinsclient.Jenkins logger logr.Logger jenkins *virtuslabv1alpha1.Jenkins } // New create structure which takes care of user configuration func New(k8sClient k8s.Client, jenkinsClient jenkinsclient.Jenkins, logger logr.Logger, jenkins *virtuslabv1alpha1.Jenkins) *ReconcileUserConfiguration { return &ReconcileUserConfiguration{ k8sClient: k8sClient, jenkinsClient: jenkinsClient, logger: logger, jenkins: jenkins, } } // Reconcile it's a main reconciliation loop for user supplied configuration func (r *ReconcileUserConfiguration) Reconcile() (reconcile.Result, error) { backupManager := backup.New(r.jenkins, r.k8sClient, r.logger, r.jenkinsClient) if err := backupManager.EnsureRestoreJob(); err != nil { return reconcile.Result{}, err } result, err := backupManager.RestoreBackup() if err != nil { return reconcile.Result{}, err } if result.Requeue { return result, nil } // reconcile seed jobs result, err = r.ensureSeedJobs() if err != nil { return reconcile.Result{}, err } if result.Requeue { return result, nil } result, err = r.ensureUserConfiguration(r.jenkinsClient) if err != nil { return reconcile.Result{}, err } if result.Requeue { return result, nil } err = backupManager.EnsureBackupJob() if err != nil { return reconcile.Result{}, err } return reconcile.Result{}, nil } func (r *ReconcileUserConfiguration) ensureSeedJobs() (reconcile.Result, error) { seedJobs := seedjobs.New(r.jenkinsClient, r.k8sClient, r.logger) done, err := seedJobs.EnsureSeedJobs(r.jenkins) if err != nil { // build failed and can be recovered - retry build and requeue reconciliation loop with timeout if err == jobs.ErrorBuildFailed { return reconcile.Result{Requeue: true, RequeueAfter: time.Second * 10}, nil } // build failed and cannot be recovered if err == jobs.ErrorUnrecoverableBuildFailed { return reconcile.Result{}, nil } // unexpected error - requeue reconciliation loop return reconcile.Result{}, err } // build not finished yet - requeue reconciliation loop with timeout if !done { return reconcile.Result{Requeue: true, RequeueAfter: time.Second * 10}, nil } return reconcile.Result{}, nil } func (r *ReconcileUserConfiguration) ensureUserConfiguration(jenkinsClient jenkinsclient.Jenkins) (reconcile.Result, error) { groovyClient := groovy.New(jenkinsClient, r.k8sClient, r.logger, fmt.Sprintf("%s-user-configuration", constants.OperatorName), resources.JenkinsUserConfigurationVolumePath) err := groovyClient.ConfigureGroovyJob() if err != nil { return reconcile.Result{}, err } configuration := &corev1.ConfigMap{} namespaceName := types.NamespacedName{Namespace: r.jenkins.Namespace, Name: resources.GetUserConfigurationConfigMapName(r.jenkins)} err = r.k8sClient.Get(context.TODO(), namespaceName, configuration) if err != nil { return reconcile.Result{}, err } done, err := groovyClient.EnsureGroovyJob(configuration.Data, r.jenkins) if err != nil { return reconcile.Result{}, err } if !done { return reconcile.Result{Requeue: true, RequeueAfter: time.Second * 10}, nil } return reconcile.Result{}, nil }