173 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			173 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Go
		
	
	
	
| package e2e
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"reflect"
 | |
| 	"testing"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/jenkinsci/kubernetes-operator/pkg/apis/jenkinsio/v1alpha1"
 | |
| 	"github.com/jenkinsci/kubernetes-operator/pkg/controller/jenkins/configuration/user/seedjobs"
 | |
| 	"github.com/jenkinsci/kubernetes-operator/pkg/controller/jenkins/plugins"
 | |
| 
 | |
| 	"github.com/bndr/gojenkins"
 | |
| 	framework "github.com/operator-framework/operator-sdk/pkg/test"
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| 	corev1 "k8s.io/api/core/v1"
 | |
| 	"k8s.io/apimachinery/pkg/api/resource"
 | |
| 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | |
| 	"k8s.io/apimachinery/pkg/types"
 | |
| 	"k8s.io/apimachinery/pkg/util/wait"
 | |
| )
 | |
| 
 | |
| func TestConfiguration(t *testing.T) {
 | |
| 	t.Parallel()
 | |
| 	namespace, ctx := setupTest(t)
 | |
| 	// Deletes test namespace
 | |
| 	defer ctx.Cleanup()
 | |
| 
 | |
| 	// base
 | |
| 	jenkins := createJenkinsCR(t, namespace)
 | |
| 	createDefaultLimitsForContainersInNamespace(t, namespace)
 | |
| 	waitForJenkinsBaseConfigurationToComplete(t, jenkins)
 | |
| 
 | |
| 	verifyJenkinsMasterPodAttributes(t, jenkins)
 | |
| 	client := verifyJenkinsAPIConnection(t, jenkins)
 | |
| 	verifyPlugins(t, client, jenkins)
 | |
| 
 | |
| 	// user
 | |
| 	waitForJenkinsUserConfigurationToComplete(t, jenkins)
 | |
| 	verifyJenkinsSeedJobs(t, client, jenkins)
 | |
| }
 | |
| 
 | |
