
This should only happen if the Jobs were created by an older version of the CronJob controller, since from now on we add ControllerRef upon creation. CronJob doesn't do actual adoption because it doesn't use label selectors to find its Jobs. However, we should apply ControllerRef for potential server-side cascading deletion, and to advise other controllers we own these objects.
397 lines
12 KiB
Go
397 lines
12 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 cronjob
|
|
|
|
import (
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/types"
|
|
"k8s.io/kubernetes/pkg/api/v1"
|
|
batchv1 "k8s.io/kubernetes/pkg/apis/batch/v1"
|
|
batchv2alpha1 "k8s.io/kubernetes/pkg/apis/batch/v2alpha1"
|
|
"k8s.io/kubernetes/pkg/controller"
|
|
)
|
|
|
|
func TestGetJobFromTemplate(t *testing.T) {
|
|
// getJobFromTemplate() needs to take the job template and copy the labels and annotations
|
|
// and other fields, and add a created-by reference.
|
|
|
|
var one int64 = 1
|
|
var no bool = false
|
|
|
|
sj := batchv2alpha1.CronJob{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "mycronjob",
|
|
Namespace: "snazzycats",
|
|
UID: types.UID("1a2b3c"),
|
|
SelfLink: "/apis/batch/v1/namespaces/snazzycats/jobs/mycronjob",
|
|
},
|
|
Spec: batchv2alpha1.CronJobSpec{
|
|
Schedule: "* * * * ?",
|
|
ConcurrencyPolicy: batchv2alpha1.AllowConcurrent,
|
|
JobTemplate: batchv2alpha1.JobTemplateSpec{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Labels: map[string]string{"a": "b"},
|
|
Annotations: map[string]string{"x": "y"},
|
|
},
|
|
Spec: batchv1.JobSpec{
|
|
ActiveDeadlineSeconds: &one,
|
|
ManualSelector: &no,
|
|
Template: v1.PodTemplateSpec{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Labels: map[string]string{
|
|
"foo": "bar",
|
|
},
|
|
},
|
|
Spec: v1.PodSpec{
|
|
Containers: []v1.Container{
|
|
{Image: "foo/bar"},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
var job *batchv1.Job
|
|
job, err := getJobFromTemplate(&sj, time.Time{})
|
|
if err != nil {
|
|
t.Errorf("Did not expect error: %s", err)
|
|
}
|
|
if !strings.HasPrefix(job.ObjectMeta.Name, "mycronjob-") {
|
|
t.Errorf("Wrong Name")
|
|
}
|
|
if len(job.ObjectMeta.Labels) != 1 {
|
|
t.Errorf("Wrong number of labels")
|
|
}
|
|
if len(job.ObjectMeta.Annotations) != 2 {
|
|
t.Errorf("Wrong number of annotations")
|
|
}
|
|
v, ok := job.ObjectMeta.Annotations[v1.CreatedByAnnotation]
|
|
if !ok {
|
|
t.Errorf("Missing created-by annotation")
|
|
}
|
|
expectedCreatedBy := `{"kind":"SerializedReference","apiVersion":"v1","reference":{"kind":"CronJob","namespace":"snazzycats","name":"mycronjob","uid":"1a2b3c","apiVersion":"batch"}}
|
|
`
|
|
if len(v) != len(expectedCreatedBy) {
|
|
t.Errorf("Wrong length for created-by annotation, expected %v got %v", len(expectedCreatedBy), len(v))
|
|
}
|
|
if v != expectedCreatedBy {
|
|
t.Errorf("Wrong value for created-by annotation, expected %v got %v", expectedCreatedBy, v)
|
|
}
|
|
}
|
|
|
|
func TestGetParentUIDFromJob(t *testing.T) {
|
|
j := &batchv1.Job{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "foobar",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: batchv1.JobSpec{
|
|
Selector: &metav1.LabelSelector{
|
|
MatchLabels: map[string]string{"foo": "bar"},
|
|
},
|
|
Template: v1.PodTemplateSpec{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Labels: map[string]string{
|
|
"foo": "bar",
|
|
},
|
|
},
|
|
Spec: v1.PodSpec{
|
|
Containers: []v1.Container{
|
|
{Image: "foo/bar"},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
Status: batchv1.JobStatus{
|
|
Conditions: []batchv1.JobCondition{{
|
|
Type: batchv1.JobComplete,
|
|
Status: v1.ConditionTrue,
|
|
}},
|
|
},
|
|
}
|
|
{
|
|
// Case 1: No UID annotation
|
|
_, found := getParentUIDFromJob(*j)
|
|
|
|
if found {
|
|
t.Errorf("Unexpectedly found uid")
|
|
}
|
|
}
|
|
{
|
|
// Case 2: Has UID annotation
|
|
j.ObjectMeta.Annotations = map[string]string{v1.CreatedByAnnotation: `{"kind":"SerializedReference","apiVersion":"v1","reference":{"kind":"CronJob","namespace":"default","name":"pi","uid":"5ef034e0-1890-11e6-8935-42010af0003e","apiVersion":"extensions","resourceVersion":"427339"}}`}
|
|
|
|
expectedUID := types.UID("5ef034e0-1890-11e6-8935-42010af0003e")
|
|
|
|
uid, found := getParentUIDFromJob(*j)
|
|
if !found {
|
|
t.Errorf("Unexpectedly did not find uid")
|
|
} else if uid != expectedUID {
|
|
t.Errorf("Wrong UID: %v", uid)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
func TestGroupJobsByParent(t *testing.T) {
|
|
uid1 := types.UID("11111111-1111-1111-1111-111111111111")
|
|
uid2 := types.UID("22222222-2222-2222-2222-222222222222")
|
|
uid3 := types.UID("33333333-3333-3333-3333-333333333333")
|
|
createdBy1 := map[string]string{v1.CreatedByAnnotation: `{"kind":"SerializedReference","apiVersion":"v1","reference":{"kind":"CronJob","namespace":"x","name":"pi","uid":"11111111-1111-1111-1111-111111111111","apiVersion":"extensions","resourceVersion":"111111"}}`}
|
|
createdBy2 := map[string]string{v1.CreatedByAnnotation: `{"kind":"SerializedReference","apiVersion":"v1","reference":{"kind":"CronJob","namespace":"x","name":"pi","uid":"22222222-2222-2222-2222-222222222222","apiVersion":"extensions","resourceVersion":"222222"}}`}
|
|
createdBy3 := map[string]string{v1.CreatedByAnnotation: `{"kind":"SerializedReference","apiVersion":"v1","reference":{"kind":"CronJob","namespace":"y","name":"pi","uid":"33333333-3333-3333-3333-333333333333","apiVersion":"extensions","resourceVersion":"333333"}}`}
|
|
noCreatedBy := map[string]string{}
|
|
|
|
{
|
|
// Case 1: There are no jobs and scheduledJobs
|
|
js := []batchv1.Job{}
|
|
jobsBySj := groupJobsByParent(js)
|
|
if len(jobsBySj) != 0 {
|
|
t.Errorf("Wrong number of items in map")
|
|
}
|
|
}
|
|
|
|
{
|
|
// Case 2: there is one controller with one job it created.
|
|
js := []batchv1.Job{
|
|
{ObjectMeta: metav1.ObjectMeta{Name: "a", Namespace: "x", Annotations: createdBy1}},
|
|
}
|
|
jobsBySj := groupJobsByParent(js)
|
|
|
|
if len(jobsBySj) != 1 {
|
|
t.Errorf("Wrong number of items in map")
|
|
}
|
|
jobList1, found := jobsBySj[uid1]
|
|
if !found {
|
|
t.Errorf("Key not found")
|
|
}
|
|
if len(jobList1) != 1 {
|
|
t.Errorf("Wrong number of items in map")
|
|
}
|
|
}
|
|
|
|
{
|
|
// Case 3: Two namespaces, one has two jobs from one controller, other has 3 jobs from two controllers.
|
|
// There are also two jobs with no created-by annotation.
|
|
js := []batchv1.Job{
|
|
{ObjectMeta: metav1.ObjectMeta{Name: "a", Namespace: "x", Annotations: createdBy1}},
|
|
{ObjectMeta: metav1.ObjectMeta{Name: "b", Namespace: "x", Annotations: createdBy2}},
|
|
{ObjectMeta: metav1.ObjectMeta{Name: "c", Namespace: "x", Annotations: createdBy1}},
|
|
{ObjectMeta: metav1.ObjectMeta{Name: "d", Namespace: "x", Annotations: noCreatedBy}},
|
|
{ObjectMeta: metav1.ObjectMeta{Name: "a", Namespace: "y", Annotations: createdBy3}},
|
|
{ObjectMeta: metav1.ObjectMeta{Name: "b", Namespace: "y", Annotations: createdBy3}},
|
|
{ObjectMeta: metav1.ObjectMeta{Name: "d", Namespace: "y", Annotations: noCreatedBy}},
|
|
}
|
|
|
|
jobsBySj := groupJobsByParent(js)
|
|
|
|
if len(jobsBySj) != 3 {
|
|
t.Errorf("Wrong number of items in map")
|
|
}
|
|
jobList1, found := jobsBySj[uid1]
|
|
if !found {
|
|
t.Errorf("Key not found")
|
|
}
|
|
if len(jobList1) != 2 {
|
|
t.Errorf("Wrong number of items in map")
|
|
}
|
|
jobList2, found := jobsBySj[uid2]
|
|
if !found {
|
|
t.Errorf("Key not found")
|
|
}
|
|
if len(jobList2) != 1 {
|
|
t.Errorf("Wrong number of items in map")
|
|
}
|
|
jobList3, found := jobsBySj[uid3]
|
|
if !found {
|
|
t.Errorf("Key not found")
|
|
}
|
|
if len(jobList3) != 2 {
|
|
t.Errorf("Wrong number of items in map")
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
func TestGetRecentUnmetScheduleTimes(t *testing.T) {
|
|
// schedule is hourly on the hour
|
|
schedule := "0 * * * ?"
|
|
// T1 is a scheduled start time of that schedule
|
|
T1, err := time.Parse(time.RFC3339, "2016-05-19T10:00:00Z")
|
|
if err != nil {
|
|
t.Errorf("test setup error: %v", err)
|
|
}
|
|
// T2 is a scheduled start time of that schedule after T1
|
|
T2, err := time.Parse(time.RFC3339, "2016-05-19T11:00:00Z")
|
|
if err != nil {
|
|
t.Errorf("test setup error: %v", err)
|
|
}
|
|
|
|
sj := batchv2alpha1.CronJob{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "mycronjob",
|
|
Namespace: metav1.NamespaceDefault,
|
|
UID: types.UID("1a2b3c"),
|
|
},
|
|
Spec: batchv2alpha1.CronJobSpec{
|
|
Schedule: schedule,
|
|
ConcurrencyPolicy: batchv2alpha1.AllowConcurrent,
|
|
JobTemplate: batchv2alpha1.JobTemplateSpec{},
|
|
},
|
|
}
|
|
{
|
|
// Case 1: no known start times, and none needed yet.
|
|
// Creation time is before T1.
|
|
sj.ObjectMeta.CreationTimestamp = metav1.Time{Time: T1.Add(-10 * time.Minute)}
|
|
// Current time is more than creation time, but less than T1.
|
|
now := T1.Add(-7 * time.Minute)
|
|
times, err := getRecentUnmetScheduleTimes(sj, now)
|
|
if err != nil {
|
|
t.Errorf("unexpected error: %v", err)
|
|
}
|
|
if len(times) != 0 {
|
|
t.Errorf("expected no start times, got: %v", times)
|
|
}
|
|
}
|
|
{
|
|
// Case 2: no known start times, and one needed.
|
|
// Creation time is before T1.
|
|
sj.ObjectMeta.CreationTimestamp = metav1.Time{Time: T1.Add(-10 * time.Minute)}
|
|
// Current time is after T1
|
|
now := T1.Add(2 * time.Second)
|
|
times, err := getRecentUnmetScheduleTimes(sj, now)
|
|
if err != nil {
|
|
t.Errorf("unexpected error: %v", err)
|
|
}
|
|
if len(times) != 1 {
|
|
t.Errorf("expected 1 start time, got: %v", times)
|
|
} else if !times[0].Equal(T1) {
|
|
t.Errorf("expected: %v, got: %v", T1, times[0])
|
|
}
|
|
}
|
|
{
|
|
// Case 3: known LastScheduleTime, no start needed.
|
|
// Creation time is before T1.
|
|
sj.ObjectMeta.CreationTimestamp = metav1.Time{Time: T1.Add(-10 * time.Minute)}
|
|
// Status shows a start at the expected time.
|
|
sj.Status.LastScheduleTime = &metav1.Time{Time: T1}
|
|
// Current time is after T1
|
|
now := T1.Add(2 * time.Minute)
|
|
times, err := getRecentUnmetScheduleTimes(sj, now)
|
|
if err != nil {
|
|
t.Errorf("unexpected error: %v", err)
|
|
}
|
|
if len(times) != 0 {
|
|
t.Errorf("expected 0 start times, got: , got: %v", times)
|
|
}
|
|
}
|
|
{
|
|
// Case 4: known LastScheduleTime, a start needed
|
|
// Creation time is before T1.
|
|
sj.ObjectMeta.CreationTimestamp = metav1.Time{Time: T1.Add(-10 * time.Minute)}
|
|
// Status shows a start at the expected time.
|
|
sj.Status.LastScheduleTime = &metav1.Time{Time: T1}
|
|
// Current time is after T1 and after T2
|
|
now := T2.Add(5 * time.Minute)
|
|
times, err := getRecentUnmetScheduleTimes(sj, now)
|
|
if err != nil {
|
|
t.Errorf("unexpected error: %v", err)
|
|
}
|
|
if len(times) != 1 {
|
|
t.Errorf("expected 2 start times, got: , got: %v", times)
|
|
} else if !times[0].Equal(T2) {
|
|
t.Errorf("expected: %v, got: %v", T1, times[0])
|
|
}
|
|
}
|
|
{
|
|
// Case 5: known LastScheduleTime, two starts needed
|
|
sj.ObjectMeta.CreationTimestamp = metav1.Time{Time: T1.Add(-2 * time.Hour)}
|
|
sj.Status.LastScheduleTime = &metav1.Time{Time: T1.Add(-1 * time.Hour)}
|
|
// Current time is after T1 and after T2
|
|
now := T2.Add(5 * time.Minute)
|
|
times, err := getRecentUnmetScheduleTimes(sj, now)
|
|
if err != nil {
|
|
t.Errorf("unexpected error: %v", err)
|
|
}
|
|
if len(times) != 2 {
|
|
t.Errorf("expected 2 start times, got: , got: %v", times)
|
|
} else {
|
|
if !times[0].Equal(T1) {
|
|
t.Errorf("expected: %v, got: %v", T1, times[0])
|
|
}
|
|
if !times[1].Equal(T2) {
|
|
t.Errorf("expected: %v, got: %v", T2, times[1])
|
|
}
|
|
}
|
|
}
|
|
{
|
|
// Case 6: now is way way ahead of last start time, and there is no deadline.
|
|
sj.ObjectMeta.CreationTimestamp = metav1.Time{Time: T1.Add(-2 * time.Hour)}
|
|
sj.Status.LastScheduleTime = &metav1.Time{Time: T1.Add(-1 * time.Hour)}
|
|
now := T2.Add(10 * 24 * time.Hour)
|
|
_, err := getRecentUnmetScheduleTimes(sj, now)
|
|
if err == nil {
|
|
t.Errorf("unexpected lack of error")
|
|
}
|
|
}
|
|
{
|
|
// Case 7: now is way way ahead of last start time, but there is a short deadline.
|
|
sj.ObjectMeta.CreationTimestamp = metav1.Time{Time: T1.Add(-2 * time.Hour)}
|
|
sj.Status.LastScheduleTime = &metav1.Time{Time: T1.Add(-1 * time.Hour)}
|
|
now := T2.Add(10 * 24 * time.Hour)
|
|
// Deadline is short
|
|
deadline := int64(2 * 60 * 60)
|
|
sj.Spec.StartingDeadlineSeconds = &deadline
|
|
_, err := getRecentUnmetScheduleTimes(sj, now)
|
|
if err != nil {
|
|
t.Errorf("unexpected error")
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestAdoptJobs(t *testing.T) {
|
|
sj := cronJob()
|
|
jc := &fakeJobControl{}
|
|
jobs := []batchv1.Job{newJob("uid0"), newJob("uid1")}
|
|
jobs[0].OwnerReferences = nil
|
|
jobs[0].Name = "job0"
|
|
jobs[1].OwnerReferences = []metav1.OwnerReference{*newControllerRef(&sj)}
|
|
jobs[1].Name = "job1"
|
|
|
|
if err := adoptJobs(&sj, jobs, jc); err != nil {
|
|
t.Errorf("adoptJobs() error: %v", err)
|
|
}
|
|
for i := range jobs {
|
|
controllerRef := controller.GetControllerOf(&jobs[i])
|
|
if controllerRef == nil {
|
|
t.Errorf("Job should have ControllerRef: %#v", jobs[i])
|
|
}
|
|
}
|
|
if got, want := len(jc.UpdateJobName), 1; got != want {
|
|
t.Errorf("len(UpdateJobName) = %v, want %v", got, want)
|
|
}
|
|
if got, want := jc.UpdateJobName[0], "job0"; got != want {
|
|
t.Errorf("UpdateJobName = %v, want %v", got, want)
|
|
}
|
|
}
|