mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-25 01:20:18 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			123 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			123 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
| Copyright 2014 Google Inc. All rights reserved.
 | |
| 
 | |
| 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 registry
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"math/rand"
 | |
| 
 | |
| 	. "github.com/GoogleCloudPlatform/kubernetes/pkg/api"
 | |
| )
 | |
| 
 | |
| // Scheduler is an interface implemented by things that know how to schedule pods onto machines.
 | |
| type Scheduler interface {
 | |
| 	Schedule(Pod) (string, error)
 | |
| }
 | |
| 
 | |
| // RandomScheduler choses machines uniformly at random.
 | |
| type RandomScheduler struct {
 | |
| 	machines []string
 | |
| 	random   rand.Rand
 | |
| }
 | |
| 
 | |
| func MakeRandomScheduler(machines []string, random rand.Rand) Scheduler {
 | |
| 	return &RandomScheduler{
 | |
| 		machines: machines,
 | |
| 		random:   random,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (s *RandomScheduler) Schedule(pod Pod) (string, error) {
 | |
| 	return s.machines[s.random.Int()%len(s.machines)], nil
 | |
| }
 | |
| 
 | |
| // RoundRobinScheduler chooses machines in order.
 | |
| type RoundRobinScheduler struct {
 | |
| 	machines     []string
 | |
| 	currentIndex int
 | |
| }
 | |
| 
 | |
| func MakeRoundRobinScheduler(machines []string) Scheduler {
 | |
| 	return &RoundRobinScheduler{
 | |
| 		machines:     machines,
 | |
| 		currentIndex: 0,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (s *RoundRobinScheduler) Schedule(pod Pod) (string, error) {
 | |
| 	result := s.machines[s.currentIndex]
 | |
| 	s.currentIndex = (s.currentIndex + 1) % len(s.machines)
 | |
| 	return result, nil
 | |
| }
 | |
| 
 | |
| type FirstFitScheduler struct {
 | |
| 	machines []string
 | |
| 	registry PodRegistry
 | |
| 	random   *rand.Rand
 | |
| }
 | |
| 
 | |
| func MakeFirstFitScheduler(machines []string, registry PodRegistry, random *rand.Rand) Scheduler {
 | |
| 	return &FirstFitScheduler{
 | |
| 		machines: machines,
 | |
| 		registry: registry,
 | |
| 		random:   random,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (s *FirstFitScheduler) containsPort(pod Pod, port Port) bool {
 | |
| 	for _, container := range pod.DesiredState.Manifest.Containers {
 | |
| 		for _, podPort := range container.Ports {
 | |
| 			if podPort.HostPort == port.HostPort {
 | |
| 				return true
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| func (s *FirstFitScheduler) Schedule(pod Pod) (string, error) {
 | |
| 	machineToPods := map[string][]Pod{}
 | |
| 	pods, err := s.registry.ListPods(nil)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	for _, scheduledPod := range pods {
 | |
| 		host := scheduledPod.CurrentState.Host
 | |
| 		machineToPods[host] = append(machineToPods[host], scheduledPod)
 | |
| 	}
 | |
| 	var machineOptions []string
 | |
| 	for _, machine := range s.machines {
 | |
| 		podFits := true
 | |
| 		for _, scheduledPod := range machineToPods[machine] {
 | |
| 			for _, container := range pod.DesiredState.Manifest.Containers {
 | |
| 				for _, port := range container.Ports {
 | |
| 					if s.containsPort(scheduledPod, port) {
 | |
| 						podFits = false
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		if podFits {
 | |
| 			machineOptions = append(machineOptions, machine)
 | |
| 		}
 | |
| 	}
 | |
| 	if len(machineOptions) == 0 {
 | |
| 		return "", fmt.Errorf("Failed to find fit for %#v", pod)
 | |
| 	} else {
 | |
| 		return machineOptions[s.random.Int()%len(machineOptions)], nil
 | |
| 	}
 | |
| }
 |