
This is part of the "Debug Containers" feature and is hidden behind a feature gate. Debug containers have no stored spec, so this new runtime label allows the kubelet to treat containers differently without relying on spec.
292 lines
8.2 KiB
Go
292 lines
8.2 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 kuberuntime
|
|
|
|
import (
|
|
"reflect"
|
|
"testing"
|
|
|
|
"k8s.io/api/core/v1"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/util/intstr"
|
|
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
|
kubecontainer "k8s.io/kubernetes/pkg/kubelet/container"
|
|
)
|
|
|
|
func TestContainerLabels(t *testing.T) {
|
|
deletionGracePeriod := int64(10)
|
|
terminationGracePeriod := int64(10)
|
|
lifecycle := &v1.Lifecycle{
|
|
// Left PostStart as nil
|
|
PreStop: &v1.Handler{
|
|
Exec: &v1.ExecAction{
|
|
Command: []string{"action1", "action2"},
|
|
},
|
|
HTTPGet: &v1.HTTPGetAction{
|
|
Path: "path",
|
|
Host: "host",
|
|
Port: intstr.FromInt(8080),
|
|
Scheme: "scheme",
|
|
},
|
|
TCPSocket: &v1.TCPSocketAction{
|
|
Port: intstr.FromString("80"),
|
|
},
|
|
},
|
|
}
|
|
container := &v1.Container{
|
|
Name: "test_container",
|
|
TerminationMessagePath: "/somepath",
|
|
Lifecycle: lifecycle,
|
|
}
|
|
pod := &v1.Pod{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "test_pod",
|
|
Namespace: "test_pod_namespace",
|
|
UID: "test_pod_uid",
|
|
DeletionGracePeriodSeconds: &deletionGracePeriod,
|
|
},
|
|
Spec: v1.PodSpec{
|
|
Containers: []v1.Container{*container},
|
|
TerminationGracePeriodSeconds: &terminationGracePeriod,
|
|
},
|
|
}
|
|
|
|
var tests = []struct {
|
|
description string
|
|
featuresCreated string // Features enabled when container is created
|
|
featuresStatus string // Features enabled when container status is read
|
|
typeLabel kubecontainer.ContainerType
|
|
expected *labeledContainerInfo
|
|
}{
|
|
{
|
|
"Debug containers disabled",
|
|
"DebugContainers=False",
|
|
"DebugContainers=False",
|
|
"ignored",
|
|
&labeledContainerInfo{
|
|
PodName: pod.Name,
|
|
PodNamespace: pod.Namespace,
|
|
PodUID: pod.UID,
|
|
ContainerName: container.Name,
|
|
ContainerType: "",
|
|
},
|
|
},
|
|
{
|
|
"Regular containers",
|
|
"DebugContainers=True",
|
|
"DebugContainers=True",
|
|
kubecontainer.ContainerTypeRegular,
|
|
&labeledContainerInfo{
|
|
PodName: pod.Name,
|
|
PodNamespace: pod.Namespace,
|
|
PodUID: pod.UID,
|
|
ContainerName: container.Name,
|
|
ContainerType: kubecontainer.ContainerTypeRegular,
|
|
},
|
|
},
|
|
{
|
|
"Init containers",
|
|
"DebugContainers=True",
|
|
"DebugContainers=True",
|
|
kubecontainer.ContainerTypeInit,
|
|
&labeledContainerInfo{
|
|
PodName: pod.Name,
|
|
PodNamespace: pod.Namespace,
|
|
PodUID: pod.UID,
|
|
ContainerName: container.Name,
|
|
ContainerType: kubecontainer.ContainerTypeInit,
|
|
},
|
|
},
|
|
{
|
|
"Created without type label",
|
|
"DebugContainers=False",
|
|
"DebugContainers=True",
|
|
"ignored",
|
|
&labeledContainerInfo{
|
|
PodName: pod.Name,
|
|
PodNamespace: pod.Namespace,
|
|
PodUID: pod.UID,
|
|
ContainerName: container.Name,
|
|
ContainerType: "",
|
|
},
|
|
},
|
|
{
|
|
"Created with type label, subsequently disabled",
|
|
"DebugContainers=True",
|
|
"DebugContainers=False",
|
|
kubecontainer.ContainerTypeRegular,
|
|
&labeledContainerInfo{
|
|
PodName: pod.Name,
|
|
PodNamespace: pod.Namespace,
|
|
PodUID: pod.UID,
|
|
ContainerName: container.Name,
|
|
ContainerType: "",
|
|
},
|
|
},
|
|
}
|
|
|
|
// Test whether we can get right information from label
|
|
for _, test := range tests {
|
|
utilfeature.DefaultFeatureGate.Set(test.featuresCreated)
|
|
labels := newContainerLabels(container, pod, test.typeLabel)
|
|
utilfeature.DefaultFeatureGate.Set(test.featuresStatus)
|
|
containerInfo := getContainerInfoFromLabels(labels)
|
|
if !reflect.DeepEqual(containerInfo, test.expected) {
|
|
t.Errorf("%v: expected %v, got %v", test.description, test.expected, containerInfo)
|
|
}
|
|
}
|
|
utilfeature.DefaultFeatureGate.Set("DebugContainers=False")
|
|
}
|
|
|
|
func TestContainerAnnotations(t *testing.T) {
|
|
restartCount := 5
|
|
deletionGracePeriod := int64(10)
|
|
terminationGracePeriod := int64(10)
|
|
lifecycle := &v1.Lifecycle{
|
|
// Left PostStart as nil
|
|
PreStop: &v1.Handler{
|
|
Exec: &v1.ExecAction{
|
|
Command: []string{"action1", "action2"},
|
|
},
|
|
HTTPGet: &v1.HTTPGetAction{
|
|
Path: "path",
|
|
Host: "host",
|
|
Port: intstr.FromInt(8080),
|
|
Scheme: "scheme",
|
|
},
|
|
TCPSocket: &v1.TCPSocketAction{
|
|
Port: intstr.FromString("80"),
|
|
},
|
|
},
|
|
}
|
|
containerPorts := []v1.ContainerPort{
|
|
{
|
|
Name: "http",
|
|
HostPort: 80,
|
|
ContainerPort: 8080,
|
|
Protocol: v1.ProtocolTCP,
|
|
},
|
|
{
|
|
Name: "https",
|
|
HostPort: 443,
|
|
ContainerPort: 6443,
|
|
Protocol: v1.ProtocolTCP,
|
|
},
|
|
}
|
|
container := &v1.Container{
|
|
Name: "test_container",
|
|
Ports: containerPorts,
|
|
TerminationMessagePath: "/somepath",
|
|
Lifecycle: lifecycle,
|
|
}
|
|
pod := &v1.Pod{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "test_pod",
|
|
Namespace: "test_pod_namespace",
|
|
UID: "test_pod_uid",
|
|
DeletionGracePeriodSeconds: &deletionGracePeriod,
|
|
},
|
|
Spec: v1.PodSpec{
|
|
Containers: []v1.Container{*container},
|
|
TerminationGracePeriodSeconds: &terminationGracePeriod,
|
|
},
|
|
}
|
|
expected := &annotatedContainerInfo{
|
|
ContainerPorts: containerPorts,
|
|
PodDeletionGracePeriod: pod.DeletionGracePeriodSeconds,
|
|
PodTerminationGracePeriod: pod.Spec.TerminationGracePeriodSeconds,
|
|
Hash: kubecontainer.HashContainer(container),
|
|
RestartCount: restartCount,
|
|
TerminationMessagePath: container.TerminationMessagePath,
|
|
PreStopHandler: container.Lifecycle.PreStop,
|
|
}
|
|
|
|
// Test whether we can get right information from label
|
|
annotations := newContainerAnnotations(container, pod, restartCount)
|
|
containerInfo := getContainerInfoFromAnnotations(annotations)
|
|
if !reflect.DeepEqual(containerInfo, expected) {
|
|
t.Errorf("expected %v, got %v", expected, containerInfo)
|
|
}
|
|
|
|
// Test when DeletionGracePeriodSeconds, TerminationGracePeriodSeconds and Lifecycle are nil,
|
|
// the information got from annotations should also be nil
|
|
container.Lifecycle = nil
|
|
pod.DeletionGracePeriodSeconds = nil
|
|
pod.Spec.TerminationGracePeriodSeconds = nil
|
|
expected.PodDeletionGracePeriod = nil
|
|
expected.PodTerminationGracePeriod = nil
|
|
expected.PreStopHandler = nil
|
|
// Because container is changed, the Hash should be updated
|
|
expected.Hash = kubecontainer.HashContainer(container)
|
|
annotations = newContainerAnnotations(container, pod, restartCount)
|
|
containerInfo = getContainerInfoFromAnnotations(annotations)
|
|
if !reflect.DeepEqual(containerInfo, expected) {
|
|
t.Errorf("expected %v, got %v", expected, containerInfo)
|
|
}
|
|
}
|
|
|
|
func TestPodLabels(t *testing.T) {
|
|
pod := &v1.Pod{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "test_pod",
|
|
Namespace: "test_pod_namespace",
|
|
UID: "test_pod_uid",
|
|
Labels: map[string]string{"foo": "bar"},
|
|
},
|
|
Spec: v1.PodSpec{
|
|
Containers: []v1.Container{},
|
|
},
|
|
}
|
|
expected := &labeledPodSandboxInfo{
|
|
Labels: pod.Labels,
|
|
PodName: pod.Name,
|
|
PodNamespace: pod.Namespace,
|
|
PodUID: pod.UID,
|
|
}
|
|
|
|
// Test whether we can get right information from label
|
|
labels := newPodLabels(pod)
|
|
podSandboxInfo := getPodSandboxInfoFromLabels(labels)
|
|
if !reflect.DeepEqual(podSandboxInfo, expected) {
|
|
t.Errorf("expected %v, got %v", expected, podSandboxInfo)
|
|
}
|
|
}
|
|
|
|
func TestPodAnnotations(t *testing.T) {
|
|
pod := &v1.Pod{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "test_pod",
|
|
Namespace: "test_pod_namespace",
|
|
UID: "test_pod_uid",
|
|
Annotations: map[string]string{"foo": "bar"},
|
|
},
|
|
Spec: v1.PodSpec{
|
|
Containers: []v1.Container{},
|
|
},
|
|
}
|
|
expected := &annotatedPodSandboxInfo{
|
|
Annotations: map[string]string{"foo": "bar"},
|
|
}
|
|
|
|
// Test whether we can get right information from annotations
|
|
annotations := newPodAnnotations(pod)
|
|
podSandboxInfo := getPodSandboxInfoFromAnnotations(annotations)
|
|
if !reflect.DeepEqual(podSandboxInfo, expected) {
|
|
t.Errorf("expected %v, got %v", expected, podSandboxInfo)
|
|
}
|
|
}
|