// Copyright The OpenTelemetry Authors // SPDX-License-Identifier: Apache-2.0 package featuregate // import "go.opentelemetry.io/collector/featuregate" import ( "fmt" "sort" "sync" "sync/atomic" ) var globalRegistry = NewRegistry() // GlobalRegistry returns the global Registry. func GlobalRegistry() *Registry { return globalRegistry } type Registry struct { gates sync.Map } // NewRegistry returns a new empty Registry. func NewRegistry() *Registry { return &Registry{} } // RegisterOption allows to configure additional information about a Gate during registration. type RegisterOption interface { apply(g *Gate) } type registerOptionFunc func(g *Gate) func (ro registerOptionFunc) apply(g *Gate) { ro(g) } // WithRegisterDescription adds description for the Gate. func WithRegisterDescription(description string) RegisterOption { return registerOptionFunc(func(g *Gate) { g.description = description }) } // WithRegisterReferenceURL adds a URL that has all the contextual information about the Gate. func WithRegisterReferenceURL(url string) RegisterOption { return registerOptionFunc(func(g *Gate) { g.referenceURL = url }) } // WithRegisterFromVersion is used to set the Gate "FromVersion". // The "FromVersion" contains the Collector release when a feature is introduced. func WithRegisterFromVersion(fromVersion string) RegisterOption { return registerOptionFunc(func(g *Gate) { g.fromVersion = fromVersion }) } // WithRegisterToVersion is used to set the Gate "ToVersion". // The "ToVersion", if not empty, contains the last Collector release in which you can still use a feature gate. // If the feature stage is either "Deprecated" or "Stable", the "ToVersion" is the Collector release when the feature is removed. func WithRegisterToVersion(toVersion string) RegisterOption { return registerOptionFunc(func(g *Gate) { g.toVersion = toVersion }) } // MustRegister like Register but panics if an invalid ID or gate options are provided. func (r *Registry) MustRegister(id string, stage Stage, opts ...RegisterOption) *Gate { g, err := r.Register(id, stage, opts...) if err != nil { panic(err) } return g } // Register a Gate and return it. The returned Gate can be used to check if is enabled or not. func (r *Registry) Register(id string, stage Stage, opts ...RegisterOption) (*Gate, error) { g := &Gate{ id: id, stage: stage, } for _, opt := range opts { opt.apply(g) } switch g.stage { case StageAlpha, StageDeprecated: g.enabled = &atomic.Bool{} case StageBeta, StageStable: enabled := &atomic.Bool{} enabled.Store(true) g.enabled = enabled default: return nil, fmt.Errorf("unknown stage value %q for gate %q", stage, id) } if (g.stage == StageStable || g.stage == StageDeprecated) && g.toVersion == "" { return nil, fmt.Errorf("no removal version set for %v gate %q", g.stage.String(), id) } if _, loaded := r.gates.LoadOrStore(id, g); loaded { return nil, fmt.Errorf("attempted to add pre-existing gate %q", id) } return g, nil } // Set the enabled valued for a Gate identified by the given id. func (r *Registry) Set(id string, enabled bool) error { v, ok := r.gates.Load(id) if !ok { return fmt.Errorf("no such feature gate %q", id) } g := v.(*Gate) switch g.stage { case StageStable: if !enabled { return fmt.Errorf("feature gate %q is stable, can not be disabled", id) } fmt.Printf("Feature gate %q is stable and already enabled. It will be removed in version %v and continued use of the gate after version %v will result in an error.\n", id, g.toVersion, g.toVersion) case StageDeprecated: if enabled { return fmt.Errorf("feature gate %q is deprecated, can not be enabled", id) } fmt.Printf("Feature gate %q is deprecated and already disabled. It will be removed in version %v and continued use of the gate after version %v will result in an error.\n", id, g.toVersion, g.toVersion) default: g.enabled.Store(enabled) } return nil } // VisitAll visits all the gates in lexicographical order, calling fn for each. func (r *Registry) VisitAll(fn func(*Gate)) { var gates []*Gate r.gates.Range(func(key, value any) bool { gates = append(gates, value.(*Gate)) return true }) sort.Slice(gates, func(i, j int) bool { return gates[i].ID() < gates[j].ID() }) for i := range gates { fn(gates[i]) } }