2019-01-12 04:58:27 +00:00
|
|
|
/*
|
|
|
|
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 factory
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"regexp"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"k8s.io/apimachinery/pkg/util/sets"
|
|
|
|
"k8s.io/kubernetes/pkg/scheduler/algorithm"
|
|
|
|
"k8s.io/kubernetes/pkg/scheduler/algorithm/predicates"
|
|
|
|
"k8s.io/kubernetes/pkg/scheduler/algorithm/priorities"
|
|
|
|
schedulerapi "k8s.io/kubernetes/pkg/scheduler/api"
|
|
|
|
"k8s.io/kubernetes/pkg/scheduler/volumebinder"
|
|
|
|
|
|
|
|
"k8s.io/klog"
|
|
|
|
)
|
|
|
|
|
|
|
|
// PluginFactoryArgs are passed to all plugin factory functions.
|
|
|
|
type PluginFactoryArgs struct {
|
|
|
|
PodLister algorithm.PodLister
|
|
|
|
ServiceLister algorithm.ServiceLister
|
|
|
|
ControllerLister algorithm.ControllerLister
|
|
|
|
ReplicaSetLister algorithm.ReplicaSetLister
|
|
|
|
StatefulSetLister algorithm.StatefulSetLister
|
|
|
|
NodeLister algorithm.NodeLister
|
|
|
|
PDBLister algorithm.PDBLister
|
|
|
|
NodeInfo predicates.NodeInfo
|
2019-09-27 21:51:53 +00:00
|
|
|
CSINodeInfo predicates.CSINodeInfo
|
2019-01-12 04:58:27 +00:00
|
|
|
PVInfo predicates.PersistentVolumeInfo
|
|
|
|
PVCInfo predicates.PersistentVolumeClaimInfo
|
|
|
|
StorageClassInfo predicates.StorageClassInfo
|
|
|
|
VolumeBinder *volumebinder.VolumeBinder
|
|
|
|
HardPodAffinitySymmetricWeight int32
|
|
|
|
}
|
|
|
|
|
|
|
|
// PriorityMetadataProducerFactory produces PriorityMetadataProducer from the given args.
|
2019-04-07 17:07:55 +00:00
|
|
|
type PriorityMetadataProducerFactory func(PluginFactoryArgs) priorities.PriorityMetadataProducer
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
// PredicateMetadataProducerFactory produces PredicateMetadataProducer from the given args.
|
2019-04-07 17:07:55 +00:00
|
|
|
type PredicateMetadataProducerFactory func(PluginFactoryArgs) predicates.PredicateMetadataProducer
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
// FitPredicateFactory produces a FitPredicate from the given args.
|
2019-04-07 17:07:55 +00:00
|
|
|
type FitPredicateFactory func(PluginFactoryArgs) predicates.FitPredicate
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
// PriorityFunctionFactory produces a PriorityConfig from the given args.
|
|
|
|
// DEPRECATED
|
|
|
|
// Use Map-Reduce pattern for priority functions.
|
2019-04-07 17:07:55 +00:00
|
|
|
type PriorityFunctionFactory func(PluginFactoryArgs) priorities.PriorityFunction
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
// PriorityFunctionFactory2 produces map & reduce priority functions
|
|
|
|
// from a given args.
|
|
|
|
// FIXME: Rename to PriorityFunctionFactory.
|
2019-04-07 17:07:55 +00:00
|
|
|
type PriorityFunctionFactory2 func(PluginFactoryArgs) (priorities.PriorityMapFunction, priorities.PriorityReduceFunction)
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
// PriorityConfigFactory produces a PriorityConfig from the given function and weight
|
|
|
|
type PriorityConfigFactory struct {
|
|
|
|
Function PriorityFunctionFactory
|
|
|
|
MapReduceFunction PriorityFunctionFactory2
|
|
|
|
Weight int
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
2019-08-30 18:33:25 +00:00
|
|
|
schedulerFactoryMutex sync.RWMutex
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
// maps that hold registered algorithm types
|
|
|
|
fitPredicateMap = make(map[string]FitPredicateFactory)
|
|
|
|
mandatoryFitPredicates = sets.NewString()
|
|
|
|
priorityFunctionMap = make(map[string]PriorityConfigFactory)
|
|
|
|
algorithmProviderMap = make(map[string]AlgorithmProviderConfig)
|
|
|
|
|
|
|
|
// Registered metadata producers
|
|
|
|
priorityMetadataProducer PriorityMetadataProducerFactory
|
|
|
|
predicateMetadataProducer PredicateMetadataProducerFactory
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// DefaultProvider defines the default algorithm provider name.
|
|
|
|
DefaultProvider = "DefaultProvider"
|
|
|
|
)
|
|
|
|
|
|
|
|
// AlgorithmProviderConfig is used to store the configuration of algorithm providers.
|
|
|
|
type AlgorithmProviderConfig struct {
|
|
|
|
FitPredicateKeys sets.String
|
|
|
|
PriorityFunctionKeys sets.String
|
|
|
|
}
|
|
|
|
|
2019-09-27 21:51:53 +00:00
|
|
|
// Snapshot is used to store current state of registered predicates and priorities.
|
|
|
|
type Snapshot struct {
|
|
|
|
fitPredicateMap map[string]FitPredicateFactory
|
|
|
|
mandatoryFitPredicates sets.String
|
|
|
|
priorityFunctionMap map[string]PriorityConfigFactory
|
|
|
|
algorithmProviderMap map[string]AlgorithmProviderConfig
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy returns a snapshot of current registered predicates and priorities.
|
|
|
|
func Copy() *Snapshot {
|
|
|
|
schedulerFactoryMutex.RLock()
|
|
|
|
defer schedulerFactoryMutex.RUnlock()
|
|
|
|
|
|
|
|
copy := Snapshot{
|
|
|
|
fitPredicateMap: make(map[string]FitPredicateFactory),
|
|
|
|
mandatoryFitPredicates: sets.NewString(),
|
|
|
|
priorityFunctionMap: make(map[string]PriorityConfigFactory),
|
|
|
|
algorithmProviderMap: make(map[string]AlgorithmProviderConfig),
|
|
|
|
}
|
|
|
|
for k, v := range fitPredicateMap {
|
|
|
|
copy.fitPredicateMap[k] = v
|
|
|
|
}
|
|
|
|
for k := range mandatoryFitPredicates {
|
|
|
|
copy.mandatoryFitPredicates[k] = struct{}{}
|
|
|
|
}
|
|
|
|
for k, v := range priorityFunctionMap {
|
|
|
|
copy.priorityFunctionMap[k] = v
|
|
|
|
}
|
|
|
|
for provider, config := range algorithmProviderMap {
|
|
|
|
copyPredKeys, copyPrioKeys := sets.NewString(), sets.NewString()
|
|
|
|
for k := range config.FitPredicateKeys {
|
|
|
|
copyPredKeys[k] = struct{}{}
|
|
|
|
}
|
|
|
|
for k := range config.PriorityFunctionKeys {
|
|
|
|
copyPrioKeys[k] = struct{}{}
|
|
|
|
}
|
|
|
|
copy.algorithmProviderMap[provider] = AlgorithmProviderConfig{
|
|
|
|
FitPredicateKeys: copyPredKeys,
|
|
|
|
PriorityFunctionKeys: copyPrioKeys,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ©
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply sets state of predicates and priorities to `s`.
|
|
|
|
func Apply(s *Snapshot) {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
fitPredicateMap = s.fitPredicateMap
|
|
|
|
mandatoryFitPredicates = s.mandatoryFitPredicates
|
|
|
|
priorityFunctionMap = s.priorityFunctionMap
|
|
|
|
algorithmProviderMap = s.algorithmProviderMap
|
|
|
|
schedulerFactoryMutex.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
// RegisterFitPredicate registers a fit predicate with the algorithm
|
|
|
|
// registry. Returns the name with which the predicate was registered.
|
2019-04-07 17:07:55 +00:00
|
|
|
func RegisterFitPredicate(name string, predicate predicates.FitPredicate) string {
|
|
|
|
return RegisterFitPredicateFactory(name, func(PluginFactoryArgs) predicates.FitPredicate { return predicate })
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// RemoveFitPredicate removes a fit predicate from factory.
|
|
|
|
func RemoveFitPredicate(name string) {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
|
|
|
|
validateAlgorithmNameOrDie(name)
|
|
|
|
delete(fitPredicateMap, name)
|
|
|
|
mandatoryFitPredicates.Delete(name)
|
|
|
|
}
|
|
|
|
|
|
|
|
// RemovePredicateKeyFromAlgoProvider removes a fit predicate key from algorithmProvider.
|
|
|
|
func RemovePredicateKeyFromAlgoProvider(providerName, key string) error {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
|
|
|
|
validateAlgorithmNameOrDie(providerName)
|
|
|
|
provider, ok := algorithmProviderMap[providerName]
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("plugin %v has not been registered", providerName)
|
|
|
|
}
|
|
|
|
provider.FitPredicateKeys.Delete(key)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// RemovePredicateKeyFromAlgorithmProviderMap removes a fit predicate key from all algorithmProviders which in algorithmProviderMap.
|
|
|
|
func RemovePredicateKeyFromAlgorithmProviderMap(key string) {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
|
|
|
|
for _, provider := range algorithmProviderMap {
|
|
|
|
provider.FitPredicateKeys.Delete(key)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// InsertPredicateKeyToAlgoProvider insert a fit predicate key to algorithmProvider.
|
|
|
|
func InsertPredicateKeyToAlgoProvider(providerName, key string) error {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
|
|
|
|
validateAlgorithmNameOrDie(providerName)
|
|
|
|
provider, ok := algorithmProviderMap[providerName]
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("plugin %v has not been registered", providerName)
|
|
|
|
}
|
|
|
|
provider.FitPredicateKeys.Insert(key)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// InsertPredicateKeyToAlgorithmProviderMap insert a fit predicate key to all algorithmProviders which in algorithmProviderMap.
|
|
|
|
func InsertPredicateKeyToAlgorithmProviderMap(key string) {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
|
|
|
|
for _, provider := range algorithmProviderMap {
|
|
|
|
provider.FitPredicateKeys.Insert(key)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// InsertPriorityKeyToAlgorithmProviderMap inserts a priority function to all algorithmProviders which are in algorithmProviderMap.
|
|
|
|
func InsertPriorityKeyToAlgorithmProviderMap(key string) {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
|
|
|
|
for _, provider := range algorithmProviderMap {
|
|
|
|
provider.PriorityFunctionKeys.Insert(key)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterMandatoryFitPredicate registers a fit predicate with the algorithm registry, the predicate is used by
|
|
|
|
// kubelet, DaemonSet; it is always included in configuration. Returns the name with which the predicate was
|
|
|
|
// registered.
|
2019-04-07 17:07:55 +00:00
|
|
|
func RegisterMandatoryFitPredicate(name string, predicate predicates.FitPredicate) string {
|
2019-01-12 04:58:27 +00:00
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
validateAlgorithmNameOrDie(name)
|
2019-04-07 17:07:55 +00:00
|
|
|
fitPredicateMap[name] = func(PluginFactoryArgs) predicates.FitPredicate { return predicate }
|
2019-01-12 04:58:27 +00:00
|
|
|
mandatoryFitPredicates.Insert(name)
|
|
|
|
return name
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterFitPredicateFactory registers a fit predicate factory with the
|
|
|
|
// algorithm registry. Returns the name with which the predicate was registered.
|
|
|
|
func RegisterFitPredicateFactory(name string, predicateFactory FitPredicateFactory) string {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
validateAlgorithmNameOrDie(name)
|
|
|
|
fitPredicateMap[name] = predicateFactory
|
|
|
|
return name
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterCustomFitPredicate registers a custom fit predicate with the algorithm registry.
|
|
|
|
// Returns the name, with which the predicate was registered.
|
|
|
|
func RegisterCustomFitPredicate(policy schedulerapi.PredicatePolicy) string {
|
|
|
|
var predicateFactory FitPredicateFactory
|
|
|
|
var ok bool
|
|
|
|
|
|
|
|
validatePredicateOrDie(policy)
|
|
|
|
|
|
|
|
// generate the predicate function, if a custom type is requested
|
|
|
|
if policy.Argument != nil {
|
|
|
|
if policy.Argument.ServiceAffinity != nil {
|
2019-04-07 17:07:55 +00:00
|
|
|
predicateFactory = func(args PluginFactoryArgs) predicates.FitPredicate {
|
2019-01-12 04:58:27 +00:00
|
|
|
predicate, precomputationFunction := predicates.NewServiceAffinityPredicate(
|
|
|
|
args.PodLister,
|
|
|
|
args.ServiceLister,
|
|
|
|
args.NodeInfo,
|
|
|
|
policy.Argument.ServiceAffinity.Labels,
|
|
|
|
)
|
|
|
|
|
|
|
|
// Once we generate the predicate we should also Register the Precomputation
|
|
|
|
predicates.RegisterPredicateMetadataProducer(policy.Name, precomputationFunction)
|
|
|
|
return predicate
|
|
|
|
}
|
|
|
|
} else if policy.Argument.LabelsPresence != nil {
|
2019-04-07 17:07:55 +00:00
|
|
|
predicateFactory = func(args PluginFactoryArgs) predicates.FitPredicate {
|
2019-01-12 04:58:27 +00:00
|
|
|
return predicates.NewNodeLabelPredicate(
|
|
|
|
policy.Argument.LabelsPresence.Labels,
|
|
|
|
policy.Argument.LabelsPresence.Presence,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if predicateFactory, ok = fitPredicateMap[policy.Name]; ok {
|
|
|
|
// checking to see if a pre-defined predicate is requested
|
|
|
|
klog.V(2).Infof("Predicate type %s already registered, reusing.", policy.Name)
|
|
|
|
return policy.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
if predicateFactory == nil {
|
|
|
|
klog.Fatalf("Invalid configuration: Predicate type not found for %s", policy.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
return RegisterFitPredicateFactory(policy.Name, predicateFactory)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsFitPredicateRegistered is useful for testing providers.
|
|
|
|
func IsFitPredicateRegistered(name string) bool {
|
2019-08-30 18:33:25 +00:00
|
|
|
schedulerFactoryMutex.RLock()
|
|
|
|
defer schedulerFactoryMutex.RUnlock()
|
2019-01-12 04:58:27 +00:00
|
|
|
_, ok := fitPredicateMap[name]
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterPriorityMetadataProducerFactory registers a PriorityMetadataProducerFactory.
|
|
|
|
func RegisterPriorityMetadataProducerFactory(factory PriorityMetadataProducerFactory) {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
priorityMetadataProducer = factory
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterPredicateMetadataProducerFactory registers a PredicateMetadataProducerFactory.
|
|
|
|
func RegisterPredicateMetadataProducerFactory(factory PredicateMetadataProducerFactory) {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
predicateMetadataProducer = factory
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterPriorityFunction registers a priority function with the algorithm registry. Returns the name,
|
|
|
|
// with which the function was registered.
|
|
|
|
// DEPRECATED
|
|
|
|
// Use Map-Reduce pattern for priority functions.
|
2019-04-07 17:07:55 +00:00
|
|
|
func RegisterPriorityFunction(name string, function priorities.PriorityFunction, weight int) string {
|
2019-01-12 04:58:27 +00:00
|
|
|
return RegisterPriorityConfigFactory(name, PriorityConfigFactory{
|
2019-04-07 17:07:55 +00:00
|
|
|
Function: func(PluginFactoryArgs) priorities.PriorityFunction {
|
2019-01-12 04:58:27 +00:00
|
|
|
return function
|
|
|
|
},
|
|
|
|
Weight: weight,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-09-27 21:51:53 +00:00
|
|
|
// RegisterPriorityMapReduceFunction registers a priority function with the algorithm registry. Returns the name,
|
2019-01-12 04:58:27 +00:00
|
|
|
// with which the function was registered.
|
2019-09-27 21:51:53 +00:00
|
|
|
func RegisterPriorityMapReduceFunction(
|
2019-01-12 04:58:27 +00:00
|
|
|
name string,
|
2019-04-07 17:07:55 +00:00
|
|
|
mapFunction priorities.PriorityMapFunction,
|
|
|
|
reduceFunction priorities.PriorityReduceFunction,
|
2019-01-12 04:58:27 +00:00
|
|
|
weight int) string {
|
|
|
|
return RegisterPriorityConfigFactory(name, PriorityConfigFactory{
|
2019-04-07 17:07:55 +00:00
|
|
|
MapReduceFunction: func(PluginFactoryArgs) (priorities.PriorityMapFunction, priorities.PriorityReduceFunction) {
|
2019-01-12 04:58:27 +00:00
|
|
|
return mapFunction, reduceFunction
|
|
|
|
},
|
|
|
|
Weight: weight,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterPriorityConfigFactory registers a priority config factory with its name.
|
|
|
|
func RegisterPriorityConfigFactory(name string, pcf PriorityConfigFactory) string {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
validateAlgorithmNameOrDie(name)
|
|
|
|
priorityFunctionMap[name] = pcf
|
|
|
|
return name
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterCustomPriorityFunction registers a custom priority function with the algorithm registry.
|
|
|
|
// Returns the name, with which the priority function was registered.
|
|
|
|
func RegisterCustomPriorityFunction(policy schedulerapi.PriorityPolicy) string {
|
|
|
|
var pcf *PriorityConfigFactory
|
|
|
|
|
|
|
|
validatePriorityOrDie(policy)
|
|
|
|
|
|
|
|
// generate the priority function, if a custom priority is requested
|
|
|
|
if policy.Argument != nil {
|
|
|
|
if policy.Argument.ServiceAntiAffinity != nil {
|
|
|
|
pcf = &PriorityConfigFactory{
|
2019-04-07 17:07:55 +00:00
|
|
|
MapReduceFunction: func(args PluginFactoryArgs) (priorities.PriorityMapFunction, priorities.PriorityReduceFunction) {
|
2019-01-12 04:58:27 +00:00
|
|
|
return priorities.NewServiceAntiAffinityPriority(
|
|
|
|
args.PodLister,
|
|
|
|
args.ServiceLister,
|
|
|
|
policy.Argument.ServiceAntiAffinity.Label,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
Weight: policy.Weight,
|
|
|
|
}
|
|
|
|
} else if policy.Argument.LabelPreference != nil {
|
|
|
|
pcf = &PriorityConfigFactory{
|
2019-04-07 17:07:55 +00:00
|
|
|
MapReduceFunction: func(args PluginFactoryArgs) (priorities.PriorityMapFunction, priorities.PriorityReduceFunction) {
|
2019-01-12 04:58:27 +00:00
|
|
|
return priorities.NewNodeLabelPriority(
|
|
|
|
policy.Argument.LabelPreference.Label,
|
|
|
|
policy.Argument.LabelPreference.Presence,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
Weight: policy.Weight,
|
|
|
|
}
|
|
|
|
} else if policy.Argument.RequestedToCapacityRatioArguments != nil {
|
|
|
|
pcf = &PriorityConfigFactory{
|
2019-04-07 17:07:55 +00:00
|
|
|
MapReduceFunction: func(args PluginFactoryArgs) (priorities.PriorityMapFunction, priorities.PriorityReduceFunction) {
|
2019-09-27 21:51:53 +00:00
|
|
|
scoringFunctionShape, resources := buildScoringFunctionShapeFromRequestedToCapacityRatioArguments(policy.Argument.RequestedToCapacityRatioArguments)
|
|
|
|
p := priorities.RequestedToCapacityRatioResourceAllocationPriority(scoringFunctionShape, resources)
|
2019-01-12 04:58:27 +00:00
|
|
|
return p.PriorityMap, nil
|
|
|
|
},
|
|
|
|
Weight: policy.Weight,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if existingPcf, ok := priorityFunctionMap[policy.Name]; ok {
|
|
|
|
klog.V(2).Infof("Priority type %s already registered, reusing.", policy.Name)
|
|
|
|
// set/update the weight based on the policy
|
|
|
|
pcf = &PriorityConfigFactory{
|
|
|
|
Function: existingPcf.Function,
|
|
|
|
MapReduceFunction: existingPcf.MapReduceFunction,
|
|
|
|
Weight: policy.Weight,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if pcf == nil {
|
|
|
|
klog.Fatalf("Invalid configuration: Priority type not found for %s", policy.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
return RegisterPriorityConfigFactory(policy.Name, *pcf)
|
|
|
|
}
|
|
|
|
|
2019-09-27 21:51:53 +00:00
|
|
|
func buildScoringFunctionShapeFromRequestedToCapacityRatioArguments(arguments *schedulerapi.RequestedToCapacityRatioArguments) (priorities.FunctionShape, priorities.ResourceToWeightMap) {
|
2019-01-12 04:58:27 +00:00
|
|
|
n := len(arguments.UtilizationShape)
|
|
|
|
points := make([]priorities.FunctionShapePoint, 0, n)
|
|
|
|
for _, point := range arguments.UtilizationShape {
|
|
|
|
points = append(points, priorities.FunctionShapePoint{Utilization: int64(point.Utilization), Score: int64(point.Score)})
|
|
|
|
}
|
|
|
|
shape, err := priorities.NewFunctionShape(points)
|
|
|
|
if err != nil {
|
|
|
|
klog.Fatalf("invalid RequestedToCapacityRatioPriority arguments: %s", err.Error())
|
|
|
|
}
|
2019-09-27 21:51:53 +00:00
|
|
|
resourceToWeightMap := make(priorities.ResourceToWeightMap, 0)
|
|
|
|
if len(arguments.Resources) == 0 {
|
|
|
|
resourceToWeightMap = priorities.DefaultRequestedRatioResources
|
|
|
|
return shape, resourceToWeightMap
|
|
|
|
}
|
|
|
|
for _, resource := range arguments.Resources {
|
|
|
|
resourceToWeightMap[resource.Name] = int64(resource.Weight)
|
|
|
|
if resource.Weight == 0 {
|
|
|
|
resourceToWeightMap[resource.Name] = 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return shape, resourceToWeightMap
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// IsPriorityFunctionRegistered is useful for testing providers.
|
|
|
|
func IsPriorityFunctionRegistered(name string) bool {
|
2019-08-30 18:33:25 +00:00
|
|
|
schedulerFactoryMutex.RLock()
|
|
|
|
defer schedulerFactoryMutex.RUnlock()
|
2019-01-12 04:58:27 +00:00
|
|
|
_, ok := priorityFunctionMap[name]
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterAlgorithmProvider registers a new algorithm provider with the algorithm registry. This should
|
|
|
|
// be called from the init function in a provider plugin.
|
|
|
|
func RegisterAlgorithmProvider(name string, predicateKeys, priorityKeys sets.String) string {
|
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
validateAlgorithmNameOrDie(name)
|
|
|
|
algorithmProviderMap[name] = AlgorithmProviderConfig{
|
|
|
|
FitPredicateKeys: predicateKeys,
|
|
|
|
PriorityFunctionKeys: priorityKeys,
|
|
|
|
}
|
|
|
|
return name
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetAlgorithmProvider should not be used to modify providers. It is publicly visible for testing.
|
|
|
|
func GetAlgorithmProvider(name string) (*AlgorithmProviderConfig, error) {
|
2019-08-30 18:33:25 +00:00
|
|
|
schedulerFactoryMutex.RLock()
|
|
|
|
defer schedulerFactoryMutex.RUnlock()
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
provider, ok := algorithmProviderMap[name]
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("plugin %q has not been registered", name)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &provider, nil
|
|
|
|
}
|
|
|
|
|
2019-04-07 17:07:55 +00:00
|
|
|
func getFitPredicateFunctions(names sets.String, args PluginFactoryArgs) (map[string]predicates.FitPredicate, error) {
|
2019-08-30 18:33:25 +00:00
|
|
|
schedulerFactoryMutex.RLock()
|
|
|
|
defer schedulerFactoryMutex.RUnlock()
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-04-07 17:07:55 +00:00
|
|
|
fitPredicates := map[string]predicates.FitPredicate{}
|
2019-01-12 04:58:27 +00:00
|
|
|
for _, name := range names.List() {
|
|
|
|
factory, ok := fitPredicateMap[name]
|
|
|
|
if !ok {
|
2019-04-07 17:07:55 +00:00
|
|
|
return nil, fmt.Errorf("invalid predicate name %q specified - no corresponding function found", name)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
2019-04-07 17:07:55 +00:00
|
|
|
fitPredicates[name] = factory(args)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Always include mandatory fit predicates.
|
|
|
|
for name := range mandatoryFitPredicates {
|
|
|
|
if factory, found := fitPredicateMap[name]; found {
|
2019-04-07 17:07:55 +00:00
|
|
|
fitPredicates[name] = factory(args)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-07 17:07:55 +00:00
|
|
|
return fitPredicates, nil
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2019-04-07 17:07:55 +00:00
|
|
|
func getPriorityMetadataProducer(args PluginFactoryArgs) (priorities.PriorityMetadataProducer, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
|
|
|
|
if priorityMetadataProducer == nil {
|
2019-04-07 17:07:55 +00:00
|
|
|
return priorities.EmptyPriorityMetadataProducer, nil
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
return priorityMetadataProducer(args), nil
|
|
|
|
}
|
|
|
|
|
2019-04-07 17:07:55 +00:00
|
|
|
func getPredicateMetadataProducer(args PluginFactoryArgs) (predicates.PredicateMetadataProducer, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
schedulerFactoryMutex.Lock()
|
|
|
|
defer schedulerFactoryMutex.Unlock()
|
|
|
|
|
|
|
|
if predicateMetadataProducer == nil {
|
2019-04-07 17:07:55 +00:00
|
|
|
return predicates.EmptyPredicateMetadataProducer, nil
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
return predicateMetadataProducer(args), nil
|
|
|
|
}
|
|
|
|
|
2019-04-07 17:07:55 +00:00
|
|
|
func getPriorityFunctionConfigs(names sets.String, args PluginFactoryArgs) ([]priorities.PriorityConfig, error) {
|
2019-08-30 18:33:25 +00:00
|
|
|
schedulerFactoryMutex.RLock()
|
|
|
|
defer schedulerFactoryMutex.RUnlock()
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-04-07 17:07:55 +00:00
|
|
|
var configs []priorities.PriorityConfig
|
2019-01-12 04:58:27 +00:00
|
|
|
for _, name := range names.List() {
|
|
|
|
factory, ok := priorityFunctionMap[name]
|
|
|
|
if !ok {
|
2019-04-07 17:07:55 +00:00
|
|
|
return nil, fmt.Errorf("invalid priority name %s specified - no corresponding function found", name)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
if factory.Function != nil {
|
2019-04-07 17:07:55 +00:00
|
|
|
configs = append(configs, priorities.PriorityConfig{
|
2019-01-12 04:58:27 +00:00
|
|
|
Name: name,
|
|
|
|
Function: factory.Function(args),
|
|
|
|
Weight: factory.Weight,
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
mapFunction, reduceFunction := factory.MapReduceFunction(args)
|
2019-04-07 17:07:55 +00:00
|
|
|
configs = append(configs, priorities.PriorityConfig{
|
2019-01-12 04:58:27 +00:00
|
|
|
Name: name,
|
|
|
|
Map: mapFunction,
|
|
|
|
Reduce: reduceFunction,
|
|
|
|
Weight: factory.Weight,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err := validateSelectedConfigs(configs); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return configs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// validateSelectedConfigs validates the config weights to avoid the overflow.
|
2019-04-07 17:07:55 +00:00
|
|
|
func validateSelectedConfigs(configs []priorities.PriorityConfig) error {
|
2019-01-12 04:58:27 +00:00
|
|
|
var totalPriority int
|
|
|
|
for _, config := range configs {
|
|
|
|
// Checks totalPriority against MaxTotalPriority to avoid overflow
|
|
|
|
if config.Weight*schedulerapi.MaxPriority > schedulerapi.MaxTotalPriority-totalPriority {
|
2019-04-07 17:07:55 +00:00
|
|
|
return fmt.Errorf("total priority of priority functions has overflown")
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
totalPriority += config.Weight * schedulerapi.MaxPriority
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var validName = regexp.MustCompile("^[a-zA-Z0-9]([-a-zA-Z0-9]*[a-zA-Z0-9])$")
|
|
|
|
|
|
|
|
func validateAlgorithmNameOrDie(name string) {
|
|
|
|
if !validName.MatchString(name) {
|
|
|
|
klog.Fatalf("Algorithm name %v does not match the name validation regexp \"%v\".", name, validName)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func validatePredicateOrDie(predicate schedulerapi.PredicatePolicy) {
|
|
|
|
if predicate.Argument != nil {
|
|
|
|
numArgs := 0
|
|
|
|
if predicate.Argument.ServiceAffinity != nil {
|
|
|
|
numArgs++
|
|
|
|
}
|
|
|
|
if predicate.Argument.LabelsPresence != nil {
|
|
|
|
numArgs++
|
|
|
|
}
|
|
|
|
if numArgs != 1 {
|
|
|
|
klog.Fatalf("Exactly 1 predicate argument is required, numArgs: %v, Predicate: %s", numArgs, predicate.Name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func validatePriorityOrDie(priority schedulerapi.PriorityPolicy) {
|
|
|
|
if priority.Argument != nil {
|
|
|
|
numArgs := 0
|
|
|
|
if priority.Argument.ServiceAntiAffinity != nil {
|
|
|
|
numArgs++
|
|
|
|
}
|
|
|
|
if priority.Argument.LabelPreference != nil {
|
|
|
|
numArgs++
|
|
|
|
}
|
|
|
|
if priority.Argument.RequestedToCapacityRatioArguments != nil {
|
|
|
|
numArgs++
|
|
|
|
}
|
|
|
|
if numArgs != 1 {
|
|
|
|
klog.Fatalf("Exactly 1 priority argument is required, numArgs: %v, Priority: %s", numArgs, priority.Name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListRegisteredFitPredicates returns the registered fit predicates.
|
|
|
|
func ListRegisteredFitPredicates() []string {
|
2019-08-30 18:33:25 +00:00
|
|
|
schedulerFactoryMutex.RLock()
|
|
|
|
defer schedulerFactoryMutex.RUnlock()
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-04-07 17:07:55 +00:00
|
|
|
var names []string
|
2019-01-12 04:58:27 +00:00
|
|
|
for name := range fitPredicateMap {
|
|
|
|
names = append(names, name)
|
|
|
|
}
|
|
|
|
return names
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListRegisteredPriorityFunctions returns the registered priority functions.
|
|
|
|
func ListRegisteredPriorityFunctions() []string {
|
2019-08-30 18:33:25 +00:00
|
|
|
schedulerFactoryMutex.RLock()
|
|
|
|
defer schedulerFactoryMutex.RUnlock()
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-04-07 17:07:55 +00:00
|
|
|
var names []string
|
2019-01-12 04:58:27 +00:00
|
|
|
for name := range priorityFunctionMap {
|
|
|
|
names = append(names, name)
|
|
|
|
}
|
|
|
|
return names
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListAlgorithmProviders is called when listing all available algorithm providers in `kube-scheduler --help`
|
|
|
|
func ListAlgorithmProviders() string {
|
|
|
|
var availableAlgorithmProviders []string
|
|
|
|
for name := range algorithmProviderMap {
|
|
|
|
availableAlgorithmProviders = append(availableAlgorithmProviders, name)
|
|
|
|
}
|
|
|
|
sort.Strings(availableAlgorithmProviders)
|
|
|
|
return strings.Join(availableAlgorithmProviders, " | ")
|
|
|
|
}
|