mirror of
https://github.com/ceph/ceph-csi.git
synced 2024-12-01 10:40:23 +00:00
455 lines
18 KiB
Go
455 lines
18 KiB
Go
|
/*
|
||
|
Copyright 2024 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 version
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"sort"
|
||
|
"strings"
|
||
|
"sync"
|
||
|
|
||
|
"github.com/spf13/pflag"
|
||
|
utilruntime "k8s.io/apimachinery/pkg/util/runtime"
|
||
|
"k8s.io/apimachinery/pkg/util/version"
|
||
|
cliflag "k8s.io/component-base/cli/flag"
|
||
|
"k8s.io/component-base/featuregate"
|
||
|
"k8s.io/klog/v2"
|
||
|
)
|
||
|
|
||
|
// DefaultComponentGlobalsRegistry is the global var to store the effective versions and feature gates for all components for easy access.
|
||
|
// Example usage:
|
||
|
// // register the component effective version and feature gate first
|
||
|
// _, _ = utilversion.DefaultComponentGlobalsRegistry.ComponentGlobalsOrRegister(utilversion.DefaultKubeComponent, utilversion.DefaultKubeEffectiveVersion(), utilfeature.DefaultMutableFeatureGate)
|
||
|
// wardleEffectiveVersion := utilversion.NewEffectiveVersion("1.2")
|
||
|
// wardleFeatureGate := featuregate.NewFeatureGate()
|
||
|
// utilruntime.Must(utilversion.DefaultComponentGlobalsRegistry.Register(apiserver.WardleComponentName, wardleEffectiveVersion, wardleFeatureGate, false))
|
||
|
//
|
||
|
// cmd := &cobra.Command{
|
||
|
// ...
|
||
|
// // call DefaultComponentGlobalsRegistry.Set() in PersistentPreRunE
|
||
|
// PersistentPreRunE: func(*cobra.Command, []string) error {
|
||
|
// if err := utilversion.DefaultComponentGlobalsRegistry.Set(); err != nil {
|
||
|
// return err
|
||
|
// }
|
||
|
// ...
|
||
|
// },
|
||
|
// RunE: func(c *cobra.Command, args []string) error {
|
||
|
// // call utilversion.DefaultComponentGlobalsRegistry.Validate() somewhere
|
||
|
// },
|
||
|
// }
|
||
|
//
|
||
|
// flags := cmd.Flags()
|
||
|
// // add flags
|
||
|
// utilversion.DefaultComponentGlobalsRegistry.AddFlags(flags)
|
||
|
var DefaultComponentGlobalsRegistry ComponentGlobalsRegistry = NewComponentGlobalsRegistry()
|
||
|
|
||
|
const (
|
||
|
DefaultKubeComponent = "kube"
|
||
|
|
||
|
klogLevel = 2
|
||
|
)
|
||
|
|
||
|
type VersionMapping func(from *version.Version) *version.Version
|
||
|
|
||
|
// ComponentGlobals stores the global variables for a component for easy access.
|
||
|
type ComponentGlobals struct {
|
||
|
effectiveVersion MutableEffectiveVersion
|
||
|
featureGate featuregate.MutableVersionedFeatureGate
|
||
|
|
||
|
// emulationVersionMapping contains the mapping from the emulation version of this component
|
||
|
// to the emulation version of another component.
|
||
|
emulationVersionMapping map[string]VersionMapping
|
||
|
// dependentEmulationVersion stores whether or not this component's EmulationVersion is dependent through mapping on another component.
|
||
|
// If true, the emulation version cannot be set from the flag, or version mapping from another component.
|
||
|
dependentEmulationVersion bool
|
||
|
// minCompatibilityVersionMapping contains the mapping from the min compatibility version of this component
|
||
|
// to the min compatibility version of another component.
|
||
|
minCompatibilityVersionMapping map[string]VersionMapping
|
||
|
// dependentMinCompatibilityVersion stores whether or not this component's MinCompatibilityVersion is dependent through mapping on another component
|
||
|
// If true, the min compatibility version cannot be set from the flag, or version mapping from another component.
|
||
|
dependentMinCompatibilityVersion bool
|
||
|
}
|
||
|
|
||
|
type ComponentGlobalsRegistry interface {
|
||
|
// EffectiveVersionFor returns the EffectiveVersion registered under the component.
|
||
|
// Returns nil if the component is not registered.
|
||
|
EffectiveVersionFor(component string) EffectiveVersion
|
||
|
// FeatureGateFor returns the FeatureGate registered under the component.
|
||
|
// Returns nil if the component is not registered.
|
||
|
FeatureGateFor(component string) featuregate.FeatureGate
|
||
|
// Register registers the EffectiveVersion and FeatureGate for a component.
|
||
|
// returns error if the component is already registered.
|
||
|
Register(component string, effectiveVersion MutableEffectiveVersion, featureGate featuregate.MutableVersionedFeatureGate) error
|
||
|
// ComponentGlobalsOrRegister would return the registered global variables for the component if it already exists in the registry.
|
||
|
// Otherwise, the provided variables would be registered under the component, and the same variables would be returned.
|
||
|
ComponentGlobalsOrRegister(component string, effectiveVersion MutableEffectiveVersion, featureGate featuregate.MutableVersionedFeatureGate) (MutableEffectiveVersion, featuregate.MutableVersionedFeatureGate)
|
||
|
// AddFlags adds flags of "--emulated-version" and "--feature-gates"
|
||
|
AddFlags(fs *pflag.FlagSet)
|
||
|
// Set sets the flags for all global variables for all components registered.
|
||
|
Set() error
|
||
|
// SetFallback calls Set() if it has never been called.
|
||
|
SetFallback() error
|
||
|
// Validate calls the Validate() function for all the global variables for all components registered.
|
||
|
Validate() []error
|
||
|
// Reset removes all stored ComponentGlobals, configurations, and version mappings.
|
||
|
Reset()
|
||
|
// SetEmulationVersionMapping sets the mapping from the emulation version of one component
|
||
|
// to the emulation version of another component.
|
||
|
// Once set, the emulation version of the toComponent will be determined by the emulation version of the fromComponent,
|
||
|
// and cannot be set from cmd flags anymore.
|
||
|
// For a given component, its emulation version can only depend on one other component, no multiple dependency is allowed.
|
||
|
SetEmulationVersionMapping(fromComponent, toComponent string, f VersionMapping) error
|
||
|
}
|
||
|
|
||
|
type componentGlobalsRegistry struct {
|
||
|
componentGlobals map[string]*ComponentGlobals
|
||
|
mutex sync.RWMutex
|
||
|
// list of component name to emulation version set from the flag.
|
||
|
emulationVersionConfig []string
|
||
|
// map of component name to the list of feature gates set from the flag.
|
||
|
featureGatesConfig map[string][]string
|
||
|
// set stores if the Set() function for the registry is already called.
|
||
|
set bool
|
||
|
}
|
||
|
|
||
|
func NewComponentGlobalsRegistry() *componentGlobalsRegistry {
|
||
|
return &componentGlobalsRegistry{
|
||
|
componentGlobals: make(map[string]*ComponentGlobals),
|
||
|
emulationVersionConfig: nil,
|
||
|
featureGatesConfig: nil,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) Reset() {
|
||
|
r.mutex.Lock()
|
||
|
defer r.mutex.Unlock()
|
||
|
r.componentGlobals = make(map[string]*ComponentGlobals)
|
||
|
r.emulationVersionConfig = nil
|
||
|
r.featureGatesConfig = nil
|
||
|
r.set = false
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) EffectiveVersionFor(component string) EffectiveVersion {
|
||
|
r.mutex.RLock()
|
||
|
defer r.mutex.RUnlock()
|
||
|
globals, ok := r.componentGlobals[component]
|
||
|
if !ok {
|
||
|
return nil
|
||
|
}
|
||
|
return globals.effectiveVersion
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) FeatureGateFor(component string) featuregate.FeatureGate {
|
||
|
r.mutex.RLock()
|
||
|
defer r.mutex.RUnlock()
|
||
|
globals, ok := r.componentGlobals[component]
|
||
|
if !ok {
|
||
|
return nil
|
||
|
}
|
||
|
return globals.featureGate
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) unsafeRegister(component string, effectiveVersion MutableEffectiveVersion, featureGate featuregate.MutableVersionedFeatureGate) error {
|
||
|
if _, ok := r.componentGlobals[component]; ok {
|
||
|
return fmt.Errorf("component globals of %s already registered", component)
|
||
|
}
|
||
|
if featureGate != nil {
|
||
|
if err := featureGate.SetEmulationVersion(effectiveVersion.EmulationVersion()); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
c := ComponentGlobals{
|
||
|
effectiveVersion: effectiveVersion,
|
||
|
featureGate: featureGate,
|
||
|
emulationVersionMapping: make(map[string]VersionMapping),
|
||
|
minCompatibilityVersionMapping: make(map[string]VersionMapping),
|
||
|
}
|
||
|
r.componentGlobals[component] = &c
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) Register(component string, effectiveVersion MutableEffectiveVersion, featureGate featuregate.MutableVersionedFeatureGate) error {
|
||
|
if effectiveVersion == nil {
|
||
|
return fmt.Errorf("cannot register nil effectiveVersion")
|
||
|
}
|
||
|
r.mutex.Lock()
|
||
|
defer r.mutex.Unlock()
|
||
|
return r.unsafeRegister(component, effectiveVersion, featureGate)
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) ComponentGlobalsOrRegister(component string, effectiveVersion MutableEffectiveVersion, featureGate featuregate.MutableVersionedFeatureGate) (MutableEffectiveVersion, featuregate.MutableVersionedFeatureGate) {
|
||
|
r.mutex.Lock()
|
||
|
defer r.mutex.Unlock()
|
||
|
globals, ok := r.componentGlobals[component]
|
||
|
if ok {
|
||
|
return globals.effectiveVersion, globals.featureGate
|
||
|
}
|
||
|
utilruntime.Must(r.unsafeRegister(component, effectiveVersion, featureGate))
|
||
|
return effectiveVersion, featureGate
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) unsafeKnownFeatures() []string {
|
||
|
var known []string
|
||
|
for component, globals := range r.componentGlobals {
|
||
|
if globals.featureGate == nil {
|
||
|
continue
|
||
|
}
|
||
|
for _, f := range globals.featureGate.KnownFeatures() {
|
||
|
known = append(known, component+":"+f)
|
||
|
}
|
||
|
}
|
||
|
sort.Strings(known)
|
||
|
return known
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) unsafeVersionFlagOptions(isEmulation bool) []string {
|
||
|
var vs []string
|
||
|
for component, globals := range r.componentGlobals {
|
||
|
binaryVer := globals.effectiveVersion.BinaryVersion()
|
||
|
if isEmulation {
|
||
|
if globals.dependentEmulationVersion {
|
||
|
continue
|
||
|
}
|
||
|
// emulated version could be between binaryMajor.{binaryMinor} and binaryMajor.{binaryMinor}
|
||
|
// TODO: change to binaryMajor.{binaryMinor-1} and binaryMajor.{binaryMinor} in 1.32
|
||
|
vs = append(vs, fmt.Sprintf("%s=%s..%s (default=%s)", component,
|
||
|
binaryVer.SubtractMinor(0).String(), binaryVer.String(), globals.effectiveVersion.EmulationVersion().String()))
|
||
|
} else {
|
||
|
if globals.dependentMinCompatibilityVersion {
|
||
|
continue
|
||
|
}
|
||
|
// min compatibility version could be between binaryMajor.{binaryMinor-1} and binaryMajor.{binaryMinor}
|
||
|
vs = append(vs, fmt.Sprintf("%s=%s..%s (default=%s)", component,
|
||
|
binaryVer.SubtractMinor(1).String(), binaryVer.String(), globals.effectiveVersion.MinCompatibilityVersion().String()))
|
||
|
}
|
||
|
}
|
||
|
sort.Strings(vs)
|
||
|
return vs
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) AddFlags(fs *pflag.FlagSet) {
|
||
|
if r == nil {
|
||
|
return
|
||
|
}
|
||
|
r.mutex.Lock()
|
||
|
defer r.mutex.Unlock()
|
||
|
for _, globals := range r.componentGlobals {
|
||
|
if globals.featureGate != nil {
|
||
|
globals.featureGate.Close()
|
||
|
}
|
||
|
}
|
||
|
if r.emulationVersionConfig != nil || r.featureGatesConfig != nil {
|
||
|
klog.Warning("calling componentGlobalsRegistry.AddFlags more than once, the registry will be set by the latest flags")
|
||
|
}
|
||
|
r.emulationVersionConfig = []string{}
|
||
|
r.featureGatesConfig = make(map[string][]string)
|
||
|
|
||
|
fs.StringSliceVar(&r.emulationVersionConfig, "emulated-version", r.emulationVersionConfig, ""+
|
||
|
"The versions different components emulate their capabilities (APIs, features, ...) of.\n"+
|
||
|
"If set, the component will emulate the behavior of this version instead of the underlying binary version.\n"+
|
||
|
"Version format could only be major.minor, for example: '--emulated-version=wardle=1.2,kube=1.31'. Options are:\n"+strings.Join(r.unsafeVersionFlagOptions(true), "\n")+
|
||
|
"If the component is not specified, defaults to \"kube\"")
|
||
|
|
||
|
fs.Var(cliflag.NewColonSeparatedMultimapStringStringAllowDefaultEmptyKey(&r.featureGatesConfig), "feature-gates", "Comma-separated list of component:key=value pairs that describe feature gates for alpha/experimental features of different components.\n"+
|
||
|
"If the component is not specified, defaults to \"kube\". This flag can be repeatedly invoked. For example: --feature-gates 'wardle:featureA=true,wardle:featureB=false' --feature-gates 'kube:featureC=true'"+
|
||
|
"Options are:\n"+strings.Join(r.unsafeKnownFeatures(), "\n"))
|
||
|
}
|
||
|
|
||
|
type componentVersion struct {
|
||
|
component string
|
||
|
ver *version.Version
|
||
|
}
|
||
|
|
||
|
// getFullEmulationVersionConfig expands the given version config with version registered version mapping,
|
||
|
// and returns the map of component to Version.
|
||
|
func (r *componentGlobalsRegistry) getFullEmulationVersionConfig(
|
||
|
versionConfigMap map[string]*version.Version) (map[string]*version.Version, error) {
|
||
|
result := map[string]*version.Version{}
|
||
|
setQueue := []componentVersion{}
|
||
|
for comp, ver := range versionConfigMap {
|
||
|
if _, ok := r.componentGlobals[comp]; !ok {
|
||
|
return result, fmt.Errorf("component not registered: %s", comp)
|
||
|
}
|
||
|
klog.V(klogLevel).Infof("setting version %s=%s", comp, ver.String())
|
||
|
setQueue = append(setQueue, componentVersion{comp, ver})
|
||
|
}
|
||
|
for len(setQueue) > 0 {
|
||
|
cv := setQueue[0]
|
||
|
if _, visited := result[cv.component]; visited {
|
||
|
return result, fmt.Errorf("setting version of %s more than once, probably version mapping loop", cv.component)
|
||
|
}
|
||
|
setQueue = setQueue[1:]
|
||
|
result[cv.component] = cv.ver
|
||
|
for toComp, f := range r.componentGlobals[cv.component].emulationVersionMapping {
|
||
|
toVer := f(cv.ver)
|
||
|
if toVer == nil {
|
||
|
return result, fmt.Errorf("got nil version from mapping of %s=%s to component:%s", cv.component, cv.ver.String(), toComp)
|
||
|
}
|
||
|
klog.V(klogLevel).Infof("setting version %s=%s from version mapping of %s=%s", toComp, toVer.String(), cv.component, cv.ver.String())
|
||
|
setQueue = append(setQueue, componentVersion{toComp, toVer})
|
||
|
}
|
||
|
}
|
||
|
return result, nil
|
||
|
}
|
||
|
|
||
|
func toVersionMap(versionConfig []string) (map[string]*version.Version, error) {
|
||
|
m := map[string]*version.Version{}
|
||
|
for _, compVer := range versionConfig {
|
||
|
// default to "kube" of component is not specified
|
||
|
k := "kube"
|
||
|
v := compVer
|
||
|
if strings.Contains(compVer, "=") {
|
||
|
arr := strings.SplitN(compVer, "=", 2)
|
||
|
if len(arr) != 2 {
|
||
|
return m, fmt.Errorf("malformed pair, expect string=string")
|
||
|
}
|
||
|
k = strings.TrimSpace(arr[0])
|
||
|
v = strings.TrimSpace(arr[1])
|
||
|
}
|
||
|
ver, err := version.Parse(v)
|
||
|
if err != nil {
|
||
|
return m, err
|
||
|
}
|
||
|
if ver.Patch() != 0 {
|
||
|
return m, fmt.Errorf("patch version not allowed, got: %s=%s", k, ver.String())
|
||
|
}
|
||
|
if existingVer, ok := m[k]; ok {
|
||
|
return m, fmt.Errorf("duplicate version flag, %s=%s and %s=%s", k, existingVer.String(), k, ver.String())
|
||
|
}
|
||
|
m[k] = ver
|
||
|
}
|
||
|
return m, nil
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) SetFallback() error {
|
||
|
r.mutex.Lock()
|
||
|
set := r.set
|
||
|
r.mutex.Unlock()
|
||
|
if set {
|
||
|
return nil
|
||
|
}
|
||
|
klog.Warning("setting componentGlobalsRegistry in SetFallback. We recommend calling componentGlobalsRegistry.Set()" +
|
||
|
" right after parsing flags to avoid using feature gates before their final values are set by the flags.")
|
||
|
return r.Set()
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) Set() error {
|
||
|
r.mutex.Lock()
|
||
|
defer r.mutex.Unlock()
|
||
|
r.set = true
|
||
|
emulationVersionConfigMap, err := toVersionMap(r.emulationVersionConfig)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
for comp := range emulationVersionConfigMap {
|
||
|
if _, ok := r.componentGlobals[comp]; !ok {
|
||
|
return fmt.Errorf("component not registered: %s", comp)
|
||
|
}
|
||
|
// only components without any dependencies can be set from the flag.
|
||
|
if r.componentGlobals[comp].dependentEmulationVersion {
|
||
|
return fmt.Errorf("EmulationVersion of %s is set by mapping, cannot set it by flag", comp)
|
||
|
}
|
||
|
}
|
||
|
if emulationVersions, err := r.getFullEmulationVersionConfig(emulationVersionConfigMap); err != nil {
|
||
|
return err
|
||
|
} else {
|
||
|
for comp, ver := range emulationVersions {
|
||
|
r.componentGlobals[comp].effectiveVersion.SetEmulationVersion(ver)
|
||
|
}
|
||
|
}
|
||
|
// Set feature gate emulation version before setting feature gate flag values.
|
||
|
for comp, globals := range r.componentGlobals {
|
||
|
if globals.featureGate == nil {
|
||
|
continue
|
||
|
}
|
||
|
klog.V(klogLevel).Infof("setting %s:feature gate emulation version to %s", comp, globals.effectiveVersion.EmulationVersion().String())
|
||
|
if err := globals.featureGate.SetEmulationVersion(globals.effectiveVersion.EmulationVersion()); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
for comp, fg := range r.featureGatesConfig {
|
||
|
if comp == "" {
|
||
|
if _, ok := r.featureGatesConfig[DefaultKubeComponent]; ok {
|
||
|
return fmt.Errorf("set kube feature gates with default empty prefix or kube: prefix consistently, do not mix use")
|
||
|
}
|
||
|
comp = DefaultKubeComponent
|
||
|
}
|
||
|
if _, ok := r.componentGlobals[comp]; !ok {
|
||
|
return fmt.Errorf("component not registered: %s", comp)
|
||
|
}
|
||
|
featureGate := r.componentGlobals[comp].featureGate
|
||
|
if featureGate == nil {
|
||
|
return fmt.Errorf("component featureGate not registered: %s", comp)
|
||
|
}
|
||
|
flagVal := strings.Join(fg, ",")
|
||
|
klog.V(klogLevel).Infof("setting %s:feature-gates=%s", comp, flagVal)
|
||
|
if err := featureGate.Set(flagVal); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) Validate() []error {
|
||
|
var errs []error
|
||
|
r.mutex.Lock()
|
||
|
defer r.mutex.Unlock()
|
||
|
for _, globals := range r.componentGlobals {
|
||
|
errs = append(errs, globals.effectiveVersion.Validate()...)
|
||
|
if globals.featureGate != nil {
|
||
|
errs = append(errs, globals.featureGate.Validate()...)
|
||
|
}
|
||
|
}
|
||
|
return errs
|
||
|
}
|
||
|
|
||
|
func (r *componentGlobalsRegistry) SetEmulationVersionMapping(fromComponent, toComponent string, f VersionMapping) error {
|
||
|
if f == nil {
|
||
|
return nil
|
||
|
}
|
||
|
klog.V(klogLevel).Infof("setting EmulationVersion mapping from %s to %s", fromComponent, toComponent)
|
||
|
r.mutex.Lock()
|
||
|
defer r.mutex.Unlock()
|
||
|
if _, ok := r.componentGlobals[fromComponent]; !ok {
|
||
|
return fmt.Errorf("component not registered: %s", fromComponent)
|
||
|
}
|
||
|
if _, ok := r.componentGlobals[toComponent]; !ok {
|
||
|
return fmt.Errorf("component not registered: %s", toComponent)
|
||
|
}
|
||
|
// check multiple dependency
|
||
|
if r.componentGlobals[toComponent].dependentEmulationVersion {
|
||
|
return fmt.Errorf("mapping of %s already exists from another component", toComponent)
|
||
|
}
|
||
|
r.componentGlobals[toComponent].dependentEmulationVersion = true
|
||
|
|
||
|
versionMapping := r.componentGlobals[fromComponent].emulationVersionMapping
|
||
|
if _, ok := versionMapping[toComponent]; ok {
|
||
|
return fmt.Errorf("EmulationVersion from %s to %s already exists", fromComponent, toComponent)
|
||
|
}
|
||
|
versionMapping[toComponent] = f
|
||
|
klog.V(klogLevel).Infof("setting the default EmulationVersion of %s based on mapping from the default EmulationVersion of %s", fromComponent, toComponent)
|
||
|
defaultFromVersion := r.componentGlobals[fromComponent].effectiveVersion.EmulationVersion()
|
||
|
emulationVersions, err := r.getFullEmulationVersionConfig(map[string]*version.Version{fromComponent: defaultFromVersion})
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
for comp, ver := range emulationVersions {
|
||
|
r.componentGlobals[comp].effectiveVersion.SetEmulationVersion(ver)
|
||
|
}
|
||
|
return nil
|
||
|
}
|