123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285 |
- /*
- 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 common
- import (
- "strings"
- "time"
- "github.com/onsi/ginkgo"
- v1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/klog"
- "k8s.io/kubernetes/test/e2e/framework"
- imageutils "k8s.io/kubernetes/test/utils/image"
- )
- const (
- etcHostsPodName = "test-pod"
- etcHostsHostNetworkPodName = "test-host-network-pod"
- etcHostsPartialContent = "# Kubernetes-managed hosts file."
- etcHostsPath = "/etc/hosts"
- etcHostsOriginalPath = "/etc/hosts-original"
- )
- var etcHostsImageName = imageutils.GetE2EImage(imageutils.Agnhost)
- type KubeletManagedHostConfig struct {
- hostNetworkPod *v1.Pod
- pod *v1.Pod
- f *framework.Framework
- }
- var _ = framework.KubeDescribe("KubeletManagedEtcHosts", func() {
- f := framework.NewDefaultFramework("e2e-kubelet-etc-hosts")
- config := &KubeletManagedHostConfig{
- f: f,
- }
- /*
- Release : v1.9
- Testname: Kubelet, managed etc hosts
- Description: Create a Pod with containers with hostNetwork set to false, one of the containers mounts the /etc/hosts file form the host. Create a second Pod with hostNetwork set to true.
- 1. The Pod with hostNetwork=false MUST have /etc/hosts of containers managed by the Kubelet.
- 2. The Pod with hostNetwork=false but the container mounts /etc/hosts file from the host. The /etc/hosts file MUST not be managed by the Kubelet.
- 3. The Pod with hostNetwork=true , /etc/hosts file MUST not be managed by the Kubelet.
- This test is marked LinuxOnly since Windows cannot mount individual files in Containers.
- */
- framework.ConformanceIt("should test kubelet managed /etc/hosts file [LinuxOnly] [NodeConformance]", func() {
- ginkgo.By("Setting up the test")
- config.setup()
- ginkgo.By("Running the test")
- config.verifyEtcHosts()
- })
- })
- func (config *KubeletManagedHostConfig) verifyEtcHosts() {
- ginkgo.By("Verifying /etc/hosts of container is kubelet-managed for pod with hostNetwork=false")
- assertManagedStatus(config, etcHostsPodName, true, "busybox-1")
- assertManagedStatus(config, etcHostsPodName, true, "busybox-2")
- ginkgo.By("Verifying /etc/hosts of container is not kubelet-managed since container specifies /etc/hosts mount")
- assertManagedStatus(config, etcHostsPodName, false, "busybox-3")
- ginkgo.By("Verifying /etc/hosts content of container is not kubelet-managed for pod with hostNetwork=true")
- assertManagedStatus(config, etcHostsHostNetworkPodName, false, "busybox-1")
- assertManagedStatus(config, etcHostsHostNetworkPodName, false, "busybox-2")
- }
- func (config *KubeletManagedHostConfig) setup() {
- ginkgo.By("Creating hostNetwork=false pod")
- config.createPodWithoutHostNetwork()
- ginkgo.By("Creating hostNetwork=true pod")
- config.createPodWithHostNetwork()
- }
- func (config *KubeletManagedHostConfig) createPodWithoutHostNetwork() {
- podSpec := config.createPodSpec(etcHostsPodName)
- config.pod = config.f.PodClient().CreateSync(podSpec)
- }
- func (config *KubeletManagedHostConfig) createPodWithHostNetwork() {
- podSpec := config.createPodSpecWithHostNetwork(etcHostsHostNetworkPodName)
- config.hostNetworkPod = config.f.PodClient().CreateSync(podSpec)
- }
- func assertManagedStatus(
- config *KubeletManagedHostConfig, podName string, expectedIsManaged bool, name string) {
- // TODO: workaround for https://github.com/kubernetes/kubernetes/issues/34256
- //
- // Retry until timeout for the contents of /etc/hosts to show
- // up. Note: if /etc/hosts is properly mounted, then this will
- // succeed immediately.
- const retryTimeout = 30 * time.Second
- retryCount := 0
- etcHostsContent := ""
- for startTime := time.Now(); time.Since(startTime) < retryTimeout; {
- etcHostsContent = config.getFileContents(podName, name, etcHostsPath)
- etcHostsOriginalContent := config.getFileContents(podName, name, etcHostsOriginalPath)
- // Make sure there is some content in both files
- if len(etcHostsContent) > 0 && len(etcHostsOriginalContent) > 0 {
- // if the files match, kubernetes did not touch the file at all
- // if the file has the header, kubernetes is not using host network
- // and is constructing the file based on Pod IP
- isManaged := strings.HasPrefix(etcHostsContent, etcHostsPartialContent) &&
- etcHostsContent != etcHostsOriginalContent
- if expectedIsManaged == isManaged {
- return
- }
- }
- klog.Warningf(
- "For pod: %s, name: %s, expected %t, (/etc/hosts was %q), (/etc/hosts-original was %q), retryCount: %d",
- podName, name, expectedIsManaged, etcHostsContent, etcHostsOriginalContent, retryCount)
- retryCount++
- time.Sleep(100 * time.Millisecond)
- }
- if expectedIsManaged {
- framework.Failf(
- "/etc/hosts file should be kubelet managed (name: %s, retries: %d). /etc/hosts contains %q",
- name, retryCount, etcHostsContent)
- } else {
- framework.Failf(
- "/etc/hosts file should no be kubelet managed (name: %s, retries: %d). /etc/hosts contains %q",
- name, retryCount, etcHostsContent)
- }
- }
- func (config *KubeletManagedHostConfig) getFileContents(podName, containerName, path string) string {
- return config.f.ExecCommandInContainer(podName, containerName, "cat", path)
- }
- func (config *KubeletManagedHostConfig) createPodSpec(podName string) *v1.Pod {
- hostPathType := new(v1.HostPathType)
- *hostPathType = v1.HostPathType(string(v1.HostPathFileOrCreate))
- pod := &v1.Pod{
- ObjectMeta: metav1.ObjectMeta{
- Name: podName,
- },
- Spec: v1.PodSpec{
- Containers: []v1.Container{
- {
- Name: "busybox-1",
- Image: etcHostsImageName,
- ImagePullPolicy: v1.PullIfNotPresent,
- Command: []string{
- "sleep",
- "900",
- },
- VolumeMounts: []v1.VolumeMount{
- {
- Name: "host-etc-hosts",
- MountPath: etcHostsOriginalPath,
- },
- },
- },
- {
- Name: "busybox-2",
- Image: etcHostsImageName,
- ImagePullPolicy: v1.PullIfNotPresent,
- Command: []string{
- "sleep",
- "900",
- },
- VolumeMounts: []v1.VolumeMount{
- {
- Name: "host-etc-hosts",
- MountPath: etcHostsOriginalPath,
- },
- },
- },
- {
- Name: "busybox-3",
- Image: etcHostsImageName,
- ImagePullPolicy: v1.PullIfNotPresent,
- Command: []string{
- "sleep",
- "900",
- },
- VolumeMounts: []v1.VolumeMount{
- {
- Name: "host-etc-hosts",
- MountPath: etcHostsPath,
- },
- {
- Name: "host-etc-hosts",
- MountPath: etcHostsOriginalPath,
- },
- },
- },
- },
- Volumes: []v1.Volume{
- {
- Name: "host-etc-hosts",
- VolumeSource: v1.VolumeSource{
- HostPath: &v1.HostPathVolumeSource{
- Path: etcHostsPath,
- Type: hostPathType,
- },
- },
- },
- },
- },
- }
- return pod
- }
- func (config *KubeletManagedHostConfig) createPodSpecWithHostNetwork(podName string) *v1.Pod {
- hostPathType := new(v1.HostPathType)
- *hostPathType = v1.HostPathType(string(v1.HostPathFileOrCreate))
- pod := &v1.Pod{
- ObjectMeta: metav1.ObjectMeta{
- Name: podName,
- },
- Spec: v1.PodSpec{
- HostNetwork: true,
- SecurityContext: &v1.PodSecurityContext{},
- Containers: []v1.Container{
- {
- Name: "busybox-1",
- Image: etcHostsImageName,
- ImagePullPolicy: v1.PullIfNotPresent,
- Command: []string{
- "sleep",
- "900",
- },
- VolumeMounts: []v1.VolumeMount{
- {
- Name: "host-etc-hosts",
- MountPath: etcHostsOriginalPath,
- },
- },
- },
- {
- Name: "busybox-2",
- Image: etcHostsImageName,
- ImagePullPolicy: v1.PullIfNotPresent,
- Command: []string{
- "sleep",
- "900",
- },
- VolumeMounts: []v1.VolumeMount{
- {
- Name: "host-etc-hosts",
- MountPath: etcHostsOriginalPath,
- },
- },
- },
- },
- Volumes: []v1.Volume{
- {
- Name: "host-etc-hosts",
- VolumeSource: v1.VolumeSource{
- HostPath: &v1.HostPathVolumeSource{
- Path: etcHostsPath,
- Type: hostPathType,
- },
- },
- },
- },
- },
- }
- return pod
- }
|