k3s/pkg/kubelet/kubelet_test.go

1091 lines
26 KiB
Go
Raw Normal View History

2014-06-06 23:40:48 +00:00
/*
Copyright 2014 Google Inc. All rights reserved.
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.
*/
2014-06-23 18:32:11 +00:00
2014-06-06 23:40:48 +00:00
package kubelet
import (
"encoding/json"
"fmt"
"io/ioutil"
2014-06-06 23:40:48 +00:00
"net/http/httptest"
"reflect"
2014-06-06 23:40:48 +00:00
"sync"
"testing"
"github.com/GoogleCloudPlatform/kubernetes/pkg/api"
"github.com/GoogleCloudPlatform/kubernetes/pkg/util"
"github.com/coreos/go-etcd/etcd"
"github.com/fsouza/go-dockerclient"
2014-06-19 20:22:20 +00:00
"github.com/google/cadvisor/info"
"github.com/stretchr/testify/mock"
2014-06-06 23:40:48 +00:00
)
// TODO: This doesn't reduce typing enough to make it worth the less readable errors. Remove.
func expectNoError(t *testing.T, err error) {
if err != nil {
t.Errorf("Unexpected error: %#v", err)
}
}
// These are used for testing extract json (below)
type TestData struct {
Value string
Number int
}
type TestObject struct {
Name string
Data TestData
}
func verifyStringEquals(t *testing.T, actual, expected string) {
if actual != expected {
t.Errorf("Verification failed. Expected: %s, Found %s", expected, actual)
}
}
func verifyIntEquals(t *testing.T, actual, expected int) {
if actual != expected {
t.Errorf("Verification failed. Expected: %d, Found %d", expected, actual)
}
}
func verifyNoError(t *testing.T, e error) {
if e != nil {
t.Errorf("Expected no error, found %#v", e)
}
}
func verifyError(t *testing.T, e error) {
if e == nil {
t.Errorf("Expected error, found nil")
}
}
func makeTestKubelet() *Kubelet {
return &Kubelet{
DockerPuller: &FakeDockerPuller{},
}
}
2014-06-06 23:40:48 +00:00
func TestExtractJSON(t *testing.T) {
obj := TestObject{}
kubelet := makeTestKubelet()
2014-06-06 23:40:48 +00:00
data := `{ "name": "foo", "data": { "value": "bar", "number": 10 } }`
kubelet.ExtractYAMLData([]byte(data), &obj)
verifyStringEquals(t, obj.Name, "foo")
verifyStringEquals(t, obj.Data.Value, "bar")
verifyIntEquals(t, obj.Data.Number, 10)
}
func verifyCalls(t *testing.T, fakeDocker FakeDockerClient, calls []string) {
verifyStringArrayEquals(t, fakeDocker.called, calls)
}
func verifyStringArrayEquals(t *testing.T, actual, expected []string) {
invalid := len(actual) != len(expected)
for ix, value := range actual {
if expected[ix] != value {
invalid = true
}
}
if invalid {
t.Errorf("Expected: %#v, Actual: %#v", expected, actual)
}
}
func verifyPackUnpack(t *testing.T, manifestId, containerName string) {
name := manifestAndContainerToDockerName(
&api.ContainerManifest{Id: manifestId},
&api.Container{Name: containerName},
)
returnedManifestId, returnedContainerName := dockerNameToManifestAndContainer(name)
if manifestId != returnedManifestId || containerName != returnedContainerName {
t.Errorf("For (%s, %s), unpacked (%s, %s)", manifestId, containerName, returnedManifestId, returnedContainerName)
}
}
2014-06-09 23:50:44 +00:00
func verifyBoolean(t *testing.T, expected, value bool) {
if expected != value {
t.Errorf("Unexpected boolean. Expected %s. Found %s", expected, value)
}
}
2014-06-06 23:40:48 +00:00
func TestContainerManifestNaming(t *testing.T) {
verifyPackUnpack(t, "manifest1234", "container5678")
verifyPackUnpack(t, "manifest--", "container__")
verifyPackUnpack(t, "--manifest", "__container")
verifyPackUnpack(t, "m___anifest_", "container-_-")
verifyPackUnpack(t, "_m___anifest", "-_-container")
}
func TestContainerExists(t *testing.T) {
fakeDocker := FakeDockerClient{
err: nil,
}
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-06 23:40:48 +00:00
}
manifest := api.ContainerManifest{
Id: "qux",
}
container := api.Container{
Name: "foo",
}
fakeDocker.containerList = []docker.APIContainers{
2014-06-12 21:09:40 +00:00
{
Names: []string{"/k8s--foo--qux--1234"},
2014-06-06 23:40:48 +00:00
},
2014-06-12 21:09:40 +00:00
{
Names: []string{"/k8s--bar--qux--1234"},
2014-06-06 23:40:48 +00:00
},
}
fakeDocker.container = &docker.Container{
ID: "foobar",
}
exists, _, err := kubelet.ContainerExists(&manifest, &container)
verifyCalls(t, fakeDocker, []string{"list", "list", "inspect"})
if !exists {
t.Errorf("Failed to find container %#v", container)
}
if err != nil {
t.Errorf("Unexpected error: %#v", err)
}
fakeDocker.clearCalls()
missingManifest := api.ContainerManifest{Id: "foobar"}
exists, _, err = kubelet.ContainerExists(&missingManifest, &container)
verifyCalls(t, fakeDocker, []string{"list"})
if exists {
t.Errorf("Failed to not find container %#v, missingManifest")
}
2014-06-06 23:40:48 +00:00
}
func TestGetContainerID(t *testing.T) {
fakeDocker := FakeDockerClient{
err: nil,
}
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-06 23:40:48 +00:00
}
fakeDocker.containerList = []docker.APIContainers{
2014-06-12 21:09:40 +00:00
{
2014-06-06 23:40:48 +00:00
Names: []string{"foo"},
ID: "1234",
},
2014-06-12 21:09:40 +00:00
{
2014-06-06 23:40:48 +00:00
Names: []string{"bar"},
ID: "4567",
},
}
2014-06-09 23:50:44 +00:00
id, found, err := kubelet.GetContainerID("foo")
verifyBoolean(t, true, found)
2014-06-06 23:40:48 +00:00
verifyStringEquals(t, id, "1234")
verifyNoError(t, err)
verifyCalls(t, fakeDocker, []string{"list"})
fakeDocker.clearCalls()
2014-06-09 23:50:44 +00:00
id, found, err = kubelet.GetContainerID("bar")
verifyBoolean(t, true, found)
2014-06-06 23:40:48 +00:00
verifyStringEquals(t, id, "4567")
verifyNoError(t, err)
verifyCalls(t, fakeDocker, []string{"list"})
fakeDocker.clearCalls()
2014-06-09 23:50:44 +00:00
id, found, err = kubelet.GetContainerID("NotFound")
verifyBoolean(t, false, found)
verifyNoError(t, err)
2014-06-06 23:40:48 +00:00
verifyCalls(t, fakeDocker, []string{"list"})
}
func TestGetContainerByName(t *testing.T) {
fakeDocker := FakeDockerClient{
err: nil,
}
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-06 23:40:48 +00:00
}
fakeDocker.containerList = []docker.APIContainers{
2014-06-12 21:09:40 +00:00
{
2014-06-06 23:40:48 +00:00
Names: []string{"foo"},
},
2014-06-12 21:09:40 +00:00
{
2014-06-06 23:40:48 +00:00
Names: []string{"bar"},
},
}
fakeDocker.container = &docker.Container{
ID: "foobar",
}
container, err := kubelet.GetContainerByName("foo")
verifyCalls(t, fakeDocker, []string{"list", "inspect"})
if container == nil {
t.Errorf("Unexpected nil container")
}
verifyStringEquals(t, container.ID, "foobar")
verifyNoError(t, err)
}
func TestListContainers(t *testing.T) {
fakeDocker := FakeDockerClient{
err: nil,
}
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-06 23:40:48 +00:00
}
fakeDocker.containerList = []docker.APIContainers{
2014-06-12 21:09:40 +00:00
{
2014-06-06 23:40:48 +00:00
Names: []string{"foo"},
},
2014-06-12 21:09:40 +00:00
{
2014-06-06 23:40:48 +00:00
Names: []string{"bar"},
},
}
containers, err := kubelet.ListContainers()
verifyStringArrayEquals(t, containers, []string{"foo", "bar"})
verifyNoError(t, err)
verifyCalls(t, fakeDocker, []string{"list"})
}
func TestKillContainerWithError(t *testing.T) {
fakeDocker := FakeDockerClient{
err: fmt.Errorf("sample error"),
2014-06-06 23:40:48 +00:00
containerList: []docker.APIContainers{
2014-06-12 21:09:40 +00:00
{
2014-06-06 23:40:48 +00:00
Names: []string{"foo"},
},
2014-06-12 21:09:40 +00:00
{
2014-06-06 23:40:48 +00:00
Names: []string{"bar"},
},
},
}
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-06 23:40:48 +00:00
}
err := kubelet.KillContainer("foo")
verifyError(t, err)
verifyCalls(t, fakeDocker, []string{"list"})
}
func TestKillContainer(t *testing.T) {
fakeDocker := FakeDockerClient{
err: nil,
}
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-06 23:40:48 +00:00
}
fakeDocker.containerList = []docker.APIContainers{
2014-06-12 21:09:40 +00:00
{
2014-06-06 23:40:48 +00:00
Names: []string{"foo"},
},
2014-06-12 21:09:40 +00:00
{
2014-06-06 23:40:48 +00:00
Names: []string{"bar"},
},
}
fakeDocker.container = &docker.Container{
ID: "foobar",
}
err := kubelet.KillContainer("foo")
verifyNoError(t, err)
verifyCalls(t, fakeDocker, []string{"list", "stop"})
}
func TestResponseToContainersNil(t *testing.T) {
kubelet := makeTestKubelet()
2014-06-06 23:40:48 +00:00
list, err := kubelet.ResponseToManifests(&etcd.Response{Node: nil})
if len(list) != 0 {
t.Errorf("Unexpected non-zero list: %#v", list)
}
if err == nil {
t.Error("Unexpected non-error")
}
}
func TestResponseToManifests(t *testing.T) {
kubelet := makeTestKubelet()
2014-06-06 23:40:48 +00:00
list, err := kubelet.ResponseToManifests(&etcd.Response{
Node: &etcd.Node{
Value: util.MakeJSONString([]api.ContainerManifest{
2014-06-12 21:09:40 +00:00
{Id: "foo"},
{Id: "bar"},
2014-06-06 23:40:48 +00:00
}),
},
})
if len(list) != 2 || list[0].Id != "foo" || list[1].Id != "bar" {
t.Errorf("Unexpected list: %#v", list)
}
expectNoError(t, err)
}
type channelReader struct {
list [][]api.ContainerManifest
wg sync.WaitGroup
}
func startReading(channel <-chan manifestUpdate) *channelReader {
2014-06-06 23:40:48 +00:00
cr := &channelReader{}
cr.wg.Add(1)
go func() {
for {
update, ok := <-channel
2014-06-06 23:40:48 +00:00
if !ok {
break
}
cr.list = append(cr.list, update.manifests)
2014-06-06 23:40:48 +00:00
}
cr.wg.Done()
}()
return cr
}
func (cr *channelReader) GetList() [][]api.ContainerManifest {
cr.wg.Wait()
return cr.list
}
func TestGetKubeletStateFromEtcdNoData(t *testing.T) {
fakeClient := util.MakeFakeEtcdClient(t)
2014-06-06 23:40:48 +00:00
kubelet := Kubelet{
EtcdClient: fakeClient,
2014-06-06 23:40:48 +00:00
}
channel := make(chan manifestUpdate)
2014-06-06 23:40:48 +00:00
reader := startReading(channel)
fakeClient.Data["/registry/hosts/machine/kubelet"] = util.EtcdResponseWithError{
2014-06-06 23:40:48 +00:00
R: &etcd.Response{},
E: nil,
}
err := kubelet.getKubeletStateFromEtcd("/registry/hosts/machine", channel)
if err == nil {
t.Error("Unexpected no err.")
}
close(channel)
list := reader.GetList()
if len(list) != 0 {
t.Errorf("Unexpected list: %#v", list)
}
}
func TestGetKubeletStateFromEtcd(t *testing.T) {
fakeClient := util.MakeFakeEtcdClient(t)
2014-06-06 23:40:48 +00:00
kubelet := Kubelet{
EtcdClient: fakeClient,
2014-06-06 23:40:48 +00:00
}
channel := make(chan manifestUpdate)
2014-06-06 23:40:48 +00:00
reader := startReading(channel)
fakeClient.Data["/registry/hosts/machine/kubelet"] = util.EtcdResponseWithError{
2014-06-06 23:40:48 +00:00
R: &etcd.Response{
Node: &etcd.Node{
Value: util.MakeJSONString([]api.Container{}),
},
},
E: nil,
}
err := kubelet.getKubeletStateFromEtcd("/registry/hosts/machine", channel)
expectNoError(t, err)
close(channel)
list := reader.GetList()
if len(list) != 1 {
t.Errorf("Unexpected list: %#v", list)
}
}
func TestGetKubeletStateFromEtcdNotFound(t *testing.T) {
fakeClient := util.MakeFakeEtcdClient(t)
2014-06-06 23:40:48 +00:00
kubelet := Kubelet{
EtcdClient: fakeClient,
2014-06-06 23:40:48 +00:00
}
channel := make(chan manifestUpdate)
2014-06-06 23:40:48 +00:00
reader := startReading(channel)
fakeClient.Data["/registry/hosts/machine/kubelet"] = util.EtcdResponseWithError{
2014-06-06 23:40:48 +00:00
R: &etcd.Response{},
E: &etcd.EtcdError{
ErrorCode: 100,
},
}
err := kubelet.getKubeletStateFromEtcd("/registry/hosts/machine", channel)
expectNoError(t, err)
close(channel)
list := reader.GetList()
if len(list) != 0 {
t.Errorf("Unexpected list: %#v", list)
}
}
func TestGetKubeletStateFromEtcdError(t *testing.T) {
fakeClient := util.MakeFakeEtcdClient(t)
2014-06-06 23:40:48 +00:00
kubelet := Kubelet{
EtcdClient: fakeClient,
2014-06-06 23:40:48 +00:00
}
channel := make(chan manifestUpdate)
2014-06-06 23:40:48 +00:00
reader := startReading(channel)
fakeClient.Data["/registry/hosts/machine/kubelet"] = util.EtcdResponseWithError{
2014-06-06 23:40:48 +00:00
R: &etcd.Response{},
E: &etcd.EtcdError{
ErrorCode: 200, // non not found error
},
}
err := kubelet.getKubeletStateFromEtcd("/registry/hosts/machine", channel)
if err == nil {
t.Error("Unexpected non-error")
}
close(channel)
list := reader.GetList()
if len(list) != 0 {
t.Errorf("Unexpected list: %#v", list)
}
}
func TestSyncManifestsDoesNothing(t *testing.T) {
fakeDocker := FakeDockerClient{
err: nil,
}
fakeDocker.containerList = []docker.APIContainers{
2014-06-12 21:09:40 +00:00
{
// format is k8s--<container-id>--<manifest-id>
Names: []string{"/k8s--bar--foo"},
2014-06-06 23:40:48 +00:00
ID: "1234",
},
{
// network container
Names: []string{"/k8s--net--foo--"},
ID: "9876",
},
2014-06-06 23:40:48 +00:00
}
fakeDocker.container = &docker.Container{
ID: "1234",
}
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-06 23:40:48 +00:00
}
err := kubelet.SyncManifests([]api.ContainerManifest{
2014-06-12 21:09:40 +00:00
{
2014-06-06 23:40:48 +00:00
Id: "foo",
Containers: []api.Container{
2014-06-12 21:09:40 +00:00
{Name: "bar"},
2014-06-06 23:40:48 +00:00
},
},
})
expectNoError(t, err)
if len(fakeDocker.called) != 5 ||
2014-06-06 23:40:48 +00:00
fakeDocker.called[0] != "list" ||
fakeDocker.called[1] != "list" ||
fakeDocker.called[2] != "list" ||
fakeDocker.called[3] != "inspect" ||
fakeDocker.called[4] != "list" {
2014-06-06 23:40:48 +00:00
t.Errorf("Unexpected call sequence: %#v", fakeDocker.called)
}
}
func TestSyncManifestsDeletes(t *testing.T) {
fakeDocker := FakeDockerClient{
err: nil,
}
fakeDocker.containerList = []docker.APIContainers{
2014-06-12 21:09:40 +00:00
{
// the k8s prefix is required for the kubelet to manage the container
Names: []string{"/k8s--foo--bar"},
2014-06-06 23:40:48 +00:00
ID: "1234",
},
{
// network container
Names: []string{"/k8s--net--foo--"},
ID: "9876",
},
{
Names: []string{"foo"},
ID: "4567",
},
2014-06-06 23:40:48 +00:00
}
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-06 23:40:48 +00:00
}
err := kubelet.SyncManifests([]api.ContainerManifest{})
expectNoError(t, err)
if len(fakeDocker.called) != 5 ||
2014-06-06 23:40:48 +00:00
fakeDocker.called[0] != "list" ||
fakeDocker.called[1] != "list" ||
fakeDocker.called[2] != "stop" ||
fakeDocker.called[3] != "list" ||
fakeDocker.called[4] != "stop" ||
fakeDocker.stopped[0] != "1234" ||
fakeDocker.stopped[1] != "9876" {
t.Errorf("Unexpected call sequence: %#v %s", fakeDocker.called, fakeDocker.stopped)
2014-06-06 23:40:48 +00:00
}
}
func TestEventWriting(t *testing.T) {
fakeEtcd := util.MakeFakeEtcdClient(t)
kubelet := &Kubelet{
EtcdClient: fakeEtcd,
}
expectedEvent := api.Event{
Event: "test",
Container: &api.Container{
Name: "foo",
},
}
err := kubelet.LogEvent(&expectedEvent)
expectNoError(t, err)
if fakeEtcd.Ix != 1 {
t.Errorf("Unexpected number of children added: %d, expected 1", fakeEtcd.Ix)
}
response, err := fakeEtcd.Get("/events/foo/1", false, false)
expectNoError(t, err)
var event api.Event
err = json.Unmarshal([]byte(response.Node.Value), &event)
expectNoError(t, err)
if event.Event != expectedEvent.Event ||
event.Container.Name != expectedEvent.Container.Name {
t.Errorf("Event's don't match. Expected: %#v Saw: %#v", expectedEvent, event)
}
}
func TestEventWritingError(t *testing.T) {
fakeEtcd := util.MakeFakeEtcdClient(t)
kubelet := &Kubelet{
EtcdClient: fakeEtcd,
}
fakeEtcd.Err = fmt.Errorf("test error")
err := kubelet.LogEvent(&api.Event{
Event: "test",
Container: &api.Container{
Name: "foo",
},
})
if err == nil {
t.Errorf("Unexpected non-error")
}
}
func TestMakeEnvVariables(t *testing.T) {
container := api.Container{
Env: []api.EnvVar{
2014-06-13 01:34:47 +00:00
{
Name: "foo",
Value: "bar",
},
2014-06-13 01:34:47 +00:00
{
Name: "baz",
Value: "blah",
},
},
}
vars := makeEnvironmentVariables(&container)
if len(vars) != len(container.Env) {
t.Errorf("Vars don't match. Expected: %#v Found: %#v", container.Env, vars)
}
for ix, env := range container.Env {
value := fmt.Sprintf("%s=%s", env.Name, env.Value)
if value != vars[ix] {
t.Errorf("Unexpected value: %s. Expected: %s", vars[ix], value)
}
}
}
func TestMakeVolumesAndBinds(t *testing.T) {
container := api.Container{
VolumeMounts: []api.VolumeMount{
2014-06-13 01:34:47 +00:00
{
MountPath: "/mnt/path",
Name: "disk",
ReadOnly: false,
},
2014-06-13 01:34:47 +00:00
{
MountPath: "/mnt/path2",
Name: "disk2",
ReadOnly: true,
2014-06-19 23:59:48 +00:00
MountType: "LOCAL",
},
{
MountPath: "/mnt/path3",
Name: "disk3",
ReadOnly: false,
MountType: "HOST",
},
},
}
volumes, binds := makeVolumesAndBinds(&container)
2014-06-19 23:59:48 +00:00
expectedVolumes := []string{"/mnt/path", "/mnt/path2"}
expectedBinds := []string{"/exports/disk:/mnt/path", "/exports/disk2:/mnt/path2:ro", "/mnt/path3:/mnt/path3"}
if len(volumes) != len(expectedVolumes) {
t.Errorf("Unexpected volumes. Expected %#v got %#v. Container was: %#v", expectedVolumes, volumes, container)
}
for _, expectedVolume := range expectedVolumes {
if _, ok := volumes[expectedVolume]; !ok {
t.Errorf("Volumes map is missing key: %s. %#v", expectedVolume, volumes)
}
}
2014-06-19 23:59:48 +00:00
if len(binds) != len(expectedBinds) {
t.Errorf("Unexpected binds: Expected %# got %#v. Container was: %#v", expectedBinds, binds, container)
}
verifyStringArrayEquals(t, binds, expectedBinds)
}
func TestMakePortsAndBindings(t *testing.T) {
container := api.Container{
Ports: []api.Port{
2014-06-13 01:34:47 +00:00
{
ContainerPort: 80,
HostPort: 8080,
},
2014-06-13 01:34:47 +00:00
{
ContainerPort: 443,
HostPort: 443,
2014-06-16 04:57:29 +00:00
Protocol: "tcp",
2014-06-16 04:19:35 +00:00
},
{
ContainerPort: 444,
HostPort: 444,
2014-06-16 04:57:29 +00:00
Protocol: "udp",
2014-06-16 04:19:35 +00:00
},
{
ContainerPort: 445,
HostPort: 445,
2014-06-16 04:57:29 +00:00
Protocol: "foobar",
},
},
}
exposedPorts, bindings := makePortsAndBindings(&container)
if len(container.Ports) != len(exposedPorts) ||
len(container.Ports) != len(bindings) {
t.Errorf("Unexpected ports and bindings, %#v %#v %#v", container, exposedPorts, bindings)
}
2014-06-16 04:57:29 +00:00
for key, value := range bindings {
switch value[0].HostPort {
2014-06-16 04:19:35 +00:00
case "8080":
2014-06-16 04:57:29 +00:00
if !reflect.DeepEqual(docker.Port("80/tcp"), key) {
t.Errorf("Unexpected docker port: %#v", key)
2014-06-16 04:19:35 +00:00
}
case "443":
2014-06-16 04:57:29 +00:00
if !reflect.DeepEqual(docker.Port("443/tcp"), key) {
t.Errorf("Unexpected docker port: %#v", key)
2014-06-16 04:19:35 +00:00
}
case "444":
2014-06-16 04:57:29 +00:00
if !reflect.DeepEqual(docker.Port("444/udp"), key) {
t.Errorf("Unexpected docker port: %#v", key)
2014-06-16 04:19:35 +00:00
}
case "445":
2014-06-16 04:57:29 +00:00
if !reflect.DeepEqual(docker.Port("445/tcp"), key) {
t.Errorf("Unexpected docker port: %#v", key)
2014-06-16 04:19:35 +00:00
}
2014-06-16 04:57:29 +00:00
}
}
2014-06-16 04:19:35 +00:00
}
func TestExtractFromNonExistentFile(t *testing.T) {
kubelet := Kubelet{}
2014-06-19 20:06:52 +00:00
_, err := kubelet.extractFromFile("/some/fake/file")
if err == nil {
t.Error("Unexpected non-error.")
}
}
func TestExtractFromBadDataFile(t *testing.T) {
kubelet := Kubelet{}
2014-06-20 16:31:18 +00:00
badData := []byte{1, 2, 3}
file, err := ioutil.TempFile("", "foo")
expectNoError(t, err)
name := file.Name()
file.Close()
2014-06-20 16:31:18 +00:00
ioutil.WriteFile(name, badData, 0755)
2014-06-19 20:06:52 +00:00
_, err = kubelet.extractFromFile(name)
if err == nil {
t.Error("Unexpected non-error.")
}
}
2014-06-20 16:31:18 +00:00
func TestExtractFromValidDataFile(t *testing.T) {
kubelet := Kubelet{}
2014-06-20 16:31:18 +00:00
2014-06-19 20:06:52 +00:00
manifest := api.ContainerManifest{Id: "bar"}
data, err := json.Marshal(manifest)
expectNoError(t, err)
file, err := ioutil.TempFile("", "foo")
expectNoError(t, err)
name := file.Name()
expectNoError(t, file.Close())
2014-06-20 16:31:18 +00:00
ioutil.WriteFile(name, data, 0755)
2014-06-19 20:06:52 +00:00
read, err := kubelet.extractFromFile(name)
expectNoError(t, err)
2014-06-19 20:06:52 +00:00
if !reflect.DeepEqual(read, manifest) {
t.Errorf("Unexpected difference. Expected %#v, got %#v", manifest, read)
}
2014-06-19 20:06:52 +00:00
}
func TestExtractFromEmptyDir(t *testing.T) {
kubelet := Kubelet{}
dirName, err := ioutil.TempDir("", "foo")
expectNoError(t, err)
_, err = kubelet.extractFromDir(dirName)
expectNoError(t, err)
}
func TestExtractFromDir(t *testing.T) {
kubelet := Kubelet{}
manifests := []api.ContainerManifest{
2014-06-19 22:14:57 +00:00
{Id: "aaaa"},
{Id: "bbbb"},
2014-06-19 20:06:52 +00:00
}
dirName, err := ioutil.TempDir("", "foo")
expectNoError(t, err)
for _, manifest := range manifests {
data, err := json.Marshal(manifest)
expectNoError(t, err)
2014-06-19 22:14:57 +00:00
file, err := ioutil.TempFile(dirName, manifest.Id)
2014-06-19 20:06:52 +00:00
expectNoError(t, err)
name := file.Name()
expectNoError(t, file.Close())
ioutil.WriteFile(name, data, 0755)
}
read, err := kubelet.extractFromDir(dirName)
expectNoError(t, err)
if !reflect.DeepEqual(read, manifests) {
t.Errorf("Unexpected difference. Expected %#v, got %#v", manifests, read)
}
}
func TestExtractFromHttpBadness(t *testing.T) {
kubelet := Kubelet{}
updateChannel := make(chan manifestUpdate)
reader := startReading(updateChannel)
2014-06-20 16:31:18 +00:00
err := kubelet.extractFromHTTP("http://localhost:12345", updateChannel)
if err == nil {
t.Error("Unexpected non-error.")
}
close(updateChannel)
2014-06-20 16:31:18 +00:00
list := reader.GetList()
if len(list) != 0 {
t.Errorf("Unexpected list: %#v", list)
}
}
func TestExtractFromHttpSingle(t *testing.T) {
kubelet := Kubelet{}
updateChannel := make(chan manifestUpdate)
reader := startReading(updateChannel)
2014-06-20 16:31:18 +00:00
manifests := []api.ContainerManifest{
{Version: "v1beta1", Id: "foo"},
}
// Taking a single-manifest from a URL allows kubelet to be used
// in the implementation of google's container VM image.
data, err := json.Marshal(manifests[0])
fakeHandler := util.FakeHandler{
StatusCode: 200,
2014-06-20 16:31:18 +00:00
ResponseBody: string(data),
}
testServer := httptest.NewServer(&fakeHandler)
err = kubelet.extractFromHTTP(testServer.URL, updateChannel)
if err != nil {
t.Errorf("Unexpected error: %#v", err)
}
close(updateChannel)
read := reader.GetList()
if len(read) != 1 {
t.Errorf("Unexpected list: %#v", read)
return
}
if !reflect.DeepEqual(manifests, read[0]) {
t.Errorf("Unexpected difference. Expected: %#v, Saw: %#v", manifests, read[0])
}
}
func TestExtractFromHttpMultiple(t *testing.T) {
kubelet := Kubelet{}
updateChannel := make(chan manifestUpdate)
reader := startReading(updateChannel)
manifests := []api.ContainerManifest{
{Version: "v1beta1", Id: "foo"},
{Version: "v1beta1", Id: "bar"},
}
data, err := json.Marshal(manifests)
if err != nil {
t.Fatalf("Some weird json problem: %v", err)
}
t.Logf("Serving: %v", string(data))
fakeHandler := util.FakeHandler{
StatusCode: 200,
ResponseBody: string(data),
}
testServer := httptest.NewServer(&fakeHandler)
err = kubelet.extractFromHTTP(testServer.URL, updateChannel)
if err != nil {
t.Errorf("Unexpected error: %#v", err)
}
close(updateChannel)
read := reader.GetList()
if len(read) != 1 {
t.Errorf("Unexpected list: %#v", read)
return
}
2014-06-20 16:31:18 +00:00
if !reflect.DeepEqual(manifests, read[0]) {
t.Errorf("Unexpected difference. Expected: %#v, Saw: %#v", manifests, read[0])
}
}
func TestWatchEtcd(t *testing.T) {
watchChannel := make(chan *etcd.Response)
updateChannel := make(chan manifestUpdate)
kubelet := Kubelet{}
reader := startReading(updateChannel)
manifest := []api.ContainerManifest{
2014-06-13 01:34:47 +00:00
{
Id: "foo",
},
}
data, err := json.Marshal(manifest)
expectNoError(t, err)
go kubelet.WatchEtcd(watchChannel, updateChannel)
watchChannel <- &etcd.Response{
Node: &etcd.Node{
Value: string(data),
},
}
close(watchChannel)
close(updateChannel)
read := reader.GetList()
2014-06-26 23:56:02 +00:00
if len(read) != 1 {
t.Errorf("Expected number of results: %v", len(read))
} else if !reflect.DeepEqual(read[0], manifest) {
t.Errorf("Unexpected manifest(s) %#v %#v", read[0], manifest)
}
}
2014-06-19 20:22:20 +00:00
type mockCadvisorClient struct {
mock.Mock
}
func (self *mockCadvisorClient) ContainerInfo(name string) (*info.ContainerInfo, error) {
args := self.Called(name)
return args.Get(0).(*info.ContainerInfo), args.Error(1)
}
func (self *mockCadvisorClient) MachineInfo() (*info.MachineInfo, error) {
args := self.Called()
return args.Get(0).(*info.MachineInfo), args.Error(1)
}
func areSamePercentiles(
cadvisorPercentiles []info.Percentile,
kubePercentiles []api.Percentile,
t *testing.T,
) {
if len(cadvisorPercentiles) != len(kubePercentiles) {
t.Errorf("cadvisor gives %v percentiles; kubelet got %v", len(cadvisorPercentiles), len(kubePercentiles))
return
}
for _, ap := range cadvisorPercentiles {
found := false
for _, kp := range kubePercentiles {
if ap.Percentage == kp.Percentage {
found = true
if ap.Value != kp.Value {
t.Errorf("%v percentile from cadvisor is %v; kubelet got %v",
ap.Percentage,
ap.Value,
kp.Value)
}
}
}
if !found {
t.Errorf("Unable to find %v percentile in kubelet's data", ap.Percentage)
}
}
}
func TestGetContainerStats(t *testing.T) {
containerId := "ab2cdf"
containerPath := fmt.Sprintf("/docker/%v", containerId)
containerInfo := &info.ContainerInfo{
ContainerReference: info.ContainerReference{
Name: containerPath,
},
StatsPercentiles: &info.ContainerStatsPercentiles{
MaxMemoryUsage: 1024000,
MemoryUsagePercentiles: []info.Percentile{
2014-06-19 20:59:20 +00:00
{50, 100},
{80, 180},
{90, 190},
2014-06-19 20:22:20 +00:00
},
CpuUsagePercentiles: []info.Percentile{
2014-06-19 20:59:20 +00:00
{51, 101},
{81, 181},
{91, 191},
2014-06-19 20:22:20 +00:00
},
},
}
fakeDocker := FakeDockerClient{
err: nil,
}
mockCadvisor := &mockCadvisorClient{}
mockCadvisor.On("ContainerInfo", containerPath).Return(containerInfo, nil)
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-19 20:22:20 +00:00
CadvisorClient: mockCadvisor,
}
fakeDocker.containerList = []docker.APIContainers{
{
Names: []string{"foo"},
ID: containerId,
},
}
stats, err := kubelet.GetContainerStats("foo")
if err != nil {
t.Errorf("unexpected error: %v", err)
}
if stats.MaxMemoryUsage != containerInfo.StatsPercentiles.MaxMemoryUsage {
t.Errorf("wrong max memory usage")
}
areSamePercentiles(containerInfo.StatsPercentiles.CpuUsagePercentiles, stats.CpuUsagePercentiles, t)
areSamePercentiles(containerInfo.StatsPercentiles.MemoryUsagePercentiles, stats.MemoryUsagePercentiles, t)
mockCadvisor.AssertExpectations(t)
}
2014-06-19 20:25:54 +00:00
func TestGetContainerStatsWithoutCadvisor(t *testing.T) {
fakeDocker := FakeDockerClient{
err: nil,
}
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-19 20:25:54 +00:00
}
fakeDocker.containerList = []docker.APIContainers{
{
Names: []string{"foo"},
},
}
stats, _ := kubelet.GetContainerStats("foo")
// When there's no cAdvisor, the stats should be either nil or empty
if stats == nil {
return
}
if stats.MaxMemoryUsage != 0 {
t.Errorf("MaxMemoryUsage is %v even if there's no cadvisor", stats.MaxMemoryUsage)
}
if len(stats.CpuUsagePercentiles) > 0 {
t.Errorf("Cpu usage percentiles is not empty (%+v) even if there's no cadvisor", stats.CpuUsagePercentiles)
}
if len(stats.MemoryUsagePercentiles) > 0 {
t.Errorf("Memory usage percentiles is not empty (%+v) even if there's no cadvisor", stats.MemoryUsagePercentiles)
}
}
2014-06-19 20:34:26 +00:00
func TestGetContainerStatsWhenCadvisorFailed(t *testing.T) {
containerId := "ab2cdf"
containerPath := fmt.Sprintf("/docker/%v", containerId)
fakeDocker := FakeDockerClient{
err: nil,
}
containerInfo := &info.ContainerInfo{}
mockCadvisor := &mockCadvisorClient{}
expectedErr := fmt.Errorf("some error")
mockCadvisor.On("ContainerInfo", containerPath).Return(containerInfo, expectedErr)
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-19 20:34:26 +00:00
CadvisorClient: mockCadvisor,
}
fakeDocker.containerList = []docker.APIContainers{
{
Names: []string{"foo"},
ID: containerId,
},
}
stats, err := kubelet.GetContainerStats("foo")
if stats != nil {
t.Errorf("non-nil stats on error")
}
if err == nil {
t.Errorf("expect error but received nil error")
return
}
if err.Error() != expectedErr.Error() {
t.Errorf("wrong error message. expect %v, got %v", err, expectedErr)
}
mockCadvisor.AssertExpectations(t)
}
func TestGetContainerStatsOnNonExistContainer(t *testing.T) {
fakeDocker := FakeDockerClient{
err: nil,
}
mockCadvisor := &mockCadvisorClient{}
kubelet := Kubelet{
DockerClient: &fakeDocker,
DockerPuller: &FakeDockerPuller{},
2014-06-19 20:34:26 +00:00
CadvisorClient: mockCadvisor,
}
fakeDocker.containerList = []docker.APIContainers{}
stats, _ := kubelet.GetContainerStats("foo")
if stats != nil {
t.Errorf("non-nil stats on non exist container")
}
mockCadvisor.AssertExpectations(t)
}