mirror of
https://github.com/prometheus/alertmanager.git
synced 2026-02-05 15:45:34 +01:00
* feat(limit): add new limit package with bucket Add a new limit package with generic bucket implementation. This can be used for example to limit the number of alerts in memory. Benchmarks: ```go goos: darwin goarch: arm64 pkg: github.com/prometheus/alertmanager/limit cpu: Apple M3 Pro BenchmarkBucketUpsert/EmptyBucket-12 8816954 122.4 ns/op 56 B/op 2 allocs/op BenchmarkBucketUpsert/AddToFullBucketWithExpiredItems-12 9861010 123.0 ns/op 56 B/op 2 allocs/op BenchmarkBucketUpsert/AddToFullBucketWithActiveItems-12 8343778 143.6 ns/op 56 B/op 2 allocs/op BenchmarkBucketUpsert/UpdateExistingAlert-12 10107787 118.9 ns/op 56 B/op 2 allocs/op BenchmarkBucketUpsert/MixedWorkload-12 9436174 126.0 ns/op 56 B/op 2 allocs/op BenchmarkBucketUpsertScaling/BucketSize_10-12 10255278 115.4 ns/op 56 B/op 2 allocs/op BenchmarkBucketUpsertScaling/BucketSize_50-12 10166518 117.1 ns/op 56 B/op 2 allocs/op BenchmarkBucketUpsertScaling/BucketSize_100-12 10457394 115.0 ns/op 56 B/op 2 allocs/op BenchmarkBucketUpsertScaling/BucketSize_500-12 9644079 115.2 ns/op 56 B/op 2 allocs/op BenchmarkBucketUpsertScaling/BucketSize_1000-12 10426184 116.6 ns/op 56 B/op 2 allocs/op BenchmarkBucketUpsertConcurrent-12 5796210 216.3 ns/op 406 B/op 5 allocs/op PASS ok github.com/prometheus/alertmanager/limit 15.497s ``` Signed-off-by: Siavash Safi <siavash@cloudflare.com> * feat(provider): implement per-alert limits Use the new limit module to add optional per alert-name limits. The metrics for limited alerts can be enabled using `alerts-limited-metric` feature flag. Signed-off-by: Siavash Safi <siavash@cloudflare.com> --------- Signed-off-by: Siavash Safi <siavash@cloudflare.com>
179 lines
4.6 KiB
Go
179 lines
4.6 KiB
Go
// Copyright 2023 Prometheus Team
|
|
// 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 featurecontrol
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"log/slog"
|
|
"strings"
|
|
)
|
|
|
|
const (
|
|
FeatureAlertNamesInMetrics = "alert-names-in-metrics"
|
|
FeatureReceiverNameInMetrics = "receiver-name-in-metrics"
|
|
FeatureClassicMode = "classic-mode"
|
|
FeatureUTF8StrictMode = "utf8-strict-mode"
|
|
FeatureAutoGOMEMLIMIT = "auto-gomemlimit"
|
|
FeatureAutoGOMAXPROCS = "auto-gomaxprocs"
|
|
)
|
|
|
|
var AllowedFlags = []string{
|
|
FeatureAlertNamesInMetrics,
|
|
FeatureReceiverNameInMetrics,
|
|
FeatureClassicMode,
|
|
FeatureUTF8StrictMode,
|
|
FeatureAutoGOMEMLIMIT,
|
|
FeatureAutoGOMAXPROCS,
|
|
}
|
|
|
|
type Flagger interface {
|
|
EnableAlertNamesInMetrics() bool
|
|
EnableReceiverNamesInMetrics() bool
|
|
ClassicMode() bool
|
|
UTF8StrictMode() bool
|
|
EnableAutoGOMEMLIMIT() bool
|
|
EnableAutoGOMAXPROCS() bool
|
|
}
|
|
|
|
type Flags struct {
|
|
logger *slog.Logger
|
|
enableAlertNamesInMetrics bool
|
|
enableReceiverNamesInMetrics bool
|
|
classicMode bool
|
|
utf8StrictMode bool
|
|
enableAutoGOMEMLIMIT bool
|
|
enableAutoGOMAXPROCS bool
|
|
}
|
|
|
|
func (f *Flags) EnableAlertNamesInMetrics() bool {
|
|
return f.enableAlertNamesInMetrics
|
|
}
|
|
|
|
func (f *Flags) EnableReceiverNamesInMetrics() bool {
|
|
return f.enableReceiverNamesInMetrics
|
|
}
|
|
|
|
func (f *Flags) ClassicMode() bool {
|
|
return f.classicMode
|
|
}
|
|
|
|
func (f *Flags) UTF8StrictMode() bool {
|
|
return f.utf8StrictMode
|
|
}
|
|
|
|
func (f *Flags) EnableAutoGOMEMLIMIT() bool {
|
|
return f.enableAutoGOMEMLIMIT
|
|
}
|
|
|
|
func (f *Flags) EnableAutoGOMAXPROCS() bool {
|
|
return f.enableAutoGOMAXPROCS
|
|
}
|
|
|
|
type flagOption func(flags *Flags)
|
|
|
|
func enableReceiverNameInMetrics() flagOption {
|
|
return func(configs *Flags) {
|
|
configs.enableReceiverNamesInMetrics = true
|
|
}
|
|
}
|
|
|
|
func enableClassicMode() flagOption {
|
|
return func(configs *Flags) {
|
|
configs.classicMode = true
|
|
}
|
|
}
|
|
|
|
func enableUTF8StrictMode() flagOption {
|
|
return func(configs *Flags) {
|
|
configs.utf8StrictMode = true
|
|
}
|
|
}
|
|
|
|
func enableAutoGOMEMLIMIT() flagOption {
|
|
return func(configs *Flags) {
|
|
configs.enableAutoGOMEMLIMIT = true
|
|
}
|
|
}
|
|
|
|
func enableAutoGOMAXPROCS() flagOption {
|
|
return func(configs *Flags) {
|
|
configs.enableAutoGOMAXPROCS = true
|
|
}
|
|
}
|
|
|
|
func enableAlertNamesInMetrics() flagOption {
|
|
return func(configs *Flags) {
|
|
configs.enableAlertNamesInMetrics = true
|
|
}
|
|
}
|
|
|
|
func NewFlags(logger *slog.Logger, features string) (Flagger, error) {
|
|
fc := &Flags{logger: logger}
|
|
opts := []flagOption{}
|
|
|
|
if len(features) == 0 {
|
|
return NoopFlags{}, nil
|
|
}
|
|
|
|
for feature := range strings.SplitSeq(features, ",") {
|
|
switch feature {
|
|
case FeatureAlertNamesInMetrics:
|
|
opts = append(opts, enableAlertNamesInMetrics())
|
|
logger.Warn("Alert names in metrics enabled")
|
|
case FeatureReceiverNameInMetrics:
|
|
opts = append(opts, enableReceiverNameInMetrics())
|
|
logger.Warn("Experimental receiver name in metrics enabled")
|
|
case FeatureClassicMode:
|
|
opts = append(opts, enableClassicMode())
|
|
logger.Warn("Classic mode enabled")
|
|
case FeatureUTF8StrictMode:
|
|
opts = append(opts, enableUTF8StrictMode())
|
|
logger.Warn("UTF-8 strict mode enabled")
|
|
case FeatureAutoGOMEMLIMIT:
|
|
opts = append(opts, enableAutoGOMEMLIMIT())
|
|
logger.Warn("Automatically set GOMEMLIMIT to match the Linux container or system memory limit.")
|
|
case FeatureAutoGOMAXPROCS:
|
|
opts = append(opts, enableAutoGOMAXPROCS())
|
|
logger.Warn("Automatically set GOMAXPROCS to match Linux container CPU quota")
|
|
default:
|
|
return nil, fmt.Errorf("unknown option '%s' for --enable-feature", feature)
|
|
}
|
|
}
|
|
|
|
for _, opt := range opts {
|
|
opt(fc)
|
|
}
|
|
|
|
if fc.classicMode && fc.utf8StrictMode {
|
|
return nil, errors.New("cannot have both classic and UTF-8 modes enabled")
|
|
}
|
|
|
|
return fc, nil
|
|
}
|
|
|
|
type NoopFlags struct{}
|
|
|
|
func (n NoopFlags) EnableAlertNamesInMetrics() bool { return false }
|
|
|
|
func (n NoopFlags) EnableReceiverNamesInMetrics() bool { return false }
|
|
|
|
func (n NoopFlags) ClassicMode() bool { return false }
|
|
|
|
func (n NoopFlags) UTF8StrictMode() bool { return false }
|
|
|
|
func (n NoopFlags) EnableAutoGOMEMLIMIT() bool { return false }
|
|
|
|
func (n NoopFlags) EnableAutoGOMAXPROCS() bool { return false }
|