2015-03-20 16:37:08 +00:00
|
|
|
/*
|
2015-05-01 16:19:44 +00:00
|
|
|
Copyright 2014 The Kubernetes Authors All rights reserved.
|
2015-03-20 16:37:08 +00:00
|
|
|
|
|
|
|
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 kubelet
|
|
|
|
|
|
|
|
import (
|
2015-06-22 19:31:46 +00:00
|
|
|
"fmt"
|
2015-03-20 16:37:08 +00:00
|
|
|
"math/rand"
|
|
|
|
"strconv"
|
|
|
|
"testing"
|
2015-05-09 05:01:43 +00:00
|
|
|
"time"
|
2015-03-20 16:37:08 +00:00
|
|
|
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api"
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/client"
|
2015-04-06 23:27:53 +00:00
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/client/testclient"
|
2015-05-09 05:01:43 +00:00
|
|
|
kubecontainer "github.com/GoogleCloudPlatform/kubernetes/pkg/kubelet/container"
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/util"
|
2015-03-20 16:37:08 +00:00
|
|
|
)
|
|
|
|
|
2015-03-24 23:52:38 +00:00
|
|
|
var testPod *api.Pod = &api.Pod{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
UID: "12345678",
|
|
|
|
Name: "foo",
|
|
|
|
Namespace: "new",
|
|
|
|
},
|
|
|
|
}
|
2015-03-20 16:37:08 +00:00
|
|
|
|
|
|
|
func newTestStatusManager() *statusManager {
|
2015-04-06 23:27:53 +00:00
|
|
|
return newStatusManager(&testclient.Fake{})
|
2015-03-20 16:37:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func generateRandomMessage() string {
|
|
|
|
return strconv.Itoa(rand.Int())
|
|
|
|
}
|
|
|
|
|
|
|
|
func getRandomPodStatus() api.PodStatus {
|
|
|
|
return api.PodStatus{
|
|
|
|
Message: generateRandomMessage(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func verifyActions(t *testing.T, kubeClient client.Interface, expectedActions []string) {
|
2015-04-06 23:27:53 +00:00
|
|
|
actions := kubeClient.(*testclient.Fake).Actions
|
2015-03-20 16:37:08 +00:00
|
|
|
if len(actions) != len(expectedActions) {
|
|
|
|
t.Errorf("unexpected actions, got: %s expected: %s", actions, expectedActions)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for i := 0; i < len(actions); i++ {
|
|
|
|
if actions[i].Action != expectedActions[i] {
|
|
|
|
t.Errorf("unexpected actions, got: %s expected: %s", actions, expectedActions)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-25 16:32:16 +00:00
|
|
|
func verifyUpdates(t *testing.T, manager *statusManager, expectedUpdates int) {
|
|
|
|
// Consume all updates in the channel.
|
|
|
|
numUpdates := 0
|
|
|
|
for {
|
|
|
|
hasUpdate := true
|
|
|
|
select {
|
|
|
|
case <-manager.podStatusChannel:
|
|
|
|
numUpdates++
|
|
|
|
default:
|
|
|
|
hasUpdate = false
|
|
|
|
}
|
|
|
|
|
|
|
|
if !hasUpdate {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if numUpdates != expectedUpdates {
|
2015-03-31 22:32:02 +00:00
|
|
|
t.Errorf("unexpected number of updates %d, expected %d", numUpdates, expectedUpdates)
|
2015-03-25 16:32:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-20 16:37:08 +00:00
|
|
|
func TestNewStatus(t *testing.T) {
|
|
|
|
syncer := newTestStatusManager()
|
2015-03-24 23:52:38 +00:00
|
|
|
syncer.SetPodStatus(testPod, getRandomPodStatus())
|
2015-03-25 16:32:16 +00:00
|
|
|
verifyUpdates(t, syncer, 1)
|
2015-05-09 05:01:43 +00:00
|
|
|
|
|
|
|
status, _ := syncer.GetPodStatus(kubecontainer.GetPodFullName(testPod))
|
|
|
|
if status.StartTime.IsZero() {
|
|
|
|
t.Errorf("SetPodStatus did not set a proper start time value")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNewStatusPreservesPodStartTime(t *testing.T) {
|
|
|
|
syncer := newTestStatusManager()
|
|
|
|
pod := &api.Pod{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
UID: "12345678",
|
|
|
|
Name: "foo",
|
|
|
|
Namespace: "new",
|
|
|
|
},
|
|
|
|
Status: api.PodStatus{},
|
|
|
|
}
|
|
|
|
now := util.Now()
|
|
|
|
startTime := util.NewTime(now.Time.Add(-1 * time.Minute))
|
|
|
|
pod.Status.StartTime = &startTime
|
|
|
|
syncer.SetPodStatus(pod, getRandomPodStatus())
|
|
|
|
|
|
|
|
status, _ := syncer.GetPodStatus(kubecontainer.GetPodFullName(pod))
|
|
|
|
if !status.StartTime.Time.Equal(startTime.Time) {
|
|
|
|
t.Errorf("Unexpected start time, expected %v, actual %v", startTime, status.StartTime)
|
|
|
|
}
|
2015-03-20 16:37:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestChangedStatus(t *testing.T) {
|
|
|
|
syncer := newTestStatusManager()
|
2015-03-24 23:52:38 +00:00
|
|
|
syncer.SetPodStatus(testPod, getRandomPodStatus())
|
|
|
|
syncer.SetPodStatus(testPod, getRandomPodStatus())
|
2015-03-25 16:32:16 +00:00
|
|
|
verifyUpdates(t, syncer, 2)
|
2015-03-20 16:37:08 +00:00
|
|
|
}
|
|
|
|
|
2015-05-09 05:01:43 +00:00
|
|
|
func TestChangedStatusKeepsStartTime(t *testing.T) {
|
|
|
|
syncer := newTestStatusManager()
|
|
|
|
now := util.Now()
|
|
|
|
firstStatus := getRandomPodStatus()
|
|
|
|
firstStatus.StartTime = &now
|
|
|
|
syncer.SetPodStatus(testPod, firstStatus)
|
|
|
|
syncer.SetPodStatus(testPod, getRandomPodStatus())
|
|
|
|
verifyUpdates(t, syncer, 2)
|
|
|
|
finalStatus, _ := syncer.GetPodStatus(kubecontainer.GetPodFullName(testPod))
|
|
|
|
if finalStatus.StartTime.IsZero() {
|
|
|
|
t.Errorf("StartTime should not be zero")
|
|
|
|
}
|
|
|
|
if !finalStatus.StartTime.Time.Equal(now.Time) {
|
|
|
|
t.Errorf("Expected %v, but got %v", now.Time, finalStatus.StartTime.Time)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-20 16:37:08 +00:00
|
|
|
func TestUnchangedStatus(t *testing.T) {
|
|
|
|
syncer := newTestStatusManager()
|
|
|
|
podStatus := getRandomPodStatus()
|
2015-03-24 23:52:38 +00:00
|
|
|
syncer.SetPodStatus(testPod, podStatus)
|
|
|
|
syncer.SetPodStatus(testPod, podStatus)
|
2015-03-25 16:32:16 +00:00
|
|
|
verifyUpdates(t, syncer, 1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSyncBatch(t *testing.T) {
|
|
|
|
syncer := newTestStatusManager()
|
|
|
|
syncer.SetPodStatus(testPod, getRandomPodStatus())
|
|
|
|
err := syncer.syncBatch()
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("unexpected syncing error: %v", err)
|
|
|
|
}
|
2015-03-20 22:22:51 +00:00
|
|
|
verifyActions(t, syncer.kubeClient, []string{"get-pod", "update-status-pod"})
|
2015-03-20 16:37:08 +00:00
|
|
|
}
|
2015-06-22 19:31:46 +00:00
|
|
|
|
|
|
|
// shuffle returns a new shuffled list of container statuses.
|
|
|
|
func shuffle(statuses []api.ContainerStatus) []api.ContainerStatus {
|
|
|
|
numStatuses := len(statuses)
|
|
|
|
randIndexes := rand.Perm(numStatuses)
|
|
|
|
shuffled := make([]api.ContainerStatus, numStatuses)
|
|
|
|
for i := 0; i < numStatuses; i++ {
|
|
|
|
shuffled[i] = statuses[randIndexes[i]]
|
|
|
|
}
|
|
|
|
return shuffled
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStatusEquality(t *testing.T) {
|
|
|
|
containerStatus := []api.ContainerStatus{}
|
|
|
|
for i := 0; i < 10; i++ {
|
|
|
|
s := api.ContainerStatus{
|
|
|
|
Name: fmt.Sprintf("container%d", i),
|
|
|
|
}
|
|
|
|
containerStatus = append(containerStatus, s)
|
|
|
|
}
|
|
|
|
podStatus := api.PodStatus{
|
|
|
|
ContainerStatuses: containerStatus,
|
|
|
|
}
|
|
|
|
for i := 0; i < 10; i++ {
|
|
|
|
oldPodStatus := api.PodStatus{
|
|
|
|
ContainerStatuses: shuffle(podStatus.ContainerStatuses),
|
|
|
|
}
|
|
|
|
if !isStatusEqual(&oldPodStatus, &podStatus) {
|
|
|
|
t.Fatalf("Order of container statuses should not affect equality.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|