mirror of
				https://github.com/optim-enterprises-bv/kubernetes.git
				synced 2025-11-03 19:58:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			178 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			178 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
Copyright 2014 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 kubelet
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"os"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"k8s.io/api/core/v1"
 | 
						|
	"k8s.io/klog"
 | 
						|
	kubecontainer "k8s.io/kubernetes/pkg/kubelet/container"
 | 
						|
	kubetypes "k8s.io/kubernetes/pkg/kubelet/types"
 | 
						|
	"k8s.io/kubernetes/pkg/kubelet/util/format"
 | 
						|
)
 | 
						|
 | 
						|
const (
 | 
						|
	runOnceManifestDelay     = 1 * time.Second
 | 
						|
	runOnceMaxRetries        = 10
 | 
						|
	runOnceRetryDelay        = 1 * time.Second
 | 
						|
	runOnceRetryDelayBackoff = 2
 | 
						|
)
 | 
						|
 | 
						|
// RunPodResult defines the running results of a Pod.
 | 
						|
type RunPodResult struct {
 | 
						|
	Pod *v1.Pod
 | 
						|
	Err error
 | 
						|
}
 | 
						|
 | 
						|
// RunOnce polls from one configuration update and run the associated pods.
 | 
						|
func (kl *Kubelet) RunOnce(updates <-chan kubetypes.PodUpdate) ([]RunPodResult, error) {
 | 
						|
	// Setup filesystem directories.
 | 
						|
	if err := kl.setupDataDirs(); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	// If the container logs directory does not exist, create it.
 | 
						|
	if _, err := os.Stat(ContainerLogsDir); err != nil {
 | 
						|
		if err := kl.os.MkdirAll(ContainerLogsDir, 0755); err != nil {
 | 
						|
			klog.Errorf("Failed to create directory %q: %v", ContainerLogsDir, err)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	select {
 | 
						|
	case u := <-updates:
 | 
						|
		klog.Infof("processing manifest with %d pods", len(u.Pods))
 | 
						|
		result, err := kl.runOnce(u.Pods, runOnceRetryDelay)
 | 
						|
		klog.Infof("finished processing %d pods", len(u.Pods))
 | 
						|
		return result, err
 | 
						|
	case <-time.After(runOnceManifestDelay):
 | 
						|
		return nil, fmt.Errorf("no pod manifest update after %v", runOnceManifestDelay)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// runOnce runs a given set of pods and returns their status.
 | 
						|
func (kl *Kubelet) runOnce(pods []*v1.Pod, retryDelay time.Duration) (results []RunPodResult, err error) {
 | 
						|
	ch := make(chan RunPodResult)
 | 
						|
	admitted := []*v1.Pod{}
 | 
						|
	for _, pod := range pods {
 | 
						|
		// Check if we can admit the pod.
 | 
						|
		if ok, reason, message := kl.canAdmitPod(admitted, pod); !ok {
 | 
						|
			kl.rejectPod(pod, reason, message)
 | 
						|
			results = append(results, RunPodResult{pod, nil})
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		admitted = append(admitted, pod)
 | 
						|
		go func(pod *v1.Pod) {
 | 
						|
			err := kl.runPod(pod, retryDelay)
 | 
						|
			ch <- RunPodResult{pod, err}
 | 
						|
		}(pod)
 | 
						|
	}
 | 
						|
 | 
						|
	klog.Infof("Waiting for %d pods", len(admitted))
 | 
						|
	failedPods := []string{}
 | 
						|
	for i := 0; i < len(admitted); i++ {
 | 
						|
		res := <-ch
 | 
						|
		results = append(results, res)
 | 
						|
		if res.Err != nil {
 | 
						|
			faliedContainerName, err := kl.getFailedContainers(res.Pod)
 | 
						|
			if err != nil {
 | 
						|
				klog.Infof("unable to get failed containers' names for pod %q, error:%v", format.Pod(res.Pod), err)
 | 
						|
			} else {
 | 
						|
				klog.Infof("unable to start pod %q because container:%v failed", format.Pod(res.Pod), faliedContainerName)
 | 
						|
			}
 | 
						|
			failedPods = append(failedPods, format.Pod(res.Pod))
 | 
						|
		} else {
 | 
						|
			klog.Infof("started pod %q", format.Pod(res.Pod))
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if len(failedPods) > 0 {
 | 
						|
		return results, fmt.Errorf("error running pods: %v", failedPods)
 | 
						|
	}
 | 
						|
	klog.Infof("%d pods started", len(pods))
 | 
						|
	return results, err
 | 
						|
}
 | 
						|
 | 
						|
// runPod runs a single pod and wait until all containers are running.
 | 
						|
func (kl *Kubelet) runPod(pod *v1.Pod, retryDelay time.Duration) error {
 | 
						|
	delay := retryDelay
 | 
						|
	retry := 0
 | 
						|
	for {
 | 
						|
		status, err := kl.containerRuntime.GetPodStatus(pod.UID, pod.Name, pod.Namespace)
 | 
						|
		if err != nil {
 | 
						|
			return fmt.Errorf("unable to get status for pod %q: %v", format.Pod(pod), err)
 | 
						|
		}
 | 
						|
 | 
						|
		if kl.isPodRunning(pod, status) {
 | 
						|
			klog.Infof("pod %q containers running", format.Pod(pod))
 | 
						|
			return nil
 | 
						|
		}
 | 
						|
		klog.Infof("pod %q containers not running: syncing", format.Pod(pod))
 | 
						|
 | 
						|
		klog.Infof("Creating a mirror pod for static pod %q", format.Pod(pod))
 | 
						|
		if err := kl.podManager.CreateMirrorPod(pod); err != nil {
 | 
						|
			klog.Errorf("Failed creating a mirror pod %q: %v", format.Pod(pod), err)
 | 
						|
		}
 | 
						|
		mirrorPod, _ := kl.podManager.GetMirrorPodByPod(pod)
 | 
						|
		if err = kl.syncPod(syncPodOptions{
 | 
						|
			pod:        pod,
 | 
						|
			mirrorPod:  mirrorPod,
 | 
						|
			podStatus:  status,
 | 
						|
			updateType: kubetypes.SyncPodUpdate,
 | 
						|
		}); err != nil {
 | 
						|
			return fmt.Errorf("error syncing pod %q: %v", format.Pod(pod), err)
 | 
						|
		}
 | 
						|
		if retry >= runOnceMaxRetries {
 | 
						|
			return fmt.Errorf("timeout error: pod %q containers not running after %d retries", format.Pod(pod), runOnceMaxRetries)
 | 
						|
		}
 | 
						|
		// TODO(proppy): health checking would be better than waiting + checking the state at the next iteration.
 | 
						|
		klog.Infof("pod %q containers synced, waiting for %v", format.Pod(pod), delay)
 | 
						|
		time.Sleep(delay)
 | 
						|
		retry++
 | 
						|
		delay *= runOnceRetryDelayBackoff
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// isPodRunning returns true if all containers of a manifest are running.
 | 
						|
func (kl *Kubelet) isPodRunning(pod *v1.Pod, status *kubecontainer.PodStatus) bool {
 | 
						|
	for _, c := range pod.Spec.Containers {
 | 
						|
		cs := status.FindContainerStatusByName(c.Name)
 | 
						|
		if cs == nil || cs.State != kubecontainer.ContainerStateRunning {
 | 
						|
			klog.Infof("Container %q for pod %q not running", c.Name, format.Pod(pod))
 | 
						|
			return false
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return true
 | 
						|
}
 | 
						|
 | 
						|
// getFailedContainer returns failed container name for pod.
 | 
						|
func (kl *Kubelet) getFailedContainers(pod *v1.Pod) ([]string, error) {
 | 
						|
	status, err := kl.containerRuntime.GetPodStatus(pod.UID, pod.Name, pod.Namespace)
 | 
						|
	if err != nil {
 | 
						|
		return nil, fmt.Errorf("unable to get status for pod %q: %v", format.Pod(pod), err)
 | 
						|
	}
 | 
						|
	var containerNames []string
 | 
						|
	for _, cs := range status.ContainerStatuses {
 | 
						|
		if cs.State != kubecontainer.ContainerStateRunning && cs.ExitCode != 0 {
 | 
						|
			containerNames = append(containerNames, cs.Name)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return containerNames, nil
 | 
						|
}
 |