2020-03-26 21:07:15 +00:00
|
|
|
/*
|
|
|
|
Copyright 2020 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 profile holds the definition of a scheduling Profile.
|
|
|
|
package profile
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
|
|
"k8s.io/client-go/kubernetes/scheme"
|
|
|
|
"k8s.io/client-go/tools/events"
|
|
|
|
"k8s.io/kubernetes/pkg/scheduler/apis/config"
|
2020-12-01 01:06:26 +00:00
|
|
|
"k8s.io/kubernetes/pkg/scheduler/framework"
|
2020-08-10 17:43:49 +00:00
|
|
|
frameworkruntime "k8s.io/kubernetes/pkg/scheduler/framework/runtime"
|
2020-03-26 21:07:15 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// RecorderFactory builds an EventRecorder for a given scheduler name.
|
|
|
|
type RecorderFactory func(string) events.EventRecorder
|
|
|
|
|
2020-12-01 01:06:26 +00:00
|
|
|
// newProfile builds a Profile for the given configuration.
|
|
|
|
func newProfile(cfg config.KubeSchedulerProfile, r frameworkruntime.Registry, recorderFact RecorderFactory,
|
|
|
|
opts ...frameworkruntime.Option) (framework.Framework, error) {
|
2020-08-10 17:43:49 +00:00
|
|
|
recorder := recorderFact(cfg.SchedulerName)
|
2021-03-18 22:40:29 +00:00
|
|
|
opts = append(opts, frameworkruntime.WithEventRecorder(recorder))
|
|
|
|
fwk, err := frameworkruntime.NewFramework(r, &cfg, opts...)
|
2020-03-26 21:07:15 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-12-01 01:06:26 +00:00
|
|
|
return fwk, nil
|
2020-03-26 21:07:15 +00:00
|
|
|
}
|
|
|
|
|
2020-12-01 01:06:26 +00:00
|
|
|
// Map holds frameworks indexed by scheduler name.
|
|
|
|
type Map map[string]framework.Framework
|
2020-03-26 21:07:15 +00:00
|
|
|
|
2020-12-01 01:06:26 +00:00
|
|
|
// NewMap builds the frameworks given by the configuration, indexed by name.
|
|
|
|
func NewMap(cfgs []config.KubeSchedulerProfile, r frameworkruntime.Registry, recorderFact RecorderFactory,
|
2020-08-10 17:43:49 +00:00
|
|
|
opts ...frameworkruntime.Option) (Map, error) {
|
2020-03-26 21:07:15 +00:00
|
|
|
m := make(Map)
|
|
|
|
v := cfgValidator{m: m}
|
|
|
|
|
|
|
|
for _, cfg := range cfgs {
|
|
|
|
if err := v.validate(cfg); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-12-01 01:06:26 +00:00
|
|
|
p, err := newProfile(cfg, r, recorderFact, opts...)
|
2020-03-26 21:07:15 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("creating profile for scheduler name %s: %v", cfg.SchedulerName, err)
|
|
|
|
}
|
|
|
|
m[cfg.SchedulerName] = p
|
|
|
|
}
|
|
|
|
return m, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// HandlesSchedulerName returns whether a profile handles the given scheduler name.
|
|
|
|
func (m Map) HandlesSchedulerName(name string) bool {
|
|
|
|
_, ok := m[name]
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewRecorderFactory returns a RecorderFactory for the broadcaster.
|
|
|
|
func NewRecorderFactory(b events.EventBroadcaster) RecorderFactory {
|
|
|
|
return func(name string) events.EventRecorder {
|
|
|
|
return b.NewRecorder(scheme.Scheme, name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type cfgValidator struct {
|
|
|
|
m Map
|
|
|
|
queueSort string
|
2020-08-10 17:43:49 +00:00
|
|
|
queueSortArgs runtime.Object
|
2020-03-26 21:07:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (v *cfgValidator) validate(cfg config.KubeSchedulerProfile) error {
|
|
|
|
if len(cfg.SchedulerName) == 0 {
|
|
|
|
return errors.New("scheduler name is needed")
|
|
|
|
}
|
|
|
|
if cfg.Plugins == nil {
|
|
|
|
return fmt.Errorf("plugins required for profile with scheduler name %q", cfg.SchedulerName)
|
|
|
|
}
|
|
|
|
if v.m[cfg.SchedulerName] != nil {
|
|
|
|
return fmt.Errorf("duplicate profile with scheduler name %q", cfg.SchedulerName)
|
|
|
|
}
|
2021-03-18 22:40:29 +00:00
|
|
|
if len(cfg.Plugins.QueueSort.Enabled) != 1 {
|
2020-03-26 21:07:15 +00:00
|
|
|
return fmt.Errorf("one queue sort plugin required for profile with scheduler name %q", cfg.SchedulerName)
|
|
|
|
}
|
|
|
|
queueSort := cfg.Plugins.QueueSort.Enabled[0].Name
|
2020-08-10 17:43:49 +00:00
|
|
|
var queueSortArgs runtime.Object
|
2020-03-26 21:07:15 +00:00
|
|
|
for _, plCfg := range cfg.PluginConfig {
|
|
|
|
if plCfg.Name == queueSort {
|
|
|
|
queueSortArgs = plCfg.Args
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(v.queueSort) == 0 {
|
|
|
|
v.queueSort = queueSort
|
|
|
|
v.queueSortArgs = queueSortArgs
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if v.queueSort != queueSort {
|
|
|
|
return fmt.Errorf("different queue sort plugins for profile %q: %q, first: %q", cfg.SchedulerName, queueSort, v.queueSort)
|
|
|
|
}
|
|
|
|
if !cmp.Equal(v.queueSortArgs, queueSortArgs) {
|
2020-08-10 17:43:49 +00:00
|
|
|
return fmt.Errorf("different queue sort plugin args for profile %q", cfg.SchedulerName)
|
2020-03-26 21:07:15 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|