mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-11-04 07:49:35 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			239 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			239 lines
		
	
	
		
			7.1 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 e2e_node
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"time"
 | 
						|
 | 
						|
	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | 
						|
	"k8s.io/apimachinery/pkg/util/intstr"
 | 
						|
	"k8s.io/apimachinery/pkg/util/uuid"
 | 
						|
	"k8s.io/kubernetes/pkg/api/v1"
 | 
						|
	"k8s.io/kubernetes/test/e2e/framework"
 | 
						|
 | 
						|
	. "github.com/onsi/ginkgo"
 | 
						|
	. "github.com/onsi/gomega"
 | 
						|
)
 | 
						|
 | 
						|
var _ = framework.KubeDescribe("Container Lifecycle Hook", func() {
 | 
						|
	f := framework.NewDefaultFramework("container-lifecycle-hook")
 | 
						|
	var podClient *framework.PodClient
 | 
						|
	const (
 | 
						|
		podCheckInterval     = 1 * time.Second
 | 
						|
		podWaitTimeout       = 2 * time.Minute
 | 
						|
		postStartWaitTimeout = 2 * time.Minute
 | 
						|
		preStopWaitTimeout   = 30 * time.Second
 | 
						|
	)
 | 
						|
	Context("when create a pod with lifecycle hook", func() {
 | 
						|
		BeforeEach(func() {
 | 
						|
			podClient = f.PodClient()
 | 
						|
		})
 | 
						|
 | 
						|
		Context("when it is exec hook", func() {
 | 
						|
			var file string
 | 
						|
			testPodWithExecHook := func(podWithHook *v1.Pod) {
 | 
						|
				podCheckHook := getExecHookTestPod("pod-check-hook",
 | 
						|
					// Wait until the file is created.
 | 
						|
					[]string{"sh", "-c", fmt.Sprintf("while [ ! -e %s ]; do sleep 1; done", file)},
 | 
						|
				)
 | 
						|
				By("create the pod with lifecycle hook")
 | 
						|
				podClient.CreateSync(podWithHook)
 | 
						|
				if podWithHook.Spec.Containers[0].Lifecycle.PostStart != nil {
 | 
						|
					By("create the hook check pod")
 | 
						|
					podClient.Create(podCheckHook)
 | 
						|
					By("wait for the hook check pod to success")
 | 
						|
					podClient.WaitForSuccess(podCheckHook.Name, postStartWaitTimeout)
 | 
						|
				}
 | 
						|
				By("delete the pod with lifecycle hook")
 | 
						|
				podClient.DeleteSync(podWithHook.Name, metav1.NewDeleteOptions(15), podWaitTimeout)
 | 
						|
				if podWithHook.Spec.Containers[0].Lifecycle.PreStop != nil {
 | 
						|
					By("create the hook check pod")
 | 
						|
					podClient.Create(podCheckHook)
 | 
						|
					By("wait for the prestop check pod to success")
 | 
						|
					podClient.WaitForSuccess(podCheckHook.Name, preStopWaitTimeout)
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			BeforeEach(func() {
 | 
						|
				file = "/tmp/test-" + string(uuid.NewUUID())
 | 
						|
			})
 | 
						|
 | 
						|
			AfterEach(func() {
 | 
						|
				By("cleanup the temporary file created in the test.")
 | 
						|
				cleanupPod := getExecHookTestPod("pod-clean-up", []string{"rm", file})
 | 
						|
				podClient.Create(cleanupPod)
 | 
						|
				podClient.WaitForSuccess(cleanupPod.Name, podWaitTimeout)
 | 
						|
			})
 | 
						|
 | 
						|
			It("should execute poststart exec hook properly [Conformance]", func() {
 | 
						|
				podWithHook := getExecHookTestPod("pod-with-poststart-exec-hook",
 | 
						|
					// Block forever
 | 
						|
					[]string{"tail", "-f", "/dev/null"},
 | 
						|
				)
 | 
						|
				podWithHook.Spec.Containers[0].Lifecycle = &v1.Lifecycle{
 | 
						|
					PostStart: &v1.Handler{
 | 
						|
						Exec: &v1.ExecAction{Command: []string{"touch", file}},
 | 
						|
					},
 | 
						|
				}
 | 
						|
				testPodWithExecHook(podWithHook)
 | 
						|
			})
 | 
						|
 | 
						|
			It("should execute prestop exec hook properly [Conformance]", func() {
 | 
						|
				podWithHook := getExecHookTestPod("pod-with-prestop-exec-hook",
 | 
						|
					// Block forever
 | 
						|
					[]string{"tail", "-f", "/dev/null"},
 | 
						|
				)
 | 
						|
				podWithHook.Spec.Containers[0].Lifecycle = &v1.Lifecycle{
 | 
						|
					PreStop: &v1.Handler{
 | 
						|
						Exec: &v1.ExecAction{Command: []string{"touch", file}},
 | 
						|
					},
 | 
						|
				}
 | 
						|
				testPodWithExecHook(podWithHook)
 | 
						|
			})
 | 
						|
		})
 | 
						|
 | 
						|
		Context("when it is http hook", func() {
 | 
						|
			var targetIP string
 | 
						|
			podHandleHookRequest := &v1.Pod{
 | 
						|
				ObjectMeta: metav1.ObjectMeta{
 | 
						|
					Name: "pod-handle-http-request",
 | 
						|
				},
 | 
						|
				Spec: v1.PodSpec{
 | 
						|
					Containers: []v1.Container{
 | 
						|
						{
 | 
						|
							Name:  "pod-handle-http-request",
 | 
						|
							Image: "gcr.io/google_containers/netexec:1.7",
 | 
						|
							Ports: []v1.ContainerPort{
 | 
						|
								{
 | 
						|
									ContainerPort: 8080,
 | 
						|
									Protocol:      v1.ProtocolTCP,
 | 
						|
								},
 | 
						|
							},
 | 
						|
						},
 | 
						|
					},
 | 
						|
				},
 | 
						|
			}
 | 
						|
			BeforeEach(func() {
 | 
						|
				By("create the container to handle the HTTPGet hook request.")
 | 
						|
				newPod := podClient.CreateSync(podHandleHookRequest)
 | 
						|
				targetIP = newPod.Status.PodIP
 | 
						|
			})
 | 
						|
			testPodWithHttpHook := func(podWithHook *v1.Pod) {
 | 
						|
				By("create the pod with lifecycle hook")
 | 
						|
				podClient.CreateSync(podWithHook)
 | 
						|
				if podWithHook.Spec.Containers[0].Lifecycle.PostStart != nil {
 | 
						|
					By("check poststart hook")
 | 
						|
					Eventually(func() error {
 | 
						|
						return podClient.MatchContainerOutput(podHandleHookRequest.Name, podHandleHookRequest.Spec.Containers[0].Name,
 | 
						|
							`GET /echo\?msg=poststart`)
 | 
						|
					}, postStartWaitTimeout, podCheckInterval).Should(BeNil())
 | 
						|
				}
 | 
						|
				By("delete the pod with lifecycle hook")
 | 
						|
				podClient.DeleteSync(podWithHook.Name, metav1.NewDeleteOptions(15), podWaitTimeout)
 | 
						|
				if podWithHook.Spec.Containers[0].Lifecycle.PreStop != nil {
 | 
						|
					By("check prestop hook")
 | 
						|
					Eventually(func() error {
 | 
						|
						return podClient.MatchContainerOutput(podHandleHookRequest.Name, podHandleHookRequest.Spec.Containers[0].Name,
 | 
						|
							`GET /echo\?msg=prestop`)
 | 
						|
					}, preStopWaitTimeout, podCheckInterval).Should(BeNil())
 | 
						|
				}
 | 
						|
			}
 | 
						|
			It("should execute poststart http hook properly [Conformance]", func() {
 | 
						|
				podWithHook := &v1.Pod{
 | 
						|
					ObjectMeta: metav1.ObjectMeta{
 | 
						|
						Name: "pod-with-poststart-http-hook",
 | 
						|
					},
 | 
						|
					Spec: v1.PodSpec{
 | 
						|
						Containers: []v1.Container{
 | 
						|
							{
 | 
						|
								Name:  "pod-with-poststart-http-hook",
 | 
						|
								Image: framework.GetPauseImageNameForHostArch(),
 | 
						|
								Lifecycle: &v1.Lifecycle{
 | 
						|
									PostStart: &v1.Handler{
 | 
						|
										HTTPGet: &v1.HTTPGetAction{
 | 
						|
											Path: "/echo?msg=poststart",
 | 
						|
											Host: targetIP,
 | 
						|
											Port: intstr.FromInt(8080),
 | 
						|
										},
 | 
						|
									},
 | 
						|
								},
 | 
						|
							},
 | 
						|
						},
 | 
						|
					},
 | 
						|
				}
 | 
						|
				testPodWithHttpHook(podWithHook)
 | 
						|
			})
 | 
						|
			It("should execute prestop http hook properly [Conformance]", func() {
 | 
						|
				podWithHook := &v1.Pod{
 | 
						|
					ObjectMeta: metav1.ObjectMeta{
 | 
						|
						Name: "pod-with-prestop-http-hook",
 | 
						|
					},
 | 
						|
					Spec: v1.PodSpec{
 | 
						|
						Containers: []v1.Container{
 | 
						|
							{
 | 
						|
								Name:  "pod-with-prestop-http-hook",
 | 
						|
								Image: framework.GetPauseImageNameForHostArch(),
 | 
						|
								Lifecycle: &v1.Lifecycle{
 | 
						|
									PreStop: &v1.Handler{
 | 
						|
										HTTPGet: &v1.HTTPGetAction{
 | 
						|
											Path: "/echo?msg=prestop",
 | 
						|
											Host: targetIP,
 | 
						|
											Port: intstr.FromInt(8080),
 | 
						|
										},
 | 
						|
									},
 | 
						|
								},
 | 
						|
							},
 | 
						|
						},
 | 
						|
					},
 | 
						|
				}
 | 
						|
				testPodWithHttpHook(podWithHook)
 | 
						|
			})
 | 
						|
		})
 | 
						|
	})
 | 
						|
})
 | 
						|
 | 
						|
func getExecHookTestPod(name string, cmd []string) *v1.Pod {
 | 
						|
	return &v1.Pod{
 | 
						|
		ObjectMeta: metav1.ObjectMeta{
 | 
						|
			Name: name,
 | 
						|
		},
 | 
						|
		Spec: v1.PodSpec{
 | 
						|
			Containers: []v1.Container{
 | 
						|
				{
 | 
						|
					Name:  name,
 | 
						|
					Image: "gcr.io/google_containers/busybox:1.24",
 | 
						|
					VolumeMounts: []v1.VolumeMount{
 | 
						|
						{
 | 
						|
							Name:      "tmpfs",
 | 
						|
							MountPath: "/tmp",
 | 
						|
						},
 | 
						|
					},
 | 
						|
					Command: cmd,
 | 
						|
				},
 | 
						|
			},
 | 
						|
			RestartPolicy: v1.RestartPolicyNever,
 | 
						|
			Volumes: []v1.Volume{
 | 
						|
				{
 | 
						|
					Name:         "tmpfs",
 | 
						|
					VolumeSource: v1.VolumeSource{HostPath: &v1.HostPathVolumeSource{Path: "/tmp"}},
 | 
						|
				},
 | 
						|
			},
 | 
						|
		},
 | 
						|
	}
 | 
						|
}
 |