| func createDefaultLimitsForContainersInNamespace(t *testing.T, namespace string) {
 | |
| 	limitRange := &corev1.LimitRange{
 | |
| 		ObjectMeta: metav1.ObjectMeta{
 | |
| 			Name:      "e2e",
 | |
| 			Namespace: namespace,
 | |
| 		},
 | |
| 		Spec: corev1.LimitRangeSpec{
 | |
| 			Limits: []corev1.LimitRangeItem{
 | |
| 				{
 | |
| 					Type: corev1.LimitTypeContainer,
 | |
| 					DefaultRequest: map[corev1.ResourceName]resource.Quantity{
 | |
| 						corev1.ResourceCPU:    resource.MustParse("1"),
 | |
| 						corev1.ResourceMemory: resource.MustParse("1Gi"),
 | |
| 					},
 | |
| 					Default: map[corev1.ResourceName]resource.Quantity{
 | |
| 						corev1.ResourceCPU:    resource.MustParse("4"),
 | |
| 						corev1.ResourceMemory: resource.MustParse("4Gi"),
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	t.Logf("LimitRange %+v", *limitRange)
 | |
| 	if err := framework.Global.Client.Create(context.TODO(), limitRange, nil); err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func verifyJenkinsMasterPodAttributes(t *testing.T, jenkins *v1alpha1.Jenkins) {
 | |
| 	jenkinsPod := getJenkinsMasterPod(t, jenkins)
 | |
| 	jenkins = getJenkins(t, jenkins.Namespace, jenkins.Name)
 | |
| 
 | |
| 	for key, value := range jenkins.Spec.Master.Annotations {
 | |
| 		if jenkinsPod.ObjectMeta.Annotations[key] != value {
 | |
| 			t.Fatalf("Invalid Jenkins pod annotation expected '%+v', actual '%+v'", jenkins.Spec.Master.Annotations, jenkinsPod.ObjectMeta.Annotations)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if jenkinsPod.Spec.Containers[0].Image != jenkins.Spec.Master.Image {
 | |
| 		t.Fatalf("Invalid jenkins pod image expected '%s', actual '%s'", jenkins.Spec.Master.Image, jenkinsPod.Spec.Containers[0].Image)
 | |
| 	}
 | |
| 
 | |
| 	if !reflect.DeepEqual(jenkinsPod.Spec.Containers[0].Resources, jenkins.Spec.Master.Resources) {
 | |
| 		t.Fatalf("Invalid jenkins pod continer resources expected '%+v', actual '%+v'", jenkins.Spec.Master.Resources, jenkinsPod.Spec.Containers[0].Resources)
 | |
| 	}
 | |
| 
 | |
| 	t.Log("Jenkins pod attributes are valid")
 | |
| }
 | |
| 
 | |
| func verifyPlugins(t *testing.T, jenkinsClient *gojenkins.Jenkins, jenkins *v1alpha1.Jenkins) {
 | |
| 	installedPlugins, err := jenkinsClient.GetPlugins(1)
 | |
| 	if err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 
 | |
| 	requiredPlugins := []map[string][]string{plugins.BasePlugins(), jenkins.Spec.Master.Plugins}
 | |
| 	for _, p := range requiredPlugins {
 | |
| 		for rootPluginName, dependentPlugins := range p {
 | |
| 			rootPlugin, err := plugins.New(rootPluginName)
 | |
| 			if err != nil {
 | |
| 				t.Fatal(err)
 | |
| 			}
 | |
| 			if found, ok := isPluginValid(installedPlugins, *rootPlugin); !ok {
 | |
| 				t.Fatalf("Invalid plugin '%s', actual '%+v'", rootPlugin, found)
 | |
| 			}
 | |
| 			for _, pluginName := range dependentPlugins {
 | |
| 				plugin, err := plugins.New(pluginName)
 | |
| 				if err != nil {
 | |
| 					t.Fatal(err)
 | |
| 				}
 | |
| 				if found, ok := isPluginValid(installedPlugins, *plugin); !ok {
 | |
| 					t.Fatalf("Invalid plugin '%s', actual '%+v'", rootPlugin, found)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	t.Log("All plugins have been installed")
 | |
| }
 | |
| 
 | |
| func isPluginValid(plugins *gojenkins.Plugins, requiredPlugin plugins.Plugin) (*gojenkins.Plugin, bool) {
 | |
| 	p := plugins.Contains(requiredPlugin.Name)
 | |
| 	if p == nil {
 | |
| 		return p, false
 | |
| 	}
 | |
| 
 | |
| 	if !p.Active || !p.Enabled || p.Deleted {
 | |
| 		return p, false
 | |
| 	}
 | |
| 
 | |
| 	return p, requiredPlugin.Version == p.Version
 | |
| }
 | |
| 
 | |
| func verifyJenkinsSeedJobs(t *testing.T, client *gojenkins.Jenkins, jenkins *v1alpha1.Jenkins) {
 | |
| 	t.Logf("Attempting to get configure seed job status '%v'", seedjobs.ConfigureSeedJobsName)
 | |
| 
 | |
| 	configureSeedJobs, err := client.GetJob(seedjobs.ConfigureSeedJobsName)
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.NotNil(t, configureSeedJobs)
 | |
| 	build, err := configureSeedJobs.GetLastSuccessfulBuild()
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.NotNil(t, build)
 | |
| 
 | |
| 	seedJobName := "jenkins-operator-configure-seed-job"
 | |
| 	t.Logf("Attempting to verify if seed job has been created '%v'", seedJobName)
 | |
| 	seedJob, err := client.GetJob(seedJobName)
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.NotNil(t, seedJob)
 | |
| 
 | |
| 	build, err = seedJob.GetLastSuccessfulBuild()
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.NotNil(t, build)
 | |
| 
 | |
| 	err = framework.Global.Client.Get(context.TODO(), types.NamespacedName{Namespace: jenkins.Namespace, Name: jenkins.Name}, jenkins)
 | |
| 	assert.NoError(t, err, "couldn't get jenkins custom resource")
 | |
| 	assert.NotNil(t, jenkins.Status.Builds)
 | |
| 	assert.NotEmpty(t, jenkins.Status.Builds)
 | |
| 
 | |
| 	jobCreatedByDSLPluginName := "build-jenkins-operator"
 | |
| 	err = wait.Poll(time.Second*10, time.Minute*2, func() (bool, error) {
 | |
| 		t.Logf("Attempting to verify if job '%s' has been created ", jobCreatedByDSLPluginName)
 | |
| 		seedJob, err := client.GetJob(jobCreatedByDSLPluginName)
 | |
| 		if err != nil || seedJob == nil {
 | |
| 			return false, nil
 | |
| 		}
 | |
| 		return true, nil
 | |
| 	})
 | |
| 	assert.NoError(t, err)
 | |
| }
 |