mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-31 13:50:01 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			625 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			625 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
| Copyright 2016 The Kubernetes Authors.
 | |
| 
 | |
| Licensed under the Apache License, Version 2.0 (the "License");
 | |
| you may not use this file except in compliance with the License.
 | |
| You may obtain a copy of the License at
 | |
| 
 | |
|     http://www.apache.org/licenses/LICENSE-2.0
 | |
| 
 | |
| Unless required by applicable law or agreed to in writing, software
 | |
| distributed under the License is distributed on an "AS IS" BASIS,
 | |
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| See the License for the specific language governing permissions and
 | |
| limitations under the License.
 | |
| */
 | |
| 
 | |
| package validation
 | |
| 
 | |
| import (
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | |
| 	"k8s.io/apimachinery/pkg/types"
 | |
| 	"k8s.io/kubernetes/pkg/apis/batch"
 | |
| 	api "k8s.io/kubernetes/pkg/apis/core"
 | |
| )
 | |
| 
 | |
| func getValidManualSelector() *metav1.LabelSelector {
 | |
| 	return &metav1.LabelSelector{
 | |
| 		MatchLabels: map[string]string{"a": "b"},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func getValidPodTemplateSpecForManual(selector *metav1.LabelSelector) api.PodTemplateSpec {
 | |
| 	return api.PodTemplateSpec{
 | |
| 		ObjectMeta: metav1.ObjectMeta{
 | |
| 			Labels: selector.MatchLabels,
 | |
| 		},
 | |
| 		Spec: api.PodSpec{
 | |
| 			RestartPolicy: api.RestartPolicyOnFailure,
 | |
| 			DNSPolicy:     api.DNSClusterFirst,
 | |
| 			Containers:    []api.Container{{Name: "abc", Image: "image", ImagePullPolicy: "IfNotPresent", TerminationMessagePolicy: api.TerminationMessageReadFile}},
 | |
| 		},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func getValidGeneratedSelector() *metav1.LabelSelector {
 | |
| 	return &metav1.LabelSelector{
 | |
| 		MatchLabels: map[string]string{"controller-uid": "1a2b3c", "job-name": "myjob"},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func getValidPodTemplateSpecForGenerated(selector *metav1.LabelSelector) api.PodTemplateSpec {
 | |
| 	return api.PodTemplateSpec{
 | |
| 		ObjectMeta: metav1.ObjectMeta{
 | |
| 			Labels: selector.MatchLabels,
 | |
| 		},
 | |
| 		Spec: api.PodSpec{
 | |
| 			RestartPolicy: api.RestartPolicyOnFailure,
 | |
| 			DNSPolicy:     api.DNSClusterFirst,
 | |
| 			Containers:    []api.Container{{Name: "abc", Image: "image", ImagePullPolicy: "IfNotPresent", TerminationMessagePolicy: api.TerminationMessageReadFile}},
 | |
| 		},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestValidateJob(t *testing.T) {
 | |
| 	validManualSelector := getValidManualSelector()
 | |
| 	validPodTemplateSpecForManual := getValidPodTemplateSpecForManual(validManualSelector)
 | |
| 	validGeneratedSelector := getValidGeneratedSelector()
 | |
| 	validPodTemplateSpecForGenerated := getValidPodTemplateSpecForGenerated(validGeneratedSelector)
 | |
| 
 | |
| 	successCases := map[string]batch.Job{
 | |
| 		"manual selector": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "myjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.JobSpec{
 | |
| 				Selector:       validManualSelector,
 | |
| 				ManualSelector: newBool(true),
 | |
| 				Template:       validPodTemplateSpecForManual,
 | |
| 			},
 | |
| 		},
 | |
| 		"generated selector": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "myjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.JobSpec{
 | |
| 				Selector: validGeneratedSelector,
 | |
| 				Template: validPodTemplateSpecForGenerated,
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 	for k, v := range successCases {
 | |
| 		if errs := ValidateJob(&v); len(errs) != 0 {
 | |
| 			t.Errorf("expected success for %s: %v", k, errs)
 | |
| 		}
 | |
| 	}
 | |
| 	negative := int32(-1)
 | |
| 	negative64 := int64(-1)
 | |
| 	errorCases := map[string]batch.Job{
 | |
| 		"spec.parallelism:must be greater than or equal to 0": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "myjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.JobSpec{
 | |
| 				Parallelism: &negative,
 | |
| 				Selector:    validGeneratedSelector,
 | |
| 				Template:    validPodTemplateSpecForGenerated,
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.completions:must be greater than or equal to 0": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "myjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.JobSpec{
 | |
| 				Completions: &negative,
 | |
| 				Selector:    validGeneratedSelector,
 | |
| 				Template:    validPodTemplateSpecForGenerated,
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.activeDeadlineSeconds:must be greater than or equal to 0": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "myjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.JobSpec{
 | |
| 				ActiveDeadlineSeconds: &negative64,
 | |
| 				Selector:              validGeneratedSelector,
 | |
| 				Template:              validPodTemplateSpecForGenerated,
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.selector:Required value": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "myjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.JobSpec{
 | |
| 				Template: validPodTemplateSpecForGenerated,
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.template.metadata.labels: Invalid value: {\"y\":\"z\"}: `selector` does not match template `labels`": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "myjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.JobSpec{
 | |
| 				Selector:       validManualSelector,
 | |
| 				ManualSelector: newBool(true),
 | |
| 				Template: api.PodTemplateSpec{
 | |
| 					ObjectMeta: metav1.ObjectMeta{
 | |
| 						Labels: map[string]string{"y": "z"},
 | |
| 					},
 | |
| 					Spec: api.PodSpec{
 | |
| 						RestartPolicy: api.RestartPolicyOnFailure,
 | |
| 						DNSPolicy:     api.DNSClusterFirst,
 | |
| 						Containers:    []api.Container{{Name: "abc", Image: "image", ImagePullPolicy: "IfNotPresent", TerminationMessagePolicy: api.TerminationMessageReadFile}},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.template.metadata.labels: Invalid value: {\"controller-uid\":\"4d5e6f\"}: `selector` does not match template `labels`": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "myjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.JobSpec{
 | |
| 				Selector:       validManualSelector,
 | |
| 				ManualSelector: newBool(true),
 | |
| 				Template: api.PodTemplateSpec{
 | |
| 					ObjectMeta: metav1.ObjectMeta{
 | |
| 						Labels: map[string]string{"controller-uid": "4d5e6f"},
 | |
| 					},
 | |
| 					Spec: api.PodSpec{
 | |
| 						RestartPolicy: api.RestartPolicyOnFailure,
 | |
| 						DNSPolicy:     api.DNSClusterFirst,
 | |
| 						Containers:    []api.Container{{Name: "abc", Image: "image", ImagePullPolicy: "IfNotPresent", TerminationMessagePolicy: api.TerminationMessageReadFile}},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.template.spec.restartPolicy: Unsupported value": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "myjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.JobSpec{
 | |
| 				Selector:       validManualSelector,
 | |
| 				ManualSelector: newBool(true),
 | |
| 				Template: api.PodTemplateSpec{
 | |
| 					ObjectMeta: metav1.ObjectMeta{
 | |
| 						Labels: validManualSelector.MatchLabels,
 | |
| 					},
 | |
| 					Spec: api.PodSpec{
 | |
| 						RestartPolicy: api.RestartPolicyAlways,
 | |
| 						DNSPolicy:     api.DNSClusterFirst,
 | |
| 						Containers:    []api.Container{{Name: "abc", Image: "image", ImagePullPolicy: "IfNotPresent", TerminationMessagePolicy: api.TerminationMessageReadFile}},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for k, v := range errorCases {
 | |
| 		errs := ValidateJob(&v)
 | |
| 		if len(errs) == 0 {
 | |
| 			t.Errorf("expected failure for %s", k)
 | |
| 		} else {
 | |
| 			s := strings.Split(k, ":")
 | |
| 			err := errs[0]
 | |
| 			if err.Field != s[0] || !strings.Contains(err.Error(), s[1]) {
 | |
| 				t.Errorf("unexpected error: %v, expected: %s", err, k)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestValidateJobUpdateStatus(t *testing.T) {
 | |
| 	type testcase struct {
 | |
| 		old    batch.Job
 | |
| 		update batch.Job
 | |
| 	}
 | |
| 
 | |
| 	successCases := []testcase{
 | |
| 		{
 | |
| 			old: batch.Job{
 | |
| 				ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
 | |
| 				Status: batch.JobStatus{
 | |
| 					Active:    1,
 | |
| 					Succeeded: 2,
 | |
| 					Failed:    3,
 | |
| 				},
 | |
| 			},
 | |
| 			update: batch.Job{
 | |
| 				ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
 | |
| 				Status: batch.JobStatus{
 | |
| 					Active:    1,
 | |
| 					Succeeded: 1,
 | |
| 					Failed:    3,
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, successCase := range successCases {
 | |
| 		successCase.old.ObjectMeta.ResourceVersion = "1"
 | |
| 		successCase.update.ObjectMeta.ResourceVersion = "1"
 | |
| 		if errs := ValidateJobUpdateStatus(&successCase.update, &successCase.old); len(errs) != 0 {
 | |
| 			t.Errorf("expected success: %v", errs)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	errorCases := map[string]testcase{
 | |
| 		"[status.active: Invalid value: -1: must be greater than or equal to 0, status.succeeded: Invalid value: -2: must be greater than or equal to 0]": {
 | |
| 			old: batch.Job{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Name:            "abc",
 | |
| 					Namespace:       metav1.NamespaceDefault,
 | |
| 					ResourceVersion: "10",
 | |
| 				},
 | |
| 				Status: batch.JobStatus{
 | |
| 					Active:    1,
 | |
| 					Succeeded: 2,
 | |
| 					Failed:    3,
 | |
| 				},
 | |
| 			},
 | |
| 			update: batch.Job{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Name:            "abc",
 | |
| 					Namespace:       metav1.NamespaceDefault,
 | |
| 					ResourceVersion: "10",
 | |
| 				},
 | |
| 				Status: batch.JobStatus{
 | |
| 					Active:    -1,
 | |
| 					Succeeded: -2,
 | |
| 					Failed:    3,
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for testName, errorCase := range errorCases {
 | |
| 		errs := ValidateJobUpdateStatus(&errorCase.update, &errorCase.old)
 | |
| 		if len(errs) == 0 {
 | |
| 			t.Errorf("expected failure: %s", testName)
 | |
| 			continue
 | |
| 		}
 | |
| 		if errs.ToAggregate().Error() != testName {
 | |
| 			t.Errorf("expected '%s' got '%s'", errs.ToAggregate().Error(), testName)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestValidateCronJob(t *testing.T) {
 | |
| 	validManualSelector := getValidManualSelector()
 | |
| 	validPodTemplateSpec := getValidPodTemplateSpecForGenerated(getValidGeneratedSelector())
 | |
| 	validPodTemplateSpec.Labels = map[string]string{}
 | |
| 
 | |
| 	successCases := map[string]batch.CronJob{
 | |
| 		"basic scheduled job": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:          "* * * * ?",
 | |
| 				ConcurrencyPolicy: batch.AllowConcurrent,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Template: validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"non-standard scheduled": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:          "@hourly",
 | |
| 				ConcurrencyPolicy: batch.AllowConcurrent,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Template: validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 	for k, v := range successCases {
 | |
| 		if errs := ValidateCronJob(&v); len(errs) != 0 {
 | |
| 			t.Errorf("expected success for %s: %v", k, errs)
 | |
| 		}
 | |
| 
 | |
| 		// Update validation should pass same success cases
 | |
| 		// copy to avoid polluting the testcase object, set a resourceVersion to allow validating update, and test a no-op update
 | |
| 		v = *v.DeepCopy()
 | |
| 		v.ResourceVersion = "1"
 | |
| 		if errs := ValidateCronJobUpdate(&v, &v); len(errs) != 0 {
 | |
| 			t.Errorf("expected success for %s: %v", k, errs)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	negative := int32(-1)
 | |
| 	negative64 := int64(-1)
 | |
| 
 | |
| 	errorCases := map[string]batch.CronJob{
 | |
| 		"spec.schedule: Invalid value": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:          "error",
 | |
| 				ConcurrencyPolicy: batch.AllowConcurrent,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Template: validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.schedule: Required value": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:          "",
 | |
| 				ConcurrencyPolicy: batch.AllowConcurrent,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Template: validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.startingDeadlineSeconds:must be greater than or equal to 0": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:                "* * * * ?",
 | |
| 				ConcurrencyPolicy:       batch.AllowConcurrent,
 | |
| 				StartingDeadlineSeconds: &negative64,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Template: validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.successfulJobsHistoryLimit: must be greater than or equal to 0": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:                   "* * * * ?",
 | |
| 				ConcurrencyPolicy:          batch.AllowConcurrent,
 | |
| 				SuccessfulJobsHistoryLimit: &negative,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Template: validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.failedJobsHistoryLimit: must be greater than or equal to 0": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:               "* * * * ?",
 | |
| 				ConcurrencyPolicy:      batch.AllowConcurrent,
 | |
| 				FailedJobsHistoryLimit: &negative,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Template: validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.concurrencyPolicy: Required value": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule: "* * * * ?",
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Template: validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.jobTemplate.spec.parallelism:must be greater than or equal to 0": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:          "* * * * ?",
 | |
| 				ConcurrencyPolicy: batch.AllowConcurrent,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Parallelism: &negative,
 | |
| 						Template:    validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.jobTemplate.spec.completions:must be greater than or equal to 0": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:          "* * * * ?",
 | |
| 				ConcurrencyPolicy: batch.AllowConcurrent,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Completions: &negative,
 | |
| 						Template:    validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.jobTemplate.spec.activeDeadlineSeconds:must be greater than or equal to 0": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:          "* * * * ?",
 | |
| 				ConcurrencyPolicy: batch.AllowConcurrent,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						ActiveDeadlineSeconds: &negative64,
 | |
| 						Template:              validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.jobTemplate.spec.selector: Invalid value: {\"matchLabels\":{\"a\":\"b\"}}: `selector` will be auto-generated": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:          "* * * * ?",
 | |
| 				ConcurrencyPolicy: batch.AllowConcurrent,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Selector: validManualSelector,
 | |
| 						Template: validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"metadata.name: must be no more than 52 characters": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "10000000002000000000300000000040000000005000000000123",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:          "* * * * ?",
 | |
| 				ConcurrencyPolicy: batch.AllowConcurrent,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Template: validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.jobTemplate.spec.manualSelector: Unsupported value": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:          "* * * * ?",
 | |
| 				ConcurrencyPolicy: batch.AllowConcurrent,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						ManualSelector: newBool(true),
 | |
| 						Template:       validPodTemplateSpec,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		"spec.jobTemplate.spec.template.spec.restartPolicy: Unsupported value": {
 | |
| 			ObjectMeta: metav1.ObjectMeta{
 | |
| 				Name:      "mycronjob",
 | |
| 				Namespace: metav1.NamespaceDefault,
 | |
| 				UID:       types.UID("1a2b3c"),
 | |
| 			},
 | |
| 			Spec: batch.CronJobSpec{
 | |
| 				Schedule:          "* * * * ?",
 | |
| 				ConcurrencyPolicy: batch.AllowConcurrent,
 | |
| 				JobTemplate: batch.JobTemplateSpec{
 | |
| 					Spec: batch.JobSpec{
 | |
| 						Template: api.PodTemplateSpec{
 | |
| 							Spec: api.PodSpec{
 | |
| 								RestartPolicy: api.RestartPolicyAlways,
 | |
| 								DNSPolicy:     api.DNSClusterFirst,
 | |
| 								Containers:    []api.Container{{Name: "abc", Image: "image", ImagePullPolicy: "IfNotPresent", TerminationMessagePolicy: api.TerminationMessageReadFile}},
 | |
| 							},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for k, v := range errorCases {
 | |
| 		errs := ValidateCronJob(&v)
 | |
| 		if len(errs) == 0 {
 | |
| 			t.Errorf("expected failure for %s", k)
 | |
| 		} else {
 | |
| 			s := strings.Split(k, ":")
 | |
| 			err := errs[0]
 | |
| 			if err.Field != s[0] || !strings.Contains(err.Error(), s[1]) {
 | |
| 				t.Errorf("unexpected error: %v, expected: %s", err, k)
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Update validation should fail all failure cases other than the 52 character name limit
 | |
| 		// copy to avoid polluting the testcase object, set a resourceVersion to allow validating update, and test a no-op update
 | |
| 		v = *v.DeepCopy()
 | |
| 		v.ResourceVersion = "1"
 | |
| 		errs = ValidateCronJobUpdate(&v, &v)
 | |
| 		if len(errs) == 0 {
 | |
| 			if k == "metadata.name: must be no more than 52 characters" {
 | |
| 				continue
 | |
| 			}
 | |
| 			t.Errorf("expected failure for %s", k)
 | |
| 		} else {
 | |
| 			s := strings.Split(k, ":")
 | |
| 			err := errs[0]
 | |
| 			if err.Field != s[0] || !strings.Contains(err.Error(), s[1]) {
 | |
| 				t.Errorf("unexpected error: %v, expected: %s", err, k)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func newBool(val bool) *bool {
 | |
| 	p := new(bool)
 | |
| 	*p = val
 | |
| 	return p
 | |
| }
 |