mirror of
				https://github.com/optim-enterprises-bv/kubernetes.git
				synced 2025-11-04 04:08:16 +00:00 
			
		
		
		
	All code must use the context from Ginkgo when doing API calls or polling for a change, otherwise the code would not return immediately when the test gets aborted.
		
			
				
	
	
		
			152 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			152 lines
		
	
	
		
			4.8 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 node
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"fmt"
 | 
						|
	"time"
 | 
						|
 | 
						|
	v1 "k8s.io/api/core/v1"
 | 
						|
	apierrors "k8s.io/apimachinery/pkg/api/errors"
 | 
						|
	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | 
						|
	"k8s.io/apimachinery/pkg/util/uuid"
 | 
						|
	podutil "k8s.io/kubernetes/pkg/api/v1/pod"
 | 
						|
	e2epod "k8s.io/kubernetes/test/e2e/framework/pod"
 | 
						|
)
 | 
						|
 | 
						|
const (
 | 
						|
	// ContainerStatusRetryTimeout represents polling threshold before giving up to get the container status
 | 
						|
	ContainerStatusRetryTimeout = time.Minute * 5
 | 
						|
	// ContainerStatusPollInterval represents duration between polls to get the container status
 | 
						|
	ContainerStatusPollInterval = time.Second * 1
 | 
						|
)
 | 
						|
 | 
						|
// ConformanceContainer defines the types for running container conformance test cases
 | 
						|
// One pod one container
 | 
						|
type ConformanceContainer struct {
 | 
						|
	Container        v1.Container
 | 
						|
	RestartPolicy    v1.RestartPolicy
 | 
						|
	Volumes          []v1.Volume
 | 
						|
	ImagePullSecrets []string
 | 
						|
 | 
						|
	PodClient          *e2epod.PodClient
 | 
						|
	podName            string
 | 
						|
	PodSecurityContext *v1.PodSecurityContext
 | 
						|
}
 | 
						|
 | 
						|
// Create creates the defined conformance container
 | 
						|
func (cc *ConformanceContainer) Create(ctx context.Context) {
 | 
						|
	cc.podName = cc.Container.Name + string(uuid.NewUUID())
 | 
						|
	imagePullSecrets := []v1.LocalObjectReference{}
 | 
						|
	for _, s := range cc.ImagePullSecrets {
 | 
						|
		imagePullSecrets = append(imagePullSecrets, v1.LocalObjectReference{Name: s})
 | 
						|
	}
 | 
						|
	pod := &v1.Pod{
 | 
						|
		ObjectMeta: metav1.ObjectMeta{
 | 
						|
			Name: cc.podName,
 | 
						|
		},
 | 
						|
		Spec: v1.PodSpec{
 | 
						|
			RestartPolicy: cc.RestartPolicy,
 | 
						|
			Containers: []v1.Container{
 | 
						|
				cc.Container,
 | 
						|
			},
 | 
						|
			SecurityContext:  cc.PodSecurityContext,
 | 
						|
			Volumes:          cc.Volumes,
 | 
						|
			ImagePullSecrets: imagePullSecrets,
 | 
						|
		},
 | 
						|
	}
 | 
						|
	cc.PodClient.Create(ctx, pod)
 | 
						|
}
 | 
						|
 | 
						|
// Delete deletes the defined conformance container
 | 
						|
func (cc *ConformanceContainer) Delete(ctx context.Context) error {
 | 
						|
	return cc.PodClient.Delete(ctx, cc.podName, *metav1.NewDeleteOptions(0))
 | 
						|
}
 | 
						|
 | 
						|
// IsReady returns whether this container is ready and error if any
 | 
						|
func (cc *ConformanceContainer) IsReady(ctx context.Context) (bool, error) {
 | 
						|
	pod, err := cc.PodClient.Get(ctx, cc.podName, metav1.GetOptions{})
 | 
						|
	if err != nil {
 | 
						|
		return false, err
 | 
						|
	}
 | 
						|
	return podutil.IsPodReady(pod), nil
 | 
						|
}
 | 
						|
 | 
						|
// GetPhase returns the phase of the pod lifecycle and error if any
 | 
						|
func (cc *ConformanceContainer) GetPhase(ctx context.Context) (v1.PodPhase, error) {
 | 
						|
	pod, err := cc.PodClient.Get(ctx, cc.podName, metav1.GetOptions{})
 | 
						|
	if err != nil {
 | 
						|
		// it doesn't matter what phase to return as error would not be nil
 | 
						|
		return v1.PodSucceeded, err
 | 
						|
	}
 | 
						|
	return pod.Status.Phase, nil
 | 
						|
}
 | 
						|
 | 
						|
// GetStatus returns the details of the current status of this container and error if any
 | 
						|
func (cc *ConformanceContainer) GetStatus(ctx context.Context) (v1.ContainerStatus, error) {
 | 
						|
	pod, err := cc.PodClient.Get(ctx, cc.podName, metav1.GetOptions{})
 | 
						|
	if err != nil {
 | 
						|
		return v1.ContainerStatus{}, err
 | 
						|
	}
 | 
						|
	statuses := pod.Status.ContainerStatuses
 | 
						|
	if len(statuses) != 1 || statuses[0].Name != cc.Container.Name {
 | 
						|
		return v1.ContainerStatus{}, fmt.Errorf("unexpected container statuses %v", statuses)
 | 
						|
	}
 | 
						|
	return statuses[0], nil
 | 
						|
}
 | 
						|
 | 
						|
// Present returns whether this pod is present and error if any
 | 
						|
func (cc *ConformanceContainer) Present(ctx context.Context) (bool, error) {
 | 
						|
	_, err := cc.PodClient.Get(ctx, cc.podName, metav1.GetOptions{})
 | 
						|
	if err == nil {
 | 
						|
		return true, nil
 | 
						|
	}
 | 
						|
	if apierrors.IsNotFound(err) {
 | 
						|
		return false, nil
 | 
						|
	}
 | 
						|
	return false, err
 | 
						|
}
 | 
						|
 | 
						|
// ContainerState represents different states of its lifecycle
 | 
						|
type ContainerState string
 | 
						|
 | 
						|
const (
 | 
						|
	// ContainerStateWaiting represents 'Waiting' container state
 | 
						|
	ContainerStateWaiting ContainerState = "Waiting"
 | 
						|
	// ContainerStateRunning represents 'Running' container state
 | 
						|
	ContainerStateRunning ContainerState = "Running"
 | 
						|
	// ContainerStateTerminated represents 'Terminated' container state
 | 
						|
	ContainerStateTerminated ContainerState = "Terminated"
 | 
						|
	// ContainerStateUnknown represents 'Unknown' container state
 | 
						|
	ContainerStateUnknown ContainerState = "Unknown"
 | 
						|
)
 | 
						|
 | 
						|
// GetContainerState returns current state the container represents among its lifecycle
 | 
						|
func GetContainerState(state v1.ContainerState) ContainerState {
 | 
						|
	if state.Waiting != nil {
 | 
						|
		return ContainerStateWaiting
 | 
						|
	}
 | 
						|
	if state.Running != nil {
 | 
						|
		return ContainerStateRunning
 | 
						|
	}
 | 
						|
	if state.Terminated != nil {
 | 
						|
		return ContainerStateTerminated
 | 
						|
	}
 | 
						|
	return ContainerStateUnknown
 | 
						|
}
 |