mirror of https://github.com/k3s-io/k3s
279 lines
7.8 KiB
Go
279 lines
7.8 KiB
Go
/*
|
|
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.
|
|
*/
|
|
|
|
package dockertools
|
|
|
|
import (
|
|
"fmt"
|
|
"reflect"
|
|
"sort"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/kubelet/container"
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/util"
|
|
"github.com/fsouza/go-dockerclient"
|
|
)
|
|
|
|
// FakeDockerClient is a simple fake docker client, so that kubelet can be run for testing without requiring a real docker setup.
|
|
type FakeDockerClient struct {
|
|
sync.Mutex
|
|
ContainerList []docker.APIContainers
|
|
Container *docker.Container
|
|
ContainerMap map[string]*docker.Container
|
|
Image *docker.Image
|
|
Images []docker.APIImages
|
|
Err error
|
|
called []string
|
|
Stopped []string
|
|
pulled []string
|
|
Created []string
|
|
Removed []string
|
|
RemovedImages util.StringSet
|
|
VersionInfo docker.Env
|
|
}
|
|
|
|
func (f *FakeDockerClient) ClearCalls() {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
f.called = []string{}
|
|
f.Stopped = []string{}
|
|
f.pulled = []string{}
|
|
f.Created = []string{}
|
|
f.Removed = []string{}
|
|
}
|
|
|
|
func (f *FakeDockerClient) AssertCalls(calls []string) (err error) {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
|
|
if !reflect.DeepEqual(calls, f.called) {
|
|
err = fmt.Errorf("expected %#v, got %#v", calls, f.called)
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func (f *FakeDockerClient) AssertUnorderedCalls(calls []string) (err error) {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
|
|
actual := make([]string, len(calls))
|
|
expected := make([]string, len(f.called))
|
|
copy(actual, calls)
|
|
copy(expected, f.called)
|
|
|
|
sort.StringSlice(actual).Sort()
|
|
sort.StringSlice(expected).Sort()
|
|
|
|
if !reflect.DeepEqual(actual, expected) {
|
|
err = fmt.Errorf("expected(sorted) %#v, got(sorted) %#v", expected, actual)
|
|
}
|
|
return
|
|
}
|
|
|
|
// ListContainers is a test-spy implementation of DockerInterface.ListContainers.
|
|
// It adds an entry "list" to the internal method call record.
|
|
func (f *FakeDockerClient) ListContainers(options docker.ListContainersOptions) ([]docker.APIContainers, error) {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
f.called = append(f.called, "list")
|
|
return f.ContainerList, f.Err
|
|
}
|
|
|
|
// InspectContainer is a test-spy implementation of DockerInterface.InspectContainer.
|
|
// It adds an entry "inspect" to the internal method call record.
|
|
func (f *FakeDockerClient) InspectContainer(id string) (*docker.Container, error) {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
f.called = append(f.called, "inspect_container")
|
|
if f.ContainerMap != nil {
|
|
if container, ok := f.ContainerMap[id]; ok {
|
|
return container, f.Err
|
|
}
|
|
}
|
|
return f.Container, f.Err
|
|
}
|
|
|
|
// InspectImage is a test-spy implementation of DockerInterface.InspectImage.
|
|
// It adds an entry "inspect" to the internal method call record.
|
|
func (f *FakeDockerClient) InspectImage(name string) (*docker.Image, error) {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
f.called = append(f.called, "inspect_image")
|
|
return f.Image, f.Err
|
|
}
|
|
|
|
// CreateContainer is a test-spy implementation of DockerInterface.CreateContainer.
|
|
// It adds an entry "create" to the internal method call record.
|
|
func (f *FakeDockerClient) CreateContainer(c docker.CreateContainerOptions) (*docker.Container, error) {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
f.called = append(f.called, "create")
|
|
f.Created = append(f.Created, c.Name)
|
|
// This is not a very good fake. We'll just add this container's name to the list.
|
|
// Docker likes to add a '/', so copy that behavior.
|
|
name := "/" + c.Name
|
|
f.ContainerList = append(f.ContainerList, docker.APIContainers{ID: name, Names: []string{name}, Image: c.Config.Image})
|
|
return &docker.Container{ID: name}, nil
|
|
}
|
|
|
|
// StartContainer is a test-spy implementation of DockerInterface.StartContainer.
|
|
// It adds an entry "start" to the internal method call record.
|
|
func (f *FakeDockerClient) StartContainer(id string, hostConfig *docker.HostConfig) error {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
f.called = append(f.called, "start")
|
|
f.Container = &docker.Container{
|
|
ID: id,
|
|
Name: id, // For testing purpose, we set name to id
|
|
Config: &docker.Config{Image: "testimage"},
|
|
HostConfig: hostConfig,
|
|
State: docker.State{
|
|
Running: true,
|
|
Pid: 42,
|
|
},
|
|
NetworkSettings: &docker.NetworkSettings{IPAddress: "1.2.3.4"},
|
|
}
|
|
return f.Err
|
|
}
|
|
|
|
// StopContainer is a test-spy implementation of DockerInterface.StopContainer.
|
|
// It adds an entry "stop" to the internal method call record.
|
|
func (f *FakeDockerClient) StopContainer(id string, timeout uint) error {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
f.called = append(f.called, "stop")
|
|
f.Stopped = append(f.Stopped, id)
|
|
var newList []docker.APIContainers
|
|
for _, container := range f.ContainerList {
|
|
if container.ID != id {
|
|
newList = append(newList, container)
|
|
}
|
|
}
|
|
f.ContainerList = newList
|
|
return f.Err
|
|
}
|
|
|
|
func (f *FakeDockerClient) RemoveContainer(opts docker.RemoveContainerOptions) error {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
f.called = append(f.called, "remove")
|
|
f.Removed = append(f.Removed, opts.ID)
|
|
return f.Err
|
|
}
|
|
|
|
// Logs is a test-spy implementation of DockerInterface.Logs.
|
|
// It adds an entry "logs" to the internal method call record.
|
|
func (f *FakeDockerClient) Logs(opts docker.LogsOptions) error {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
f.called = append(f.called, "logs")
|
|
return f.Err
|
|
}
|
|
|
|
// PullImage is a test-spy implementation of DockerInterface.StopContainer.
|
|
// It adds an entry "pull" to the internal method call record.
|
|
func (f *FakeDockerClient) PullImage(opts docker.PullImageOptions, auth docker.AuthConfiguration) error {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
f.called = append(f.called, "pull")
|
|
registry := opts.Registry
|
|
if len(registry) != 0 {
|
|
registry = registry + "/"
|
|
}
|
|
f.pulled = append(f.pulled, fmt.Sprintf("%s%s:%s", registry, opts.Repository, opts.Tag))
|
|
return f.Err
|
|
}
|
|
|
|
func (f *FakeDockerClient) Version() (*docker.Env, error) {
|
|
return &f.VersionInfo, nil
|
|
}
|
|
|
|
func (f *FakeDockerClient) CreateExec(_ docker.CreateExecOptions) (*docker.Exec, error) {
|
|
return &docker.Exec{"12345678"}, nil
|
|
}
|
|
|
|
func (f *FakeDockerClient) StartExec(_ string, _ docker.StartExecOptions) error {
|
|
return nil
|
|
}
|
|
|
|
func (f *FakeDockerClient) ListImages(opts docker.ListImagesOptions) ([]docker.APIImages, error) {
|
|
return f.Images, f.Err
|
|
}
|
|
|
|
func (f *FakeDockerClient) RemoveImage(image string) error {
|
|
f.RemovedImages.Insert(image)
|
|
return f.Err
|
|
}
|
|
|
|
// FakeDockerPuller is a stub implementation of DockerPuller.
|
|
type FakeDockerPuller struct {
|
|
sync.Mutex
|
|
|
|
HasImages []string
|
|
ImagesPulled []string
|
|
|
|
// Every pull will return the first error here, and then reslice
|
|
// to remove it. Will give nil errors if this slice is empty.
|
|
ErrorsToInject []error
|
|
}
|
|
|
|
// Pull records the image pull attempt, and optionally injects an error.
|
|
func (f *FakeDockerPuller) Pull(image string) (err error) {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
f.ImagesPulled = append(f.ImagesPulled, image)
|
|
|
|
if len(f.ErrorsToInject) > 0 {
|
|
err = f.ErrorsToInject[0]
|
|
f.ErrorsToInject = f.ErrorsToInject[1:]
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (f *FakeDockerPuller) IsImagePresent(name string) (bool, error) {
|
|
f.Lock()
|
|
defer f.Unlock()
|
|
if f.HasImages == nil {
|
|
return true, nil
|
|
}
|
|
for _, s := range f.HasImages {
|
|
if s == name {
|
|
return true, nil
|
|
}
|
|
}
|
|
return false, nil
|
|
}
|
|
|
|
type FakeDockerCache struct {
|
|
client DockerInterface
|
|
}
|
|
|
|
func NewFakeDockerCache(client DockerInterface) DockerCache {
|
|
return &FakeDockerCache{
|
|
client: client,
|
|
}
|
|
}
|
|
|
|
func (f *FakeDockerCache) GetPods() ([]*container.Pod, error) {
|
|
return GetPods(f.client, false)
|
|
}
|
|
|
|
func (f *FakeDockerCache) ForceUpdateIfOlder(time.Time) error {
|
|
return nil
|
|
}
|