mirror of
https://github.com/prometheus/prometheus.git
synced 2025-03-05 20:59:13 -08:00
Merge branch 'main' into implement-ct-in-append
This commit is contained in:
commit
18f95cc994
11
CHANGELOG.md
11
CHANGELOG.md
|
@ -6,7 +6,15 @@
|
||||||
* [ENHANCEMENT] OTLP receiver: Warn when encountering exponential histograms with zero count and non-zero sum. #14706
|
* [ENHANCEMENT] OTLP receiver: Warn when encountering exponential histograms with zero count and non-zero sum. #14706
|
||||||
* [BUGFIX] tsdb/wlog.Watcher.readSegmentForGC: Only count unknown record types against record_decode_failures_total metric. #14042
|
* [BUGFIX] tsdb/wlog.Watcher.readSegmentForGC: Only count unknown record types against record_decode_failures_total metric. #14042
|
||||||
|
|
||||||
## 2.54.0-rc.1 / 2024-08-05
|
## 2.54.1 / 2024-08-27
|
||||||
|
|
||||||
|
* [BUGFIX] Scraping: allow multiple samples on same series, with explicit timestamps. #14685
|
||||||
|
* [BUGFIX] Docker SD: fix crash in `match_first_network` mode when container is reconnected to a new network. #14654
|
||||||
|
* [BUGFIX] PromQL: fix experimental native histograms getting corrupted due to vector selector bug in range queries. #14538
|
||||||
|
* [BUGFIX] PromQL: fix experimental native histogram counter reset detection on stale samples. #14514
|
||||||
|
* [BUGFIX] PromQL: fix native histograms getting corrupted due to vector selector bug in range queries. #14605
|
||||||
|
|
||||||
|
## 2.54.0 / 2024-08-09
|
||||||
|
|
||||||
Release 2.54 brings a release candidate of a major new version of [Remote Write: 2.0](https://prometheus.io/docs/specs/remote_write_spec_2_0/).
|
Release 2.54 brings a release candidate of a major new version of [Remote Write: 2.0](https://prometheus.io/docs/specs/remote_write_spec_2_0/).
|
||||||
This is experimental at this time and may still change.
|
This is experimental at this time and may still change.
|
||||||
|
@ -35,6 +43,7 @@ Remote-write v2 is enabled by default, but can be disabled via feature-flag `web
|
||||||
* [ENHANCEMENT] Notifier: Send any outstanding Alertmanager notifications when shutting down. #14290
|
* [ENHANCEMENT] Notifier: Send any outstanding Alertmanager notifications when shutting down. #14290
|
||||||
* [ENHANCEMENT] Rules: Add label-matcher support to Rules API. #10194
|
* [ENHANCEMENT] Rules: Add label-matcher support to Rules API. #10194
|
||||||
* [ENHANCEMENT] HTTP API: Add url to message logged on error while sending response. #14209
|
* [ENHANCEMENT] HTTP API: Add url to message logged on error while sending response. #14209
|
||||||
|
* [BUGFIX] TSDB: Exclude OOO chunks mapped after compaction starts (introduced by #14396). #14584
|
||||||
* [BUGFIX] CLI: escape `|` characters when generating docs. #14420
|
* [BUGFIX] CLI: escape `|` characters when generating docs. #14420
|
||||||
* [BUGFIX] PromQL (experimental native histograms): Fix some binary operators between native histogram values. #14454
|
* [BUGFIX] PromQL (experimental native histograms): Fix some binary operators between native histogram values. #14454
|
||||||
* [BUGFIX] TSDB: LabelNames API could fail during compaction. #14279
|
* [BUGFIX] TSDB: LabelNames API could fail during compaction. #14279
|
||||||
|
|
|
@ -169,6 +169,8 @@ type flagConfig struct {
|
||||||
corsRegexString string
|
corsRegexString string
|
||||||
|
|
||||||
promlogConfig promlog.Config
|
promlogConfig promlog.Config
|
||||||
|
|
||||||
|
promqlEnableDelayedNameRemoval bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// setFeatureListOptions sets the corresponding options from the featureList.
|
// setFeatureListOptions sets the corresponding options from the featureList.
|
||||||
|
@ -238,6 +240,9 @@ func (c *flagConfig) setFeatureListOptions(logger log.Logger) error {
|
||||||
case "delayed-compaction":
|
case "delayed-compaction":
|
||||||
c.tsdb.EnableDelayedCompaction = true
|
c.tsdb.EnableDelayedCompaction = true
|
||||||
level.Info(logger).Log("msg", "Experimental delayed compaction is enabled.")
|
level.Info(logger).Log("msg", "Experimental delayed compaction is enabled.")
|
||||||
|
case "promql-delayed-name-removal":
|
||||||
|
c.promqlEnableDelayedNameRemoval = true
|
||||||
|
level.Info(logger).Log("msg", "Experimental PromQL delayed name removal enabled.")
|
||||||
case "utf8-names":
|
case "utf8-names":
|
||||||
model.NameValidationScheme = model.UTF8Validation
|
model.NameValidationScheme = model.UTF8Validation
|
||||||
level.Info(logger).Log("msg", "Experimental UTF-8 support enabled")
|
level.Info(logger).Log("msg", "Experimental UTF-8 support enabled")
|
||||||
|
@ -487,7 +492,7 @@ func main() {
|
||||||
|
|
||||||
a.Flag("scrape.name-escaping-scheme", `Method for escaping legacy invalid names when sending to Prometheus that does not support UTF-8. Can be one of "values", "underscores", or "dots".`).Default(scrape.DefaultNameEscapingScheme.String()).StringVar(&cfg.nameEscapingScheme)
|
a.Flag("scrape.name-escaping-scheme", `Method for escaping legacy invalid names when sending to Prometheus that does not support UTF-8. Can be one of "values", "underscores", or "dots".`).Default(scrape.DefaultNameEscapingScheme.String()).StringVar(&cfg.nameEscapingScheme)
|
||||||
|
|
||||||
a.Flag("enable-feature", "Comma separated feature names to enable. Valid options: agent, auto-gomemlimit, exemplar-storage, expand-external-labels, memory-snapshot-on-shutdown, promql-per-step-stats, promql-experimental-functions, remote-write-receiver (DEPRECATED), extra-scrape-metrics, new-service-discovery-manager, auto-gomaxprocs, no-default-scrape-port, native-histograms, otlp-write-receiver, created-timestamp-zero-ingestion, concurrent-rule-eval, delayed-compaction, utf8-names. See https://prometheus.io/docs/prometheus/latest/feature_flags/ for more details.").
|
a.Flag("enable-feature", "Comma separated feature names to enable. Valid options: agent, auto-gomaxprocs, auto-gomemlimit, concurrent-rule-eval, created-timestamp-zero-ingestion, delayed-compaction, exemplar-storage, expand-external-labels, extra-scrape-metrics, memory-snapshot-on-shutdown, native-histograms, new-service-discovery-manager, no-default-scrape-port, otlp-write-receiver, promql-experimental-functions, promql-delayed-name-removal, promql-per-step-stats, remote-write-receiver (DEPRECATED), utf8-names. See https://prometheus.io/docs/prometheus/latest/feature_flags/ for more details.").
|
||||||
Default("").StringsVar(&cfg.featureList)
|
Default("").StringsVar(&cfg.featureList)
|
||||||
|
|
||||||
promlogflag.AddFlags(a, &cfg.promlogConfig)
|
promlogflag.AddFlags(a, &cfg.promlogConfig)
|
||||||
|
@ -799,9 +804,10 @@ func main() {
|
||||||
NoStepSubqueryIntervalFn: noStepSubqueryInterval.Get,
|
NoStepSubqueryIntervalFn: noStepSubqueryInterval.Get,
|
||||||
// EnableAtModifier and EnableNegativeOffset have to be
|
// EnableAtModifier and EnableNegativeOffset have to be
|
||||||
// always on for regular PromQL as of Prometheus v2.33.
|
// always on for regular PromQL as of Prometheus v2.33.
|
||||||
EnableAtModifier: true,
|
EnableAtModifier: true,
|
||||||
EnableNegativeOffset: true,
|
EnableNegativeOffset: true,
|
||||||
EnablePerStepStats: cfg.enablePerStepStats,
|
EnablePerStepStats: cfg.enablePerStepStats,
|
||||||
|
EnableDelayedNameRemoval: cfg.promqlEnableDelayedNameRemoval,
|
||||||
}
|
}
|
||||||
|
|
||||||
queryEngine = promql.NewEngine(opts)
|
queryEngine = promql.NewEngine(opts)
|
||||||
|
|
|
@ -85,7 +85,7 @@ func getCompatibleBlockDuration(maxBlockDuration int64) int64 {
|
||||||
return blockDuration
|
return blockDuration
|
||||||
}
|
}
|
||||||
|
|
||||||
func createBlocks(input []byte, mint, maxt, maxBlockDuration int64, maxSamplesInAppender int, outputDir string, humanReadable, quiet bool) (returnErr error) {
|
func createBlocks(input []byte, mint, maxt, maxBlockDuration int64, maxSamplesInAppender int, outputDir string, humanReadable, quiet bool, customLabels map[string]string) (returnErr error) {
|
||||||
blockDuration := getCompatibleBlockDuration(maxBlockDuration)
|
blockDuration := getCompatibleBlockDuration(maxBlockDuration)
|
||||||
mint = blockDuration * (mint / blockDuration)
|
mint = blockDuration * (mint / blockDuration)
|
||||||
|
|
||||||
|
@ -102,6 +102,8 @@ func createBlocks(input []byte, mint, maxt, maxBlockDuration int64, maxSamplesIn
|
||||||
nextSampleTs int64 = math.MaxInt64
|
nextSampleTs int64 = math.MaxInt64
|
||||||
)
|
)
|
||||||
|
|
||||||
|
lb := labels.NewBuilder(labels.EmptyLabels())
|
||||||
|
|
||||||
for t := mint; t <= maxt; t += blockDuration {
|
for t := mint; t <= maxt; t += blockDuration {
|
||||||
tsUpper := t + blockDuration
|
tsUpper := t + blockDuration
|
||||||
if nextSampleTs != math.MaxInt64 && nextSampleTs >= tsUpper {
|
if nextSampleTs != math.MaxInt64 && nextSampleTs >= tsUpper {
|
||||||
|
@ -162,7 +164,13 @@ func createBlocks(input []byte, mint, maxt, maxBlockDuration int64, maxSamplesIn
|
||||||
l := labels.Labels{}
|
l := labels.Labels{}
|
||||||
p.Metric(&l)
|
p.Metric(&l)
|
||||||
|
|
||||||
if _, err := app.Append(0, l, *ts, v); err != nil {
|
lb.Reset(l)
|
||||||
|
for name, value := range customLabels {
|
||||||
|
lb.Set(name, value)
|
||||||
|
}
|
||||||
|
lbls := lb.Labels()
|
||||||
|
|
||||||
|
if _, err := app.Append(0, lbls, *ts, v); err != nil {
|
||||||
return fmt.Errorf("add sample: %w", err)
|
return fmt.Errorf("add sample: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -221,13 +229,13 @@ func createBlocks(input []byte, mint, maxt, maxBlockDuration int64, maxSamplesIn
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func backfill(maxSamplesInAppender int, input []byte, outputDir string, humanReadable, quiet bool, maxBlockDuration time.Duration) (err error) {
|
func backfill(maxSamplesInAppender int, input []byte, outputDir string, humanReadable, quiet bool, maxBlockDuration time.Duration, customLabels map[string]string) (err error) {
|
||||||
p := textparse.NewOpenMetricsParser(input, nil) // Don't need a SymbolTable to get max and min timestamps.
|
p := textparse.NewOpenMetricsParser(input, nil) // Don't need a SymbolTable to get max and min timestamps.
|
||||||
maxt, mint, err := getMinAndMaxTimestamps(p)
|
maxt, mint, err := getMinAndMaxTimestamps(p)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("getting min and max timestamp: %w", err)
|
return fmt.Errorf("getting min and max timestamp: %w", err)
|
||||||
}
|
}
|
||||||
if err = createBlocks(input, mint, maxt, int64(maxBlockDuration/time.Millisecond), maxSamplesInAppender, outputDir, humanReadable, quiet); err != nil {
|
if err = createBlocks(input, mint, maxt, int64(maxBlockDuration/time.Millisecond), maxSamplesInAppender, outputDir, humanReadable, quiet, customLabels); err != nil {
|
||||||
return fmt.Errorf("block creation: %w", err)
|
return fmt.Errorf("block creation: %w", err)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
|
|
@ -92,6 +92,7 @@ func TestBackfill(t *testing.T) {
|
||||||
Description string
|
Description string
|
||||||
MaxSamplesInAppender int
|
MaxSamplesInAppender int
|
||||||
MaxBlockDuration time.Duration
|
MaxBlockDuration time.Duration
|
||||||
|
Labels map[string]string
|
||||||
Expected struct {
|
Expected struct {
|
||||||
MinTime int64
|
MinTime int64
|
||||||
MaxTime int64
|
MaxTime int64
|
||||||
|
@ -636,6 +637,49 @@ http_requests_total{code="400"} 1024 7199
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
ToParse: `# HELP http_requests_total The total number of HTTP requests.
|
||||||
|
# TYPE http_requests_total counter
|
||||||
|
http_requests_total{code="200"} 1 1624463088.000
|
||||||
|
http_requests_total{code="200"} 2 1629503088.000
|
||||||
|
http_requests_total{code="200"} 3 1629863088.000
|
||||||
|
# EOF
|
||||||
|
`,
|
||||||
|
IsOk: true,
|
||||||
|
Description: "Sample with external labels.",
|
||||||
|
MaxSamplesInAppender: 5000,
|
||||||
|
MaxBlockDuration: 2048 * time.Hour,
|
||||||
|
Labels: map[string]string{"cluster_id": "123", "org_id": "999"},
|
||||||
|
Expected: struct {
|
||||||
|
MinTime int64
|
||||||
|
MaxTime int64
|
||||||
|
NumBlocks int
|
||||||
|
BlockDuration int64
|
||||||
|
Samples []backfillSample
|
||||||
|
}{
|
||||||
|
MinTime: 1624463088000,
|
||||||
|
MaxTime: 1629863088000,
|
||||||
|
NumBlocks: 2,
|
||||||
|
BlockDuration: int64(1458 * time.Hour / time.Millisecond),
|
||||||
|
Samples: []backfillSample{
|
||||||
|
{
|
||||||
|
Timestamp: 1624463088000,
|
||||||
|
Value: 1,
|
||||||
|
Labels: labels.FromStrings("__name__", "http_requests_total", "code", "200", "cluster_id", "123", "org_id", "999"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Timestamp: 1629503088000,
|
||||||
|
Value: 2,
|
||||||
|
Labels: labels.FromStrings("__name__", "http_requests_total", "code", "200", "cluster_id", "123", "org_id", "999"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Timestamp: 1629863088000,
|
||||||
|
Value: 3,
|
||||||
|
Labels: labels.FromStrings("__name__", "http_requests_total", "code", "200", "cluster_id", "123", "org_id", "999"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
ToParse: `# HELP rpc_duration_seconds A summary of the RPC duration in seconds.
|
ToParse: `# HELP rpc_duration_seconds A summary of the RPC duration in seconds.
|
||||||
# TYPE rpc_duration_seconds summary
|
# TYPE rpc_duration_seconds summary
|
||||||
|
@ -689,7 +733,7 @@ after_eof 1 2
|
||||||
|
|
||||||
outputDir := t.TempDir()
|
outputDir := t.TempDir()
|
||||||
|
|
||||||
err := backfill(test.MaxSamplesInAppender, []byte(test.ToParse), outputDir, false, false, test.MaxBlockDuration)
|
err := backfill(test.MaxSamplesInAppender, []byte(test.ToParse), outputDir, false, false, test.MaxBlockDuration, test.Labels)
|
||||||
|
|
||||||
if !test.IsOk {
|
if !test.IsOk {
|
||||||
require.Error(t, err, test.Description)
|
require.Error(t, err, test.Description)
|
||||||
|
|
|
@ -253,6 +253,7 @@ func main() {
|
||||||
importQuiet := importCmd.Flag("quiet", "Do not print created blocks.").Short('q').Bool()
|
importQuiet := importCmd.Flag("quiet", "Do not print created blocks.").Short('q').Bool()
|
||||||
maxBlockDuration := importCmd.Flag("max-block-duration", "Maximum duration created blocks may span. Anything less than 2h is ignored.").Hidden().PlaceHolder("<duration>").Duration()
|
maxBlockDuration := importCmd.Flag("max-block-duration", "Maximum duration created blocks may span. Anything less than 2h is ignored.").Hidden().PlaceHolder("<duration>").Duration()
|
||||||
openMetricsImportCmd := importCmd.Command("openmetrics", "Import samples from OpenMetrics input and produce TSDB blocks. Please refer to the storage docs for more details.")
|
openMetricsImportCmd := importCmd.Command("openmetrics", "Import samples from OpenMetrics input and produce TSDB blocks. Please refer to the storage docs for more details.")
|
||||||
|
openMetricsLabels := openMetricsImportCmd.Flag("label", "Label to attach to metrics. Can be specified multiple times. Example --label=label_name=label_value").StringMap()
|
||||||
importFilePath := openMetricsImportCmd.Arg("input file", "OpenMetrics file to read samples from.").Required().String()
|
importFilePath := openMetricsImportCmd.Arg("input file", "OpenMetrics file to read samples from.").Required().String()
|
||||||
importDBPath := openMetricsImportCmd.Arg("output directory", "Output directory for generated blocks.").Default(defaultDBPath).String()
|
importDBPath := openMetricsImportCmd.Arg("output directory", "Output directory for generated blocks.").Default(defaultDBPath).String()
|
||||||
importRulesCmd := importCmd.Command("rules", "Create blocks of data for new recording rules.")
|
importRulesCmd := importCmd.Command("rules", "Create blocks of data for new recording rules.")
|
||||||
|
@ -408,7 +409,7 @@ func main() {
|
||||||
os.Exit(checkErr(dumpSamples(ctx, *dumpOpenMetricsPath, *dumpOpenMetricsSandboxDirRoot, *dumpOpenMetricsMinTime, *dumpOpenMetricsMaxTime, *dumpOpenMetricsMatch, formatSeriesSetOpenMetrics)))
|
os.Exit(checkErr(dumpSamples(ctx, *dumpOpenMetricsPath, *dumpOpenMetricsSandboxDirRoot, *dumpOpenMetricsMinTime, *dumpOpenMetricsMaxTime, *dumpOpenMetricsMatch, formatSeriesSetOpenMetrics)))
|
||||||
// TODO(aSquare14): Work on adding support for custom block size.
|
// TODO(aSquare14): Work on adding support for custom block size.
|
||||||
case openMetricsImportCmd.FullCommand():
|
case openMetricsImportCmd.FullCommand():
|
||||||
os.Exit(backfillOpenMetrics(*importFilePath, *importDBPath, *importHumanReadable, *importQuiet, *maxBlockDuration))
|
os.Exit(backfillOpenMetrics(*importFilePath, *importDBPath, *importHumanReadable, *importQuiet, *maxBlockDuration, *openMetricsLabels))
|
||||||
|
|
||||||
case importRulesCmd.FullCommand():
|
case importRulesCmd.FullCommand():
|
||||||
os.Exit(checkErr(importRules(serverURL, httpRoundTripper, *importRulesStart, *importRulesEnd, *importRulesOutputDir, *importRulesEvalInterval, *maxBlockDuration, *importRulesFiles...)))
|
os.Exit(checkErr(importRules(serverURL, httpRoundTripper, *importRulesStart, *importRulesEnd, *importRulesOutputDir, *importRulesEvalInterval, *maxBlockDuration, *importRulesFiles...)))
|
||||||
|
|
|
@ -823,7 +823,7 @@ func checkErr(err error) int {
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func backfillOpenMetrics(path, outputDir string, humanReadable, quiet bool, maxBlockDuration time.Duration) int {
|
func backfillOpenMetrics(path, outputDir string, humanReadable, quiet bool, maxBlockDuration time.Duration, customLabels map[string]string) int {
|
||||||
inputFile, err := fileutil.OpenMmapFile(path)
|
inputFile, err := fileutil.OpenMmapFile(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return checkErr(err)
|
return checkErr(err)
|
||||||
|
@ -834,7 +834,7 @@ func backfillOpenMetrics(path, outputDir string, humanReadable, quiet bool, maxB
|
||||||
return checkErr(fmt.Errorf("create output dir: %w", err))
|
return checkErr(fmt.Errorf("create output dir: %w", err))
|
||||||
}
|
}
|
||||||
|
|
||||||
return checkErr(backfill(5000, inputFile.Bytes(), outputDir, humanReadable, quiet, maxBlockDuration))
|
return checkErr(backfill(5000, inputFile.Bytes(), outputDir, humanReadable, quiet, maxBlockDuration, customLabels))
|
||||||
}
|
}
|
||||||
|
|
||||||
func displayHistogram(dataType string, datas []int, total int) {
|
func displayHistogram(dataType string, datas []int, total int) {
|
||||||
|
|
|
@ -186,7 +186,7 @@ func TestTSDBDumpOpenMetricsRoundTrip(t *testing.T) {
|
||||||
|
|
||||||
dbDir := t.TempDir()
|
dbDir := t.TempDir()
|
||||||
// Import samples from OM format
|
// Import samples from OM format
|
||||||
err = backfill(5000, initialMetrics, dbDir, false, false, 2*time.Hour)
|
err = backfill(5000, initialMetrics, dbDir, false, false, 2*time.Hour, map[string]string{})
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
db, err := tsdb.Open(dbDir, nil, nil, tsdb.DefaultOptions(), nil)
|
db, err := tsdb.Open(dbDir, nil, nil, tsdb.DefaultOptions(), nil)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
|
@ -221,6 +221,7 @@ var (
|
||||||
// DefaultRemoteReadConfig is the default remote read configuration.
|
// DefaultRemoteReadConfig is the default remote read configuration.
|
||||||
DefaultRemoteReadConfig = RemoteReadConfig{
|
DefaultRemoteReadConfig = RemoteReadConfig{
|
||||||
RemoteTimeout: model.Duration(1 * time.Minute),
|
RemoteTimeout: model.Duration(1 * time.Minute),
|
||||||
|
ChunkedReadLimit: DefaultChunkedReadLimit,
|
||||||
HTTPClientConfig: config.DefaultHTTPClientConfig,
|
HTTPClientConfig: config.DefaultHTTPClientConfig,
|
||||||
FilterExternalLabels: true,
|
FilterExternalLabels: true,
|
||||||
}
|
}
|
||||||
|
@ -1279,13 +1280,20 @@ type MetadataConfig struct {
|
||||||
MaxSamplesPerSend int `yaml:"max_samples_per_send,omitempty"`
|
MaxSamplesPerSend int `yaml:"max_samples_per_send,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
// DefaultChunkedReadLimit is the default value for the maximum size of the protobuf frame client allows.
|
||||||
|
// 50MB is the default. This is equivalent to ~100k full XOR chunks and average labelset.
|
||||||
|
DefaultChunkedReadLimit = 5e+7
|
||||||
|
)
|
||||||
|
|
||||||
// RemoteReadConfig is the configuration for reading from remote storage.
|
// RemoteReadConfig is the configuration for reading from remote storage.
|
||||||
type RemoteReadConfig struct {
|
type RemoteReadConfig struct {
|
||||||
URL *config.URL `yaml:"url"`
|
URL *config.URL `yaml:"url"`
|
||||||
RemoteTimeout model.Duration `yaml:"remote_timeout,omitempty"`
|
RemoteTimeout model.Duration `yaml:"remote_timeout,omitempty"`
|
||||||
Headers map[string]string `yaml:"headers,omitempty"`
|
ChunkedReadLimit uint64 `yaml:"chunked_read_limit,omitempty"`
|
||||||
ReadRecent bool `yaml:"read_recent,omitempty"`
|
Headers map[string]string `yaml:"headers,omitempty"`
|
||||||
Name string `yaml:"name,omitempty"`
|
ReadRecent bool `yaml:"read_recent,omitempty"`
|
||||||
|
Name string `yaml:"name,omitempty"`
|
||||||
|
|
||||||
// We cannot do proper Go type embedding below as the parser will then parse
|
// We cannot do proper Go type embedding below as the parser will then parse
|
||||||
// values arbitrarily into the overflow maps of further-down types.
|
// values arbitrarily into the overflow maps of further-down types.
|
||||||
|
|
|
@ -165,10 +165,11 @@ var expectedConf = &Config{
|
||||||
|
|
||||||
RemoteReadConfigs: []*RemoteReadConfig{
|
RemoteReadConfigs: []*RemoteReadConfig{
|
||||||
{
|
{
|
||||||
URL: mustParseURL("http://remote1/read"),
|
URL: mustParseURL("http://remote1/read"),
|
||||||
RemoteTimeout: model.Duration(1 * time.Minute),
|
RemoteTimeout: model.Duration(1 * time.Minute),
|
||||||
ReadRecent: true,
|
ChunkedReadLimit: DefaultChunkedReadLimit,
|
||||||
Name: "default",
|
ReadRecent: true,
|
||||||
|
Name: "default",
|
||||||
HTTPClientConfig: config.HTTPClientConfig{
|
HTTPClientConfig: config.HTTPClientConfig{
|
||||||
FollowRedirects: true,
|
FollowRedirects: true,
|
||||||
EnableHTTP2: false,
|
EnableHTTP2: false,
|
||||||
|
@ -178,6 +179,7 @@ var expectedConf = &Config{
|
||||||
{
|
{
|
||||||
URL: mustParseURL("http://remote3/read"),
|
URL: mustParseURL("http://remote3/read"),
|
||||||
RemoteTimeout: model.Duration(1 * time.Minute),
|
RemoteTimeout: model.Duration(1 * time.Minute),
|
||||||
|
ChunkedReadLimit: DefaultChunkedReadLimit,
|
||||||
ReadRecent: false,
|
ReadRecent: false,
|
||||||
Name: "read_special",
|
Name: "read_special",
|
||||||
RequiredMatchers: model.LabelSet{"job": "special"},
|
RequiredMatchers: model.LabelSet{"job": "special"},
|
||||||
|
|
|
@ -212,9 +212,7 @@ func (m *Manager) ApplyConfig(cfg map[string]Configs) error {
|
||||||
m.metrics.FailedConfigs.Set(float64(failedCount))
|
m.metrics.FailedConfigs.Set(float64(failedCount))
|
||||||
|
|
||||||
var (
|
var (
|
||||||
wg sync.WaitGroup
|
wg sync.WaitGroup
|
||||||
// keep shows if we keep any providers after reload.
|
|
||||||
keep bool
|
|
||||||
newProviders []*Provider
|
newProviders []*Provider
|
||||||
)
|
)
|
||||||
for _, prov := range m.providers {
|
for _, prov := range m.providers {
|
||||||
|
@ -228,13 +226,12 @@ func (m *Manager) ApplyConfig(cfg map[string]Configs) error {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
newProviders = append(newProviders, prov)
|
newProviders = append(newProviders, prov)
|
||||||
// refTargets keeps reference targets used to populate new subs' targets
|
// refTargets keeps reference targets used to populate new subs' targets as they should be the same.
|
||||||
var refTargets map[string]*targetgroup.Group
|
var refTargets map[string]*targetgroup.Group
|
||||||
prov.mu.Lock()
|
prov.mu.Lock()
|
||||||
|
|
||||||
m.targetsMtx.Lock()
|
m.targetsMtx.Lock()
|
||||||
for s := range prov.subs {
|
for s := range prov.subs {
|
||||||
keep = true
|
|
||||||
refTargets = m.targets[poolKey{s, prov.name}]
|
refTargets = m.targets[poolKey{s, prov.name}]
|
||||||
// Remove obsolete subs' targets.
|
// Remove obsolete subs' targets.
|
||||||
if _, ok := prov.newSubs[s]; !ok {
|
if _, ok := prov.newSubs[s]; !ok {
|
||||||
|
@ -267,7 +264,9 @@ func (m *Manager) ApplyConfig(cfg map[string]Configs) error {
|
||||||
// While startProvider does pull the trigger, it may take some time to do so, therefore
|
// While startProvider does pull the trigger, it may take some time to do so, therefore
|
||||||
// we pull the trigger as soon as possible so that downstream managers can populate their state.
|
// we pull the trigger as soon as possible so that downstream managers can populate their state.
|
||||||
// See https://github.com/prometheus/prometheus/pull/8639 for details.
|
// See https://github.com/prometheus/prometheus/pull/8639 for details.
|
||||||
if keep {
|
// This also helps making the downstream managers drop stale targets as soon as possible.
|
||||||
|
// See https://github.com/prometheus/prometheus/pull/13147 for details.
|
||||||
|
if len(m.providers) > 0 {
|
||||||
select {
|
select {
|
||||||
case m.triggerSend <- struct{}{}:
|
case m.triggerSend <- struct{}{}:
|
||||||
default:
|
default:
|
||||||
|
@ -288,7 +287,9 @@ func (m *Manager) StartCustomProvider(ctx context.Context, name string, worker D
|
||||||
name: {},
|
name: {},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
m.mtx.Lock()
|
||||||
m.providers = append(m.providers, p)
|
m.providers = append(m.providers, p)
|
||||||
|
m.mtx.Unlock()
|
||||||
m.startProvider(ctx, p)
|
m.startProvider(ctx, p)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -403,19 +404,33 @@ func (m *Manager) allGroups() map[string][]*targetgroup.Group {
|
||||||
tSets := map[string][]*targetgroup.Group{}
|
tSets := map[string][]*targetgroup.Group{}
|
||||||
n := map[string]int{}
|
n := map[string]int{}
|
||||||
|
|
||||||
|
m.mtx.RLock()
|
||||||
m.targetsMtx.Lock()
|
m.targetsMtx.Lock()
|
||||||
defer m.targetsMtx.Unlock()
|
for _, p := range m.providers {
|
||||||
for pkey, tsets := range m.targets {
|
p.mu.RLock()
|
||||||
for _, tg := range tsets {
|
for s := range p.subs {
|
||||||
// Even if the target group 'tg' is empty we still need to send it to the 'Scrape manager'
|
// Send empty lists for subs without any targets to make sure old stale targets are dropped by consumers.
|
||||||
// to signal that it needs to stop all scrape loops for this target set.
|
// See: https://github.com/prometheus/prometheus/issues/12858 for details.
|
||||||
tSets[pkey.setName] = append(tSets[pkey.setName], tg)
|
if _, ok := tSets[s]; !ok {
|
||||||
n[pkey.setName] += len(tg.Targets)
|
tSets[s] = []*targetgroup.Group{}
|
||||||
|
n[s] = 0
|
||||||
|
}
|
||||||
|
if tsets, ok := m.targets[poolKey{s, p.name}]; ok {
|
||||||
|
for _, tg := range tsets {
|
||||||
|
tSets[s] = append(tSets[s], tg)
|
||||||
|
n[s] += len(tg.Targets)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
p.mu.RUnlock()
|
||||||
}
|
}
|
||||||
|
m.targetsMtx.Unlock()
|
||||||
|
m.mtx.RUnlock()
|
||||||
|
|
||||||
for setName, v := range n {
|
for setName, v := range n {
|
||||||
m.metrics.DiscoveredTargets.WithLabelValues(setName).Set(float64(v))
|
m.metrics.DiscoveredTargets.WithLabelValues(setName).Set(float64(v))
|
||||||
}
|
}
|
||||||
|
|
||||||
return tSets
|
return tSets
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -939,11 +939,13 @@ func TestTargetSetTargetGroupsPresentOnConfigChange(t *testing.T) {
|
||||||
discoveryManager.ApplyConfig(c)
|
discoveryManager.ApplyConfig(c)
|
||||||
|
|
||||||
// Original targets should be present as soon as possible.
|
// Original targets should be present as soon as possible.
|
||||||
|
// An empty list should be sent for prometheus2 to drop any stale targets
|
||||||
syncedTargets = <-discoveryManager.SyncCh()
|
syncedTargets = <-discoveryManager.SyncCh()
|
||||||
mu.Unlock()
|
mu.Unlock()
|
||||||
require.Len(t, syncedTargets, 1)
|
require.Len(t, syncedTargets, 2)
|
||||||
verifySyncedPresence(t, syncedTargets, "prometheus", "{__address__=\"foo:9090\"}", true)
|
verifySyncedPresence(t, syncedTargets, "prometheus", "{__address__=\"foo:9090\"}", true)
|
||||||
require.Len(t, syncedTargets["prometheus"], 1)
|
require.Len(t, syncedTargets["prometheus"], 1)
|
||||||
|
require.Empty(t, syncedTargets["prometheus2"])
|
||||||
|
|
||||||
// prometheus2 configs should be ready on second sync.
|
// prometheus2 configs should be ready on second sync.
|
||||||
syncedTargets = <-discoveryManager.SyncCh()
|
syncedTargets = <-discoveryManager.SyncCh()
|
||||||
|
@ -1275,6 +1277,7 @@ func TestCoordinationWithReceiver(t *testing.T) {
|
||||||
Targets: []model.LabelSet{{"__instance__": "1"}},
|
Targets: []model.LabelSet{{"__instance__": "1"}},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
"mock1": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|
|
@ -19,6 +19,7 @@ import (
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -251,28 +252,26 @@ func (d *DockerDiscovery) refresh(ctx context.Context) ([]*targetgroup.Group, er
|
||||||
}
|
}
|
||||||
|
|
||||||
if d.matchFirstNetwork && len(networks) > 1 {
|
if d.matchFirstNetwork && len(networks) > 1 {
|
||||||
// Match user defined network
|
// Sort networks by name and take first non-nil network.
|
||||||
if containerNetworkMode.IsUserDefined() {
|
keys := make([]string, 0, len(networks))
|
||||||
networkMode := string(containerNetworkMode)
|
for k, n := range networks {
|
||||||
networks = map[string]*network.EndpointSettings{networkMode: networks[networkMode]}
|
if n != nil {
|
||||||
} else {
|
keys = append(keys, k)
|
||||||
// Get first network if container network mode has "none" value.
|
|
||||||
// This case appears under certain condition:
|
|
||||||
// 1. Container created with network set to "--net=none".
|
|
||||||
// 2. Disconnect network "none".
|
|
||||||
// 3. Reconnect network with user defined networks.
|
|
||||||
var first string
|
|
||||||
for k, n := range networks {
|
|
||||||
if n != nil {
|
|
||||||
first = k
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
networks = map[string]*network.EndpointSettings{first: networks[first]}
|
}
|
||||||
|
if len(keys) > 0 {
|
||||||
|
sort.Strings(keys)
|
||||||
|
firstNetworkMode := keys[0]
|
||||||
|
firstNetwork := networks[firstNetworkMode]
|
||||||
|
networks = map[string]*network.EndpointSettings{firstNetworkMode: firstNetwork}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, n := range networks {
|
for _, n := range networks {
|
||||||
|
if n == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
var added bool
|
var added bool
|
||||||
|
|
||||||
for _, p := range c.Ports {
|
for _, p := range c.Ports {
|
||||||
|
|
|
@ -60,9 +60,9 @@ host: %s
|
||||||
tg := tgs[0]
|
tg := tgs[0]
|
||||||
require.NotNil(t, tg)
|
require.NotNil(t, tg)
|
||||||
require.NotNil(t, tg.Targets)
|
require.NotNil(t, tg.Targets)
|
||||||
require.Len(t, tg.Targets, 6)
|
require.Len(t, tg.Targets, 8)
|
||||||
|
|
||||||
for i, lbls := range []model.LabelSet{
|
expected := []model.LabelSet{
|
||||||
{
|
{
|
||||||
"__address__": "172.19.0.2:9100",
|
"__address__": "172.19.0.2:9100",
|
||||||
"__meta_docker_container_id": "c301b928faceb1a18fe379f6bc178727ef920bb30b0f9b8592b32b36255a0eca",
|
"__meta_docker_container_id": "c301b928faceb1a18fe379f6bc178727ef920bb30b0f9b8592b32b36255a0eca",
|
||||||
|
@ -163,7 +163,43 @@ host: %s
|
||||||
"__meta_docker_network_scope": "local",
|
"__meta_docker_network_scope": "local",
|
||||||
"__meta_docker_port_private": "9104",
|
"__meta_docker_port_private": "9104",
|
||||||
},
|
},
|
||||||
} {
|
{
|
||||||
|
"__address__": "172.20.0.3:3306",
|
||||||
|
"__meta_docker_container_id": "f84b2a0cfaa58d9e70b0657e2b3c6f44f0e973de4163a871299b4acf127b224f",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_project": "dockersd",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_service": "mysql",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_version": "2.2.2",
|
||||||
|
"__meta_docker_container_name": "/dockersd_multi_networks",
|
||||||
|
"__meta_docker_container_network_mode": "dockersd_private_none",
|
||||||
|
"__meta_docker_network_id": "e804771e55254a360fdb70dfdd78d3610fdde231b14ef2f837a00ac1eeb9e601",
|
||||||
|
"__meta_docker_network_ingress": "false",
|
||||||
|
"__meta_docker_network_internal": "false",
|
||||||
|
"__meta_docker_network_ip": "172.20.0.3",
|
||||||
|
"__meta_docker_network_name": "dockersd_private",
|
||||||
|
"__meta_docker_network_scope": "local",
|
||||||
|
"__meta_docker_port_private": "3306",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"__address__": "172.20.0.3:33060",
|
||||||
|
"__meta_docker_container_id": "f84b2a0cfaa58d9e70b0657e2b3c6f44f0e973de4163a871299b4acf127b224f",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_project": "dockersd",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_service": "mysql",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_version": "2.2.2",
|
||||||
|
"__meta_docker_container_name": "/dockersd_multi_networks",
|
||||||
|
"__meta_docker_container_network_mode": "dockersd_private_none",
|
||||||
|
"__meta_docker_network_id": "e804771e55254a360fdb70dfdd78d3610fdde231b14ef2f837a00ac1eeb9e601",
|
||||||
|
"__meta_docker_network_ingress": "false",
|
||||||
|
"__meta_docker_network_internal": "false",
|
||||||
|
"__meta_docker_network_ip": "172.20.0.3",
|
||||||
|
"__meta_docker_network_name": "dockersd_private",
|
||||||
|
"__meta_docker_network_scope": "local",
|
||||||
|
"__meta_docker_port_private": "33060",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
sortFunc(expected)
|
||||||
|
sortFunc(tg.Targets)
|
||||||
|
|
||||||
|
for i, lbls := range expected {
|
||||||
t.Run(fmt.Sprintf("item %d", i), func(t *testing.T) {
|
t.Run(fmt.Sprintf("item %d", i), func(t *testing.T) {
|
||||||
require.Equal(t, lbls, tg.Targets[i])
|
require.Equal(t, lbls, tg.Targets[i])
|
||||||
})
|
})
|
||||||
|
@ -202,13 +238,8 @@ host: %s
|
||||||
tg := tgs[0]
|
tg := tgs[0]
|
||||||
require.NotNil(t, tg)
|
require.NotNil(t, tg)
|
||||||
require.NotNil(t, tg.Targets)
|
require.NotNil(t, tg.Targets)
|
||||||
require.Len(t, tg.Targets, 9)
|
require.Len(t, tg.Targets, 13)
|
||||||
|
|
||||||
sortFunc := func(labelSets []model.LabelSet) {
|
|
||||||
sort.Slice(labelSets, func(i, j int) bool {
|
|
||||||
return labelSets[i]["__address__"] < labelSets[j]["__address__"]
|
|
||||||
})
|
|
||||||
}
|
|
||||||
expected := []model.LabelSet{
|
expected := []model.LabelSet{
|
||||||
{
|
{
|
||||||
"__address__": "172.19.0.2:9100",
|
"__address__": "172.19.0.2:9100",
|
||||||
|
@ -359,6 +390,70 @@ host: %s
|
||||||
"__meta_docker_network_scope": "local",
|
"__meta_docker_network_scope": "local",
|
||||||
"__meta_docker_port_private": "9104",
|
"__meta_docker_port_private": "9104",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"__address__": "172.20.0.3:3306",
|
||||||
|
"__meta_docker_container_id": "f84b2a0cfaa58d9e70b0657e2b3c6f44f0e973de4163a871299b4acf127b224f",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_project": "dockersd",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_service": "mysql",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_version": "2.2.2",
|
||||||
|
"__meta_docker_container_name": "/dockersd_multi_networks",
|
||||||
|
"__meta_docker_container_network_mode": "dockersd_private_none",
|
||||||
|
"__meta_docker_network_id": "e804771e55254a360fdb70dfdd78d3610fdde231b14ef2f837a00ac1eeb9e601",
|
||||||
|
"__meta_docker_network_ingress": "false",
|
||||||
|
"__meta_docker_network_internal": "false",
|
||||||
|
"__meta_docker_network_ip": "172.20.0.3",
|
||||||
|
"__meta_docker_network_name": "dockersd_private",
|
||||||
|
"__meta_docker_network_scope": "local",
|
||||||
|
"__meta_docker_port_private": "3306",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"__address__": "172.20.0.3:33060",
|
||||||
|
"__meta_docker_container_id": "f84b2a0cfaa58d9e70b0657e2b3c6f44f0e973de4163a871299b4acf127b224f",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_project": "dockersd",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_service": "mysql",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_version": "2.2.2",
|
||||||
|
"__meta_docker_container_name": "/dockersd_multi_networks",
|
||||||
|
"__meta_docker_container_network_mode": "dockersd_private_none",
|
||||||
|
"__meta_docker_network_id": "e804771e55254a360fdb70dfdd78d3610fdde231b14ef2f837a00ac1eeb9e601",
|
||||||
|
"__meta_docker_network_ingress": "false",
|
||||||
|
"__meta_docker_network_internal": "false",
|
||||||
|
"__meta_docker_network_ip": "172.20.0.3",
|
||||||
|
"__meta_docker_network_name": "dockersd_private",
|
||||||
|
"__meta_docker_network_scope": "local",
|
||||||
|
"__meta_docker_port_private": "33060",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"__address__": "172.21.0.3:3306",
|
||||||
|
"__meta_docker_container_id": "f84b2a0cfaa58d9e70b0657e2b3c6f44f0e973de4163a871299b4acf127b224f",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_project": "dockersd",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_service": "mysql",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_version": "2.2.2",
|
||||||
|
"__meta_docker_container_name": "/dockersd_multi_networks",
|
||||||
|
"__meta_docker_container_network_mode": "dockersd_private_none",
|
||||||
|
"__meta_docker_network_id": "bfcf66a6b64f7d518f009e34290dc3f3c66a08164257ad1afc3bd31d75f656e8",
|
||||||
|
"__meta_docker_network_ingress": "false",
|
||||||
|
"__meta_docker_network_internal": "false",
|
||||||
|
"__meta_docker_network_ip": "172.21.0.3",
|
||||||
|
"__meta_docker_network_name": "dockersd_private1",
|
||||||
|
"__meta_docker_network_scope": "local",
|
||||||
|
"__meta_docker_port_private": "3306",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"__address__": "172.21.0.3:33060",
|
||||||
|
"__meta_docker_container_id": "f84b2a0cfaa58d9e70b0657e2b3c6f44f0e973de4163a871299b4acf127b224f",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_project": "dockersd",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_service": "mysql",
|
||||||
|
"__meta_docker_container_label_com_docker_compose_version": "2.2.2",
|
||||||
|
"__meta_docker_container_name": "/dockersd_multi_networks",
|
||||||
|
"__meta_docker_container_network_mode": "dockersd_private_none",
|
||||||
|
"__meta_docker_network_id": "bfcf66a6b64f7d518f009e34290dc3f3c66a08164257ad1afc3bd31d75f656e8",
|
||||||
|
"__meta_docker_network_ingress": "false",
|
||||||
|
"__meta_docker_network_internal": "false",
|
||||||
|
"__meta_docker_network_ip": "172.21.0.3",
|
||||||
|
"__meta_docker_network_name": "dockersd_private1",
|
||||||
|
"__meta_docker_network_scope": "local",
|
||||||
|
"__meta_docker_port_private": "33060",
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
sortFunc(expected)
|
sortFunc(expected)
|
||||||
|
@ -370,3 +465,9 @@ host: %s
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func sortFunc(labelSets []model.LabelSet) {
|
||||||
|
sort.Slice(labelSets, func(i, j int) bool {
|
||||||
|
return labelSets[i]["__address__"] < labelSets[j]["__address__"]
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
|
@ -228,5 +228,74 @@
|
||||||
"Networks": {}
|
"Networks": {}
|
||||||
},
|
},
|
||||||
"Mounts": []
|
"Mounts": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Id": "f84b2a0cfaa58d9e70b0657e2b3c6f44f0e973de4163a871299b4acf127b224f",
|
||||||
|
"Names": [
|
||||||
|
"/dockersd_multi_networks"
|
||||||
|
],
|
||||||
|
"Image": "mysql:5.7.29",
|
||||||
|
"ImageID": "sha256:16ae2f4625ba63a250462bedeece422e741de9f0caf3b1d89fd5b257aca80cd1",
|
||||||
|
"Command": "mysqld",
|
||||||
|
"Created": 1616273136,
|
||||||
|
"Ports": [
|
||||||
|
{
|
||||||
|
"PrivatePort": 3306,
|
||||||
|
"Type": "tcp"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"PrivatePort": 33060,
|
||||||
|
"Type": "tcp"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Labels": {
|
||||||
|
"com.docker.compose.project": "dockersd",
|
||||||
|
"com.docker.compose.service": "mysql",
|
||||||
|
"com.docker.compose.version": "2.2.2"
|
||||||
|
},
|
||||||
|
"State": "running",
|
||||||
|
"Status": "Up 40 seconds",
|
||||||
|
"HostConfig": {
|
||||||
|
"NetworkMode": "dockersd_private_none"
|
||||||
|
},
|
||||||
|
"NetworkSettings": {
|
||||||
|
"Networks": {
|
||||||
|
"dockersd_private": {
|
||||||
|
"IPAMConfig": null,
|
||||||
|
"Links": null,
|
||||||
|
"Aliases": null,
|
||||||
|
"NetworkID": "e804771e55254a360fdb70dfdd78d3610fdde231b14ef2f837a00ac1eeb9e601",
|
||||||
|
"EndpointID": "972d6807997369605ace863af58de6cb90c787a5bf2ffc4105662d393ae539b7",
|
||||||
|
"Gateway": "172.20.0.1",
|
||||||
|
"IPAddress": "172.20.0.3",
|
||||||
|
"IPPrefixLen": 16,
|
||||||
|
"IPv6Gateway": "",
|
||||||
|
"GlobalIPv6Address": "",
|
||||||
|
"GlobalIPv6PrefixLen": 0,
|
||||||
|
"MacAddress": "02:42:ac:14:00:02",
|
||||||
|
"DriverOpts": null
|
||||||
|
},
|
||||||
|
"dockersd_private1": {
|
||||||
|
"IPAMConfig": {},
|
||||||
|
"Links": null,
|
||||||
|
"Aliases": [
|
||||||
|
"mysql",
|
||||||
|
"mysql",
|
||||||
|
"f9ade4b83199"
|
||||||
|
],
|
||||||
|
"NetworkID": "bfcf66a6b64f7d518f009e34290dc3f3c66a08164257ad1afc3bd31d75f656e8",
|
||||||
|
"EndpointID": "91a98405344ee1cb7d977cafabe634837876651544b32da20a5e0155868e6f5f",
|
||||||
|
"Gateway": "172.21.0.1",
|
||||||
|
"IPAddress": "172.21.0.3",
|
||||||
|
"IPPrefixLen": 24,
|
||||||
|
"IPv6Gateway": "",
|
||||||
|
"GlobalIPv6Address": "",
|
||||||
|
"GlobalIPv6PrefixLen": 0,
|
||||||
|
"MacAddress": "02:42:ac:15:00:02",
|
||||||
|
"DriverOpts": null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Mounts": []
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|
|
@ -57,7 +57,7 @@ The Prometheus monitoring server
|
||||||
| <code class="text-nowrap">--query.max-concurrency</code> | Maximum number of queries executed concurrently. Use with server mode only. | `20` |
|
| <code class="text-nowrap">--query.max-concurrency</code> | Maximum number of queries executed concurrently. Use with server mode only. | `20` |
|
||||||
| <code class="text-nowrap">--query.max-samples</code> | Maximum number of samples a single query can load into memory. Note that queries will fail if they try to load more samples than this into memory, so this also limits the number of samples a query can return. Use with server mode only. | `50000000` |
|
| <code class="text-nowrap">--query.max-samples</code> | Maximum number of samples a single query can load into memory. Note that queries will fail if they try to load more samples than this into memory, so this also limits the number of samples a query can return. Use with server mode only. | `50000000` |
|
||||||
| <code class="text-nowrap">--scrape.name-escaping-scheme</code> | Method for escaping legacy invalid names when sending to Prometheus that does not support UTF-8. Can be one of "values", "underscores", or "dots". | `values` |
|
| <code class="text-nowrap">--scrape.name-escaping-scheme</code> | Method for escaping legacy invalid names when sending to Prometheus that does not support UTF-8. Can be one of "values", "underscores", or "dots". | `values` |
|
||||||
| <code class="text-nowrap">--enable-feature</code> <code class="text-nowrap">...<code class="text-nowrap"> | Comma separated feature names to enable. Valid options: agent, auto-gomemlimit, exemplar-storage, expand-external-labels, memory-snapshot-on-shutdown, promql-per-step-stats, promql-experimental-functions, remote-write-receiver (DEPRECATED), extra-scrape-metrics, new-service-discovery-manager, auto-gomaxprocs, no-default-scrape-port, native-histograms, otlp-write-receiver, created-timestamp-zero-ingestion, concurrent-rule-eval, delayed-compaction, utf8-names. See https://prometheus.io/docs/prometheus/latest/feature_flags/ for more details. | |
|
| <code class="text-nowrap">--enable-feature</code> <code class="text-nowrap">...<code class="text-nowrap"> | Comma separated feature names to enable. Valid options: agent, auto-gomaxprocs, auto-gomemlimit, concurrent-rule-eval, created-timestamp-zero-ingestion, delayed-compaction, exemplar-storage, expand-external-labels, extra-scrape-metrics, memory-snapshot-on-shutdown, native-histograms, new-service-discovery-manager, no-default-scrape-port, otlp-write-receiver, promql-experimental-functions, promql-delayed-name-removal, promql-per-step-stats, remote-write-receiver (DEPRECATED), utf8-names. See https://prometheus.io/docs/prometheus/latest/feature_flags/ for more details. | |
|
||||||
| <code class="text-nowrap">--log.level</code> | Only log messages with the given severity or above. One of: [debug, info, warn, error] | `info` |
|
| <code class="text-nowrap">--log.level</code> | Only log messages with the given severity or above. One of: [debug, info, warn, error] | `info` |
|
||||||
| <code class="text-nowrap">--log.format</code> | Output format of log messages. One of: [logfmt, json] | `logfmt` |
|
| <code class="text-nowrap">--log.format</code> | Output format of log messages. One of: [logfmt, json] | `logfmt` |
|
||||||
|
|
||||||
|
|
|
@ -641,6 +641,15 @@ Import samples from OpenMetrics input and produce TSDB blocks. Please refer to t
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
###### Flags
|
||||||
|
|
||||||
|
| Flag | Description |
|
||||||
|
| --- | --- |
|
||||||
|
| <code class="text-nowrap">--label</code> | Label to attach to metrics. Can be specified multiple times. Example --label=label_name=label_value |
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
###### Arguments
|
###### Arguments
|
||||||
|
|
||||||
| Argument | Description | Default | Required |
|
| Argument | Description | Default | Required |
|
||||||
|
|
|
@ -957,7 +957,9 @@ tls_config:
|
||||||
# The host to use if the container is in host networking mode.
|
# The host to use if the container is in host networking mode.
|
||||||
[ host_networking_host: <string> | default = "localhost" ]
|
[ host_networking_host: <string> | default = "localhost" ]
|
||||||
|
|
||||||
# Match the first network if the container has multiple networks defined, thus avoiding collecting duplicate targets.
|
# Sort all non-nil networks in ascending order based on network name and
|
||||||
|
# get the first network if the container has multiple networks defined,
|
||||||
|
# thus avoiding collecting duplicate targets.
|
||||||
[ match_first_network: <boolean> | default = true ]
|
[ match_first_network: <boolean> | default = true ]
|
||||||
|
|
||||||
# Optional filters to limit the discovery process to a subset of available
|
# Optional filters to limit the discovery process to a subset of available
|
||||||
|
|
|
@ -250,6 +250,14 @@ Note that during this delay, the Head continues its usual operations, which incl
|
||||||
|
|
||||||
Despite the delay in compaction, the blocks produced are time-aligned in the same manner as they would be if the delay was not in place.
|
Despite the delay in compaction, the blocks produced are time-aligned in the same manner as they would be if the delay was not in place.
|
||||||
|
|
||||||
|
## Delay __name__ label removal for PromQL engine
|
||||||
|
|
||||||
|
`--enable-feature=promql-delayed-name-removal`
|
||||||
|
|
||||||
|
When enabled, Prometheus will change the way in which the `__name__` label is removed from PromQL query results (for functions and expressions for which this is necessary). Specifically, it will delay the removal to the last step of the query evaluation, instead of every time an expression or function creating derived metrics is evaluated.
|
||||||
|
|
||||||
|
This allows optionally preserving the `__name__` label via the `label_replace` and `label_join` functions, and helps prevent the "vector cannot contain metrics with the same labelset" error, which can happen when applying a regex-matcher to the `__name__` label.
|
||||||
|
|
||||||
## UTF-8 Name Support
|
## UTF-8 Name Support
|
||||||
|
|
||||||
`--enable-feature=utf8-names`
|
`--enable-feature=utf8-names`
|
||||||
|
|
4
go.mod
4
go.mod
|
@ -52,9 +52,9 @@ require (
|
||||||
github.com/oklog/ulid v1.3.1
|
github.com/oklog/ulid v1.3.1
|
||||||
github.com/ovh/go-ovh v1.6.0
|
github.com/ovh/go-ovh v1.6.0
|
||||||
github.com/prometheus/alertmanager v0.27.0
|
github.com/prometheus/alertmanager v0.27.0
|
||||||
github.com/prometheus/client_golang v1.19.1
|
github.com/prometheus/client_golang v1.20.0
|
||||||
github.com/prometheus/client_model v0.6.1
|
github.com/prometheus/client_model v0.6.1
|
||||||
github.com/prometheus/common v0.55.0
|
github.com/prometheus/common v0.56.0
|
||||||
github.com/prometheus/common/assets v0.2.0
|
github.com/prometheus/common/assets v0.2.0
|
||||||
github.com/prometheus/common/sigv4 v0.1.0
|
github.com/prometheus/common/sigv4 v0.1.0
|
||||||
github.com/prometheus/exporter-toolkit v0.11.0
|
github.com/prometheus/exporter-toolkit v0.11.0
|
||||||
|
|
8
go.sum
8
go.sum
|
@ -608,8 +608,8 @@ github.com/prometheus/client_golang v1.3.0/go.mod h1:hJaj2vgQTGQmVCsAACORcieXFeD
|
||||||
github.com/prometheus/client_golang v1.4.0/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU=
|
github.com/prometheus/client_golang v1.4.0/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU=
|
||||||
github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M=
|
github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M=
|
||||||
github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0=
|
github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0=
|
||||||
github.com/prometheus/client_golang v1.19.1 h1:wZWJDwK+NameRJuPGDhlnFgx8e8HN3XHQeLaYJFJBOE=
|
github.com/prometheus/client_golang v1.20.0 h1:jBzTZ7B099Rg24tny+qngoynol8LtVYlA2bqx3vEloI=
|
||||||
github.com/prometheus/client_golang v1.19.1/go.mod h1:mP78NwGzrVks5S2H6ab8+ZZGJLZUq1hoULYBAYBw1Ho=
|
github.com/prometheus/client_golang v1.20.0/go.mod h1:PIEt8X02hGcP8JWbeHyeZ53Y/jReSnHgO035n//V5WE=
|
||||||
github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
|
github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
|
||||||
github.com/prometheus/client_model v0.0.0-20190115171406-56726106282f/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
|
github.com/prometheus/client_model v0.0.0-20190115171406-56726106282f/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
|
||||||
github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
|
github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
|
||||||
|
@ -625,8 +625,8 @@ github.com/prometheus/common v0.9.1/go.mod h1:yhUN8i9wzaXS3w1O07YhxHEBxD+W35wd8b
|
||||||
github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo=
|
github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo=
|
||||||
github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc=
|
github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc=
|
||||||
github.com/prometheus/common v0.29.0/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls=
|
github.com/prometheus/common v0.29.0/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls=
|
||||||
github.com/prometheus/common v0.55.0 h1:KEi6DK7lXW/m7Ig5i47x0vRzuBsHuvJdi5ee6Y3G1dc=
|
github.com/prometheus/common v0.56.0 h1:UffReloqkBtvtQEYDg2s+uDPGRrJyC6vZWPGXf6OhPY=
|
||||||
github.com/prometheus/common v0.55.0/go.mod h1:2SECS4xJG1kd8XF9IcM1gMX6510RAEL65zxzNImwdc8=
|
github.com/prometheus/common v0.56.0/go.mod h1:7uRPFSUTbfZWsJ7MHY56sqt7hLQu3bxXHDnNhl8E9qI=
|
||||||
github.com/prometheus/common/assets v0.2.0 h1:0P5OrzoHrYBOSM1OigWL3mY8ZvV2N4zIE/5AahrSrfM=
|
github.com/prometheus/common/assets v0.2.0 h1:0P5OrzoHrYBOSM1OigWL3mY8ZvV2N4zIE/5AahrSrfM=
|
||||||
github.com/prometheus/common/assets v0.2.0/go.mod h1:D17UVUE12bHbim7HzwUvtqm6gwBEaDQ0F+hIGbFbccI=
|
github.com/prometheus/common/assets v0.2.0/go.mod h1:D17UVUE12bHbim7HzwUvtqm6gwBEaDQ0F+hIGbFbccI=
|
||||||
github.com/prometheus/common/sigv4 v0.1.0 h1:qoVebwtwwEhS85Czm2dSROY5fTo2PAPEVdDeppTwGX4=
|
github.com/prometheus/common/sigv4 v0.1.0 h1:qoVebwtwwEhS85Czm2dSROY5fTo2PAPEVdDeppTwGX4=
|
||||||
|
|
|
@ -95,12 +95,23 @@ func (ls *Labels) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
// IsValid checks if the metric name or label names are valid.
|
// IsValid checks if the metric name or label names are valid.
|
||||||
func (ls Labels) IsValid() bool {
|
func (ls Labels) IsValid(validationScheme model.ValidationScheme) bool {
|
||||||
err := ls.Validate(func(l Label) error {
|
err := ls.Validate(func(l Label) error {
|
||||||
if l.Name == model.MetricNameLabel && !model.IsValidMetricName(model.LabelValue(l.Value)) {
|
if l.Name == model.MetricNameLabel {
|
||||||
return strconv.ErrSyntax
|
// If the default validation scheme has been overridden with legacy mode,
|
||||||
|
// we need to call the special legacy validation checker.
|
||||||
|
if validationScheme == model.LegacyValidation && model.NameValidationScheme == model.UTF8Validation && !model.IsValidLegacyMetricName(string(model.LabelValue(l.Value))) {
|
||||||
|
return strconv.ErrSyntax
|
||||||
|
}
|
||||||
|
if !model.IsValidMetricName(model.LabelValue(l.Value)) {
|
||||||
|
return strconv.ErrSyntax
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if !model.LabelName(l.Name).IsValid() || !model.LabelValue(l.Value).IsValid() {
|
if validationScheme == model.LegacyValidation && model.NameValidationScheme == model.UTF8Validation {
|
||||||
|
if !model.LabelName(l.Name).IsValidLegacy() || !model.LabelValue(l.Value).IsValid() {
|
||||||
|
return strconv.ErrSyntax
|
||||||
|
}
|
||||||
|
} else if !model.LabelName(l.Name).IsValid() || !model.LabelValue(l.Value).IsValid() {
|
||||||
return strconv.ErrSyntax
|
return strconv.ErrSyntax
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
|
|
@ -21,6 +21,7 @@ import (
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/prometheus/common/model"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
"gopkg.in/yaml.v2"
|
"gopkg.in/yaml.v2"
|
||||||
)
|
)
|
||||||
|
@ -272,11 +273,86 @@ func TestLabels_IsValid(t *testing.T) {
|
||||||
},
|
},
|
||||||
} {
|
} {
|
||||||
t.Run("", func(t *testing.T) {
|
t.Run("", func(t *testing.T) {
|
||||||
require.Equal(t, test.expected, test.input.IsValid())
|
require.Equal(t, test.expected, test.input.IsValid(model.LegacyValidation))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestLabels_ValidationModes(t *testing.T) {
|
||||||
|
for _, test := range []struct {
|
||||||
|
input Labels
|
||||||
|
globalMode model.ValidationScheme
|
||||||
|
callMode model.ValidationScheme
|
||||||
|
expected bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
input: FromStrings(
|
||||||
|
"__name__", "test.metric",
|
||||||
|
"hostname", "localhost",
|
||||||
|
"job", "check",
|
||||||
|
),
|
||||||
|
globalMode: model.UTF8Validation,
|
||||||
|
callMode: model.UTF8Validation,
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: FromStrings(
|
||||||
|
"__name__", "test",
|
||||||
|
"\xc5 bad utf8", "localhost",
|
||||||
|
"job", "check",
|
||||||
|
),
|
||||||
|
globalMode: model.UTF8Validation,
|
||||||
|
callMode: model.UTF8Validation,
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
// Setting the common model to legacy validation and then trying to check for UTF-8 on a
|
||||||
|
// per-call basis is not supported.
|
||||||
|
input: FromStrings(
|
||||||
|
"__name__", "test.utf8.metric",
|
||||||
|
"hostname", "localhost",
|
||||||
|
"job", "check",
|
||||||
|
),
|
||||||
|
globalMode: model.LegacyValidation,
|
||||||
|
callMode: model.UTF8Validation,
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: FromStrings(
|
||||||
|
"__name__", "test",
|
||||||
|
"hostname", "localhost",
|
||||||
|
"job", "check",
|
||||||
|
),
|
||||||
|
globalMode: model.LegacyValidation,
|
||||||
|
callMode: model.LegacyValidation,
|
||||||
|
expected: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: FromStrings(
|
||||||
|
"__name__", "test.utf8.metric",
|
||||||
|
"hostname", "localhost",
|
||||||
|
"job", "check",
|
||||||
|
),
|
||||||
|
globalMode: model.UTF8Validation,
|
||||||
|
callMode: model.LegacyValidation,
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: FromStrings(
|
||||||
|
"__name__", "test",
|
||||||
|
"host.name", "localhost",
|
||||||
|
"job", "check",
|
||||||
|
),
|
||||||
|
globalMode: model.UTF8Validation,
|
||||||
|
callMode: model.LegacyValidation,
|
||||||
|
expected: false,
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
model.NameValidationScheme = test.globalMode
|
||||||
|
require.Equal(t, test.expected, test.input.IsValid(test.callMode))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestLabels_Equal(t *testing.T) {
|
func TestLabels_Equal(t *testing.T) {
|
||||||
labels := FromStrings(
|
labels := FromStrings(
|
||||||
"aaa", "111",
|
"aaa", "111",
|
||||||
|
|
|
@ -47,7 +47,7 @@ import (
|
||||||
// the re-arrangement work is actually causing problems (which has to be seen),
|
// the re-arrangement work is actually causing problems (which has to be seen),
|
||||||
// that expectation needs to be changed.
|
// that expectation needs to be changed.
|
||||||
type ProtobufParser struct {
|
type ProtobufParser struct {
|
||||||
in []byte // The intput to parse.
|
in []byte // The input to parse.
|
||||||
inPos int // Position within the input.
|
inPos int // Position within the input.
|
||||||
metricPos int // Position within Metric slice.
|
metricPos int // Position within Metric slice.
|
||||||
// fieldPos is the position within a Summary or (legacy) Histogram. -2
|
// fieldPos is the position within a Summary or (legacy) Histogram. -2
|
||||||
|
@ -71,7 +71,7 @@ type ProtobufParser struct {
|
||||||
|
|
||||||
mf *dto.MetricFamily
|
mf *dto.MetricFamily
|
||||||
|
|
||||||
// Wether to also parse a classic histogram that is also present as a
|
// Whether to also parse a classic histogram that is also present as a
|
||||||
// native histogram.
|
// native histogram.
|
||||||
parseClassicHistograms bool
|
parseClassicHistograms bool
|
||||||
|
|
||||||
|
|
124
promql/engine.go
124
promql/engine.go
|
@ -313,6 +313,11 @@ type EngineOpts struct {
|
||||||
|
|
||||||
// EnablePerStepStats if true allows for per-step stats to be computed on request. Disabled otherwise.
|
// EnablePerStepStats if true allows for per-step stats to be computed on request. Disabled otherwise.
|
||||||
EnablePerStepStats bool
|
EnablePerStepStats bool
|
||||||
|
|
||||||
|
// EnableDelayedNameRemoval delays the removal of the __name__ label to the last step of the query evaluation.
|
||||||
|
// This is useful in certain scenarios where the __name__ label must be preserved or where applying a
|
||||||
|
// regex-matcher to the __name__ label may otherwise lead to duplicate labelset errors.
|
||||||
|
EnableDelayedNameRemoval bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Engine handles the lifetime of queries from beginning to end.
|
// Engine handles the lifetime of queries from beginning to end.
|
||||||
|
@ -330,6 +335,7 @@ type Engine struct {
|
||||||
enableAtModifier bool
|
enableAtModifier bool
|
||||||
enableNegativeOffset bool
|
enableNegativeOffset bool
|
||||||
enablePerStepStats bool
|
enablePerStepStats bool
|
||||||
|
enableDelayedNameRemoval bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewEngine returns a new engine.
|
// NewEngine returns a new engine.
|
||||||
|
@ -420,6 +426,7 @@ func NewEngine(opts EngineOpts) *Engine {
|
||||||
enableAtModifier: opts.EnableAtModifier,
|
enableAtModifier: opts.EnableAtModifier,
|
||||||
enableNegativeOffset: opts.EnableNegativeOffset,
|
enableNegativeOffset: opts.EnableNegativeOffset,
|
||||||
enablePerStepStats: opts.EnablePerStepStats,
|
enablePerStepStats: opts.EnablePerStepStats,
|
||||||
|
enableDelayedNameRemoval: opts.EnableDelayedNameRemoval,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -712,6 +719,7 @@ func (ng *Engine) execEvalStmt(ctx context.Context, query *query, s *parser.Eval
|
||||||
lookbackDelta: s.LookbackDelta,
|
lookbackDelta: s.LookbackDelta,
|
||||||
samplesStats: query.sampleStats,
|
samplesStats: query.sampleStats,
|
||||||
noStepSubqueryIntervalFn: ng.noStepSubqueryIntervalFn,
|
noStepSubqueryIntervalFn: ng.noStepSubqueryIntervalFn,
|
||||||
|
enableDelayedNameRemoval: ng.enableDelayedNameRemoval,
|
||||||
}
|
}
|
||||||
query.sampleStats.InitStepTracking(start, start, 1)
|
query.sampleStats.InitStepTracking(start, start, 1)
|
||||||
|
|
||||||
|
@ -743,9 +751,9 @@ func (ng *Engine) execEvalStmt(ctx context.Context, query *query, s *parser.Eval
|
||||||
// Point might have a different timestamp, force it to the evaluation
|
// Point might have a different timestamp, force it to the evaluation
|
||||||
// timestamp as that is when we ran the evaluation.
|
// timestamp as that is when we ran the evaluation.
|
||||||
if len(s.Histograms) > 0 {
|
if len(s.Histograms) > 0 {
|
||||||
vector[i] = Sample{Metric: s.Metric, H: s.Histograms[0].H, T: start}
|
vector[i] = Sample{Metric: s.Metric, H: s.Histograms[0].H, T: start, DropName: s.DropName}
|
||||||
} else {
|
} else {
|
||||||
vector[i] = Sample{Metric: s.Metric, F: s.Floats[0].F, T: start}
|
vector[i] = Sample{Metric: s.Metric, F: s.Floats[0].F, T: start, DropName: s.DropName}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return vector, warnings, nil
|
return vector, warnings, nil
|
||||||
|
@ -770,6 +778,7 @@ func (ng *Engine) execEvalStmt(ctx context.Context, query *query, s *parser.Eval
|
||||||
lookbackDelta: s.LookbackDelta,
|
lookbackDelta: s.LookbackDelta,
|
||||||
samplesStats: query.sampleStats,
|
samplesStats: query.sampleStats,
|
||||||
noStepSubqueryIntervalFn: ng.noStepSubqueryIntervalFn,
|
noStepSubqueryIntervalFn: ng.noStepSubqueryIntervalFn,
|
||||||
|
enableDelayedNameRemoval: ng.enableDelayedNameRemoval,
|
||||||
}
|
}
|
||||||
query.sampleStats.InitStepTracking(evaluator.startTimestamp, evaluator.endTimestamp, evaluator.interval)
|
query.sampleStats.InitStepTracking(evaluator.startTimestamp, evaluator.endTimestamp, evaluator.interval)
|
||||||
val, warnings, err := evaluator.Eval(s.Expr)
|
val, warnings, err := evaluator.Eval(s.Expr)
|
||||||
|
@ -1032,6 +1041,7 @@ type evaluator struct {
|
||||||
lookbackDelta time.Duration
|
lookbackDelta time.Duration
|
||||||
samplesStats *stats.QuerySamples
|
samplesStats *stats.QuerySamples
|
||||||
noStepSubqueryIntervalFn func(rangeMillis int64) int64
|
noStepSubqueryIntervalFn func(rangeMillis int64) int64
|
||||||
|
enableDelayedNameRemoval bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// errorf causes a panic with the input formatted into an error.
|
// errorf causes a panic with the input formatted into an error.
|
||||||
|
@ -1073,6 +1083,9 @@ func (ev *evaluator) Eval(expr parser.Expr) (v parser.Value, ws annotations.Anno
|
||||||
defer ev.recover(expr, &ws, &err)
|
defer ev.recover(expr, &ws, &err)
|
||||||
|
|
||||||
v, ws = ev.eval(expr)
|
v, ws = ev.eval(expr)
|
||||||
|
if ev.enableDelayedNameRemoval {
|
||||||
|
ev.cleanupMetricLabels(v)
|
||||||
|
}
|
||||||
return v, ws, nil
|
return v, ws, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1101,6 +1114,9 @@ type EvalNodeHelper struct {
|
||||||
rightSigs map[string]Sample
|
rightSigs map[string]Sample
|
||||||
matchedSigs map[string]map[uint64]struct{}
|
matchedSigs map[string]map[uint64]struct{}
|
||||||
resultMetric map[string]labels.Labels
|
resultMetric map[string]labels.Labels
|
||||||
|
|
||||||
|
// Additional options for the evaluation.
|
||||||
|
enableDelayedNameRemoval bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func (enh *EvalNodeHelper) resetBuilder(lbls labels.Labels) {
|
func (enh *EvalNodeHelper) resetBuilder(lbls labels.Labels) {
|
||||||
|
@ -1150,7 +1166,7 @@ func (ev *evaluator) rangeEval(prepSeries func(labels.Labels, *EvalSeriesHelper)
|
||||||
biggestLen = len(matrixes[i])
|
biggestLen = len(matrixes[i])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
enh := &EvalNodeHelper{Out: make(Vector, 0, biggestLen)}
|
enh := &EvalNodeHelper{Out: make(Vector, 0, biggestLen), enableDelayedNameRemoval: ev.enableDelayedNameRemoval}
|
||||||
type seriesAndTimestamp struct {
|
type seriesAndTimestamp struct {
|
||||||
Series
|
Series
|
||||||
ts int64
|
ts int64
|
||||||
|
@ -1196,12 +1212,12 @@ func (ev *evaluator) rangeEval(prepSeries func(labels.Labels, *EvalSeriesHelper)
|
||||||
for si, series := range matrixes[i] {
|
for si, series := range matrixes[i] {
|
||||||
switch {
|
switch {
|
||||||
case len(series.Floats) > 0 && series.Floats[0].T == ts:
|
case len(series.Floats) > 0 && series.Floats[0].T == ts:
|
||||||
vectors[i] = append(vectors[i], Sample{Metric: series.Metric, F: series.Floats[0].F, T: ts})
|
vectors[i] = append(vectors[i], Sample{Metric: series.Metric, F: series.Floats[0].F, T: ts, DropName: series.DropName})
|
||||||
// Move input vectors forward so we don't have to re-scan the same
|
// Move input vectors forward so we don't have to re-scan the same
|
||||||
// past points at the next step.
|
// past points at the next step.
|
||||||
matrixes[i][si].Floats = series.Floats[1:]
|
matrixes[i][si].Floats = series.Floats[1:]
|
||||||
case len(series.Histograms) > 0 && series.Histograms[0].T == ts:
|
case len(series.Histograms) > 0 && series.Histograms[0].T == ts:
|
||||||
vectors[i] = append(vectors[i], Sample{Metric: series.Metric, H: series.Histograms[0].H, T: ts})
|
vectors[i] = append(vectors[i], Sample{Metric: series.Metric, H: series.Histograms[0].H, T: ts, DropName: series.DropName})
|
||||||
matrixes[i][si].Histograms = series.Histograms[1:]
|
matrixes[i][si].Histograms = series.Histograms[1:]
|
||||||
default:
|
default:
|
||||||
continue
|
continue
|
||||||
|
@ -1240,15 +1256,15 @@ func (ev *evaluator) rangeEval(prepSeries func(labels.Labels, *EvalSeriesHelper)
|
||||||
|
|
||||||
// If this could be an instant query, shortcut so as not to change sort order.
|
// If this could be an instant query, shortcut so as not to change sort order.
|
||||||
if ev.endTimestamp == ev.startTimestamp {
|
if ev.endTimestamp == ev.startTimestamp {
|
||||||
if result.ContainsSameLabelset() {
|
if !ev.enableDelayedNameRemoval && result.ContainsSameLabelset() {
|
||||||
ev.errorf("vector cannot contain metrics with the same labelset")
|
ev.errorf("vector cannot contain metrics with the same labelset")
|
||||||
}
|
}
|
||||||
mat := make(Matrix, len(result))
|
mat := make(Matrix, len(result))
|
||||||
for i, s := range result {
|
for i, s := range result {
|
||||||
if s.H == nil {
|
if s.H == nil {
|
||||||
mat[i] = Series{Metric: s.Metric, Floats: []FPoint{{T: ts, F: s.F}}}
|
mat[i] = Series{Metric: s.Metric, Floats: []FPoint{{T: ts, F: s.F}}, DropName: s.DropName}
|
||||||
} else {
|
} else {
|
||||||
mat[i] = Series{Metric: s.Metric, Histograms: []HPoint{{T: ts, H: s.H}}}
|
mat[i] = Series{Metric: s.Metric, Histograms: []HPoint{{T: ts, H: s.H}}, DropName: s.DropName}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ev.currentSamples = originalNumSamples + mat.TotalSamples()
|
ev.currentSamples = originalNumSamples + mat.TotalSamples()
|
||||||
|
@ -1266,7 +1282,7 @@ func (ev *evaluator) rangeEval(prepSeries func(labels.Labels, *EvalSeriesHelper)
|
||||||
}
|
}
|
||||||
ss.ts = ts
|
ss.ts = ts
|
||||||
} else {
|
} else {
|
||||||
ss = seriesAndTimestamp{Series{Metric: sample.Metric}, ts}
|
ss = seriesAndTimestamp{Series{Metric: sample.Metric, DropName: sample.DropName}, ts}
|
||||||
}
|
}
|
||||||
addToSeries(&ss.Series, enh.Ts, sample.F, sample.H, numSteps)
|
addToSeries(&ss.Series, enh.Ts, sample.F, sample.H, numSteps)
|
||||||
seriess[h] = ss
|
seriess[h] = ss
|
||||||
|
@ -1302,7 +1318,7 @@ func (ev *evaluator) rangeEvalAgg(aggExpr *parser.AggregateExpr, sortedGrouping
|
||||||
|
|
||||||
var warnings annotations.Annotations
|
var warnings annotations.Annotations
|
||||||
|
|
||||||
enh := &EvalNodeHelper{}
|
enh := &EvalNodeHelper{enableDelayedNameRemoval: ev.enableDelayedNameRemoval}
|
||||||
tempNumSamples := ev.currentSamples
|
tempNumSamples := ev.currentSamples
|
||||||
|
|
||||||
// Create a mapping from input series to output groups.
|
// Create a mapping from input series to output groups.
|
||||||
|
@ -1611,10 +1627,17 @@ func (ev *evaluator) eval(expr parser.Expr) (parser.Value, annotations.Annotatio
|
||||||
var prevSS *Series
|
var prevSS *Series
|
||||||
inMatrix := make(Matrix, 1)
|
inMatrix := make(Matrix, 1)
|
||||||
inArgs[matrixArgIndex] = inMatrix
|
inArgs[matrixArgIndex] = inMatrix
|
||||||
enh := &EvalNodeHelper{Out: make(Vector, 0, 1)}
|
enh := &EvalNodeHelper{Out: make(Vector, 0, 1), enableDelayedNameRemoval: ev.enableDelayedNameRemoval}
|
||||||
// Process all the calls for one time series at a time.
|
// Process all the calls for one time series at a time.
|
||||||
it := storage.NewBuffer(selRange)
|
it := storage.NewBuffer(selRange)
|
||||||
var chkIter chunkenc.Iterator
|
var chkIter chunkenc.Iterator
|
||||||
|
|
||||||
|
// The last_over_time function acts like offset; thus, it
|
||||||
|
// should keep the metric name. For all the other range
|
||||||
|
// vector functions, the only change needed is to drop the
|
||||||
|
// metric name in the output.
|
||||||
|
dropName := e.Func.Name != "last_over_time"
|
||||||
|
|
||||||
for i, s := range selVS.Series {
|
for i, s := range selVS.Series {
|
||||||
if err := contextDone(ev.ctx, "expression evaluation"); err != nil {
|
if err := contextDone(ev.ctx, "expression evaluation"); err != nil {
|
||||||
ev.error(err)
|
ev.error(err)
|
||||||
|
@ -1629,15 +1652,12 @@ func (ev *evaluator) eval(expr parser.Expr) (parser.Value, annotations.Annotatio
|
||||||
chkIter = s.Iterator(chkIter)
|
chkIter = s.Iterator(chkIter)
|
||||||
it.Reset(chkIter)
|
it.Reset(chkIter)
|
||||||
metric := selVS.Series[i].Labels()
|
metric := selVS.Series[i].Labels()
|
||||||
// The last_over_time function acts like offset; thus, it
|
if !ev.enableDelayedNameRemoval && dropName {
|
||||||
// should keep the metric name. For all the other range
|
|
||||||
// vector functions, the only change needed is to drop the
|
|
||||||
// metric name in the output.
|
|
||||||
if e.Func.Name != "last_over_time" {
|
|
||||||
metric = metric.DropMetricName()
|
metric = metric.DropMetricName()
|
||||||
}
|
}
|
||||||
ss := Series{
|
ss := Series{
|
||||||
Metric: metric,
|
Metric: metric,
|
||||||
|
DropName: dropName,
|
||||||
}
|
}
|
||||||
inMatrix[0].Metric = selVS.Series[i].Labels()
|
inMatrix[0].Metric = selVS.Series[i].Labels()
|
||||||
for ts, step := ev.startTimestamp, -1; ts <= ev.endTimestamp; ts += ev.interval {
|
for ts, step := ev.startTimestamp, -1; ts <= ev.endTimestamp; ts += ev.interval {
|
||||||
|
@ -1752,16 +1772,16 @@ func (ev *evaluator) eval(expr parser.Expr) (parser.Value, annotations.Annotatio
|
||||||
|
|
||||||
return Matrix{
|
return Matrix{
|
||||||
Series{
|
Series{
|
||||||
Metric: createLabelsForAbsentFunction(e.Args[0]),
|
Metric: createLabelsForAbsentFunction(e.Args[0]),
|
||||||
Floats: newp,
|
Floats: newp,
|
||||||
|
DropName: dropName,
|
||||||
},
|
},
|
||||||
}, warnings
|
}, warnings
|
||||||
}
|
}
|
||||||
|
|
||||||
if mat.ContainsSameLabelset() {
|
if !ev.enableDelayedNameRemoval && mat.ContainsSameLabelset() {
|
||||||
ev.errorf("vector cannot contain metrics with the same labelset")
|
ev.errorf("vector cannot contain metrics with the same labelset")
|
||||||
}
|
}
|
||||||
|
|
||||||
return mat, warnings
|
return mat, warnings
|
||||||
|
|
||||||
case *parser.ParenExpr:
|
case *parser.ParenExpr:
|
||||||
|
@ -1772,12 +1792,15 @@ func (ev *evaluator) eval(expr parser.Expr) (parser.Value, annotations.Annotatio
|
||||||
mat := val.(Matrix)
|
mat := val.(Matrix)
|
||||||
if e.Op == parser.SUB {
|
if e.Op == parser.SUB {
|
||||||
for i := range mat {
|
for i := range mat {
|
||||||
mat[i].Metric = mat[i].Metric.DropMetricName()
|
if !ev.enableDelayedNameRemoval {
|
||||||
|
mat[i].Metric = mat[i].Metric.DropMetricName()
|
||||||
|
}
|
||||||
|
mat[i].DropName = true
|
||||||
for j := range mat[i].Floats {
|
for j := range mat[i].Floats {
|
||||||
mat[i].Floats[j].F = -mat[i].Floats[j].F
|
mat[i].Floats[j].F = -mat[i].Floats[j].F
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if mat.ContainsSameLabelset() {
|
if !ev.enableDelayedNameRemoval && mat.ContainsSameLabelset() {
|
||||||
ev.errorf("vector cannot contain metrics with the same labelset")
|
ev.errorf("vector cannot contain metrics with the same labelset")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1913,6 +1936,7 @@ func (ev *evaluator) eval(expr parser.Expr) (parser.Value, annotations.Annotatio
|
||||||
lookbackDelta: ev.lookbackDelta,
|
lookbackDelta: ev.lookbackDelta,
|
||||||
samplesStats: ev.samplesStats.NewChild(),
|
samplesStats: ev.samplesStats.NewChild(),
|
||||||
noStepSubqueryIntervalFn: ev.noStepSubqueryIntervalFn,
|
noStepSubqueryIntervalFn: ev.noStepSubqueryIntervalFn,
|
||||||
|
enableDelayedNameRemoval: ev.enableDelayedNameRemoval,
|
||||||
}
|
}
|
||||||
|
|
||||||
if e.Step != 0 {
|
if e.Step != 0 {
|
||||||
|
@ -1957,6 +1981,7 @@ func (ev *evaluator) eval(expr parser.Expr) (parser.Value, annotations.Annotatio
|
||||||
lookbackDelta: ev.lookbackDelta,
|
lookbackDelta: ev.lookbackDelta,
|
||||||
samplesStats: ev.samplesStats.NewChild(),
|
samplesStats: ev.samplesStats.NewChild(),
|
||||||
noStepSubqueryIntervalFn: ev.noStepSubqueryIntervalFn,
|
noStepSubqueryIntervalFn: ev.noStepSubqueryIntervalFn,
|
||||||
|
enableDelayedNameRemoval: ev.enableDelayedNameRemoval,
|
||||||
}
|
}
|
||||||
res, ws := newEv.eval(e.Expr)
|
res, ws := newEv.eval(e.Expr)
|
||||||
ev.currentSamples = newEv.currentSamples
|
ev.currentSamples = newEv.currentSamples
|
||||||
|
@ -2553,7 +2578,7 @@ func (ev *evaluator) VectorBinop(op parser.ItemType, lhs, rhs Vector, matching *
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
metric := resultMetric(ls.Metric, rs.Metric, op, matching, enh)
|
metric := resultMetric(ls.Metric, rs.Metric, op, matching, enh)
|
||||||
if returnBool {
|
if !ev.enableDelayedNameRemoval && returnBool {
|
||||||
metric = metric.DropMetricName()
|
metric = metric.DropMetricName()
|
||||||
}
|
}
|
||||||
insertedSigs, exists := matchedSigs[sig]
|
insertedSigs, exists := matchedSigs[sig]
|
||||||
|
@ -2578,9 +2603,10 @@ func (ev *evaluator) VectorBinop(op parser.ItemType, lhs, rhs Vector, matching *
|
||||||
}
|
}
|
||||||
|
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: metric,
|
Metric: metric,
|
||||||
F: floatValue,
|
F: floatValue,
|
||||||
H: histogramValue,
|
H: histogramValue,
|
||||||
|
DropName: returnBool,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, lastErr
|
return enh.Out, lastErr
|
||||||
|
@ -2680,7 +2706,10 @@ func (ev *evaluator) VectorscalarBinop(op parser.ItemType, lhs Vector, rhs Scala
|
||||||
lhsSample.F = float
|
lhsSample.F = float
|
||||||
lhsSample.H = histogram
|
lhsSample.H = histogram
|
||||||
if shouldDropMetricName(op) || returnBool {
|
if shouldDropMetricName(op) || returnBool {
|
||||||
lhsSample.Metric = lhsSample.Metric.DropMetricName()
|
if !ev.enableDelayedNameRemoval {
|
||||||
|
lhsSample.Metric = lhsSample.Metric.DropMetricName()
|
||||||
|
}
|
||||||
|
lhsSample.DropName = true
|
||||||
}
|
}
|
||||||
enh.Out = append(enh.Out, lhsSample)
|
enh.Out = append(enh.Out, lhsSample)
|
||||||
}
|
}
|
||||||
|
@ -3019,6 +3048,7 @@ func (ev *evaluator) aggregation(e *parser.AggregateExpr, q float64, inputMatrix
|
||||||
|
|
||||||
ss := &outputMatrix[ri]
|
ss := &outputMatrix[ri]
|
||||||
addToSeries(ss, enh.Ts, aggr.floatValue, aggr.histogramValue, numSteps)
|
addToSeries(ss, enh.Ts, aggr.floatValue, aggr.histogramValue, numSteps)
|
||||||
|
ss.DropName = inputMatrix[ri].DropName
|
||||||
}
|
}
|
||||||
|
|
||||||
return annos
|
return annos
|
||||||
|
@ -3045,7 +3075,7 @@ seriesLoop:
|
||||||
if !ok {
|
if !ok {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
s = Sample{Metric: inputMatrix[si].Metric, F: f}
|
s = Sample{Metric: inputMatrix[si].Metric, F: f, DropName: inputMatrix[si].DropName}
|
||||||
|
|
||||||
group := &groups[seriesToResult[si]]
|
group := &groups[seriesToResult[si]]
|
||||||
// Initialize this group if it's the first time we've seen it.
|
// Initialize this group if it's the first time we've seen it.
|
||||||
|
@ -3129,16 +3159,16 @@ seriesLoop:
|
||||||
mat = make(Matrix, 0, len(groups))
|
mat = make(Matrix, 0, len(groups))
|
||||||
}
|
}
|
||||||
|
|
||||||
add := func(lbls labels.Labels, f float64) {
|
add := func(lbls labels.Labels, f float64, dropName bool) {
|
||||||
// If this could be an instant query, add directly to the matrix so the result is in consistent order.
|
// If this could be an instant query, add directly to the matrix so the result is in consistent order.
|
||||||
if ev.endTimestamp == ev.startTimestamp {
|
if ev.endTimestamp == ev.startTimestamp {
|
||||||
mat = append(mat, Series{Metric: lbls, Floats: []FPoint{{T: enh.Ts, F: f}}})
|
mat = append(mat, Series{Metric: lbls, Floats: []FPoint{{T: enh.Ts, F: f}}, DropName: dropName})
|
||||||
} else {
|
} else {
|
||||||
// Otherwise the results are added into seriess elements.
|
// Otherwise the results are added into seriess elements.
|
||||||
hash := lbls.Hash()
|
hash := lbls.Hash()
|
||||||
ss, ok := seriess[hash]
|
ss, ok := seriess[hash]
|
||||||
if !ok {
|
if !ok {
|
||||||
ss = Series{Metric: lbls}
|
ss = Series{Metric: lbls, DropName: dropName}
|
||||||
}
|
}
|
||||||
addToSeries(&ss, enh.Ts, f, nil, numSteps)
|
addToSeries(&ss, enh.Ts, f, nil, numSteps)
|
||||||
seriess[hash] = ss
|
seriess[hash] = ss
|
||||||
|
@ -3155,7 +3185,7 @@ seriesLoop:
|
||||||
sort.Sort(sort.Reverse(aggr.heap))
|
sort.Sort(sort.Reverse(aggr.heap))
|
||||||
}
|
}
|
||||||
for _, v := range aggr.heap {
|
for _, v := range aggr.heap {
|
||||||
add(v.Metric, v.F)
|
add(v.Metric, v.F, v.DropName)
|
||||||
}
|
}
|
||||||
|
|
||||||
case parser.BOTTOMK:
|
case parser.BOTTOMK:
|
||||||
|
@ -3164,12 +3194,12 @@ seriesLoop:
|
||||||
sort.Sort(sort.Reverse((*vectorByReverseValueHeap)(&aggr.heap)))
|
sort.Sort(sort.Reverse((*vectorByReverseValueHeap)(&aggr.heap)))
|
||||||
}
|
}
|
||||||
for _, v := range aggr.heap {
|
for _, v := range aggr.heap {
|
||||||
add(v.Metric, v.F)
|
add(v.Metric, v.F, v.DropName)
|
||||||
}
|
}
|
||||||
|
|
||||||
case parser.LIMITK, parser.LIMIT_RATIO:
|
case parser.LIMITK, parser.LIMIT_RATIO:
|
||||||
for _, v := range aggr.heap {
|
for _, v := range aggr.heap {
|
||||||
add(v.Metric, v.F)
|
add(v.Metric, v.F, v.DropName)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3221,6 +3251,30 @@ func (ev *evaluator) aggregationCountValues(e *parser.AggregateExpr, grouping []
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (ev *evaluator) cleanupMetricLabels(v parser.Value) {
|
||||||
|
if v.Type() == parser.ValueTypeMatrix {
|
||||||
|
mat := v.(Matrix)
|
||||||
|
for i := range mat {
|
||||||
|
if mat[i].DropName {
|
||||||
|
mat[i].Metric = mat[i].Metric.DropMetricName()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if mat.ContainsSameLabelset() {
|
||||||
|
ev.errorf("vector cannot contain metrics with the same labelset")
|
||||||
|
}
|
||||||
|
} else if v.Type() == parser.ValueTypeVector {
|
||||||
|
vec := v.(Vector)
|
||||||
|
for i := range vec {
|
||||||
|
if vec[i].DropName {
|
||||||
|
vec[i].Metric = vec[i].Metric.DropMetricName()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if vec.ContainsSameLabelset() {
|
||||||
|
ev.errorf("vector cannot contain metrics with the same labelset")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func addToSeries(ss *Series, ts int64, f float64, h *histogram.FloatHistogram, numSteps int) {
|
func addToSeries(ss *Series, ts int64, f float64, h *histogram.FloatHistogram, numSteps int) {
|
||||||
if h == nil {
|
if h == nil {
|
||||||
if ss.Floats == nil {
|
if ss.Floats == nil {
|
||||||
|
|
|
@ -17,7 +17,6 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math"
|
|
||||||
"os"
|
"os"
|
||||||
"sort"
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
@ -1714,7 +1713,8 @@ load 1ms
|
||||||
{F: 3600, T: 6 * 60 * 1000},
|
{F: 3600, T: 6 * 60 * 1000},
|
||||||
{F: 3600, T: 7 * 60 * 1000},
|
{F: 3600, T: 7 * 60 * 1000},
|
||||||
},
|
},
|
||||||
Metric: labels.EmptyLabels(),
|
Metric: labels.EmptyLabels(),
|
||||||
|
DropName: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1930,20 +1930,24 @@ func TestSubquerySelector(t *testing.T) {
|
||||||
nil,
|
nil,
|
||||||
promql.Matrix{
|
promql.Matrix{
|
||||||
promql.Series{
|
promql.Series{
|
||||||
Floats: []promql.FPoint{{F: 3, T: 7985000}, {F: 3, T: 7990000}, {F: 3, T: 7995000}, {F: 3, T: 8000000}},
|
Floats: []promql.FPoint{{F: 3, T: 7985000}, {F: 3, T: 7990000}, {F: 3, T: 7995000}, {F: 3, T: 8000000}},
|
||||||
Metric: labels.FromStrings("job", "api-server", "instance", "0", "group", "canary"),
|
Metric: labels.FromStrings("job", "api-server", "instance", "0", "group", "canary"),
|
||||||
|
DropName: true,
|
||||||
},
|
},
|
||||||
promql.Series{
|
promql.Series{
|
||||||
Floats: []promql.FPoint{{F: 4, T: 7985000}, {F: 4, T: 7990000}, {F: 4, T: 7995000}, {F: 4, T: 8000000}},
|
Floats: []promql.FPoint{{F: 4, T: 7985000}, {F: 4, T: 7990000}, {F: 4, T: 7995000}, {F: 4, T: 8000000}},
|
||||||
Metric: labels.FromStrings("job", "api-server", "instance", "1", "group", "canary"),
|
Metric: labels.FromStrings("job", "api-server", "instance", "1", "group", "canary"),
|
||||||
|
DropName: true,
|
||||||
},
|
},
|
||||||
promql.Series{
|
promql.Series{
|
||||||
Floats: []promql.FPoint{{F: 1, T: 7985000}, {F: 1, T: 7990000}, {F: 1, T: 7995000}, {F: 1, T: 8000000}},
|
Floats: []promql.FPoint{{F: 1, T: 7985000}, {F: 1, T: 7990000}, {F: 1, T: 7995000}, {F: 1, T: 8000000}},
|
||||||
Metric: labels.FromStrings("job", "api-server", "instance", "0", "group", "production"),
|
Metric: labels.FromStrings("job", "api-server", "instance", "0", "group", "production"),
|
||||||
|
DropName: true,
|
||||||
},
|
},
|
||||||
promql.Series{
|
promql.Series{
|
||||||
Floats: []promql.FPoint{{F: 2, T: 7985000}, {F: 2, T: 7990000}, {F: 2, T: 7995000}, {F: 2, T: 8000000}},
|
Floats: []promql.FPoint{{F: 2, T: 7985000}, {F: 2, T: 7990000}, {F: 2, T: 7995000}, {F: 2, T: 8000000}},
|
||||||
Metric: labels.FromStrings("job", "api-server", "instance", "1", "group", "production"),
|
Metric: labels.FromStrings("job", "api-server", "instance", "1", "group", "production"),
|
||||||
|
DropName: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
nil,
|
nil,
|
||||||
|
@ -3097,217 +3101,6 @@ func TestInstantQueryWithRangeVectorSelector(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestNativeHistogram_Sum_Count_Add_AvgOperator(t *testing.T) {
|
|
||||||
// TODO(codesome): Integrate histograms into the PromQL testing framework
|
|
||||||
// and write more tests there.
|
|
||||||
cases := []struct {
|
|
||||||
histograms []histogram.Histogram
|
|
||||||
expected histogram.FloatHistogram
|
|
||||||
expectedAvg histogram.FloatHistogram
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
histograms: []histogram.Histogram{
|
|
||||||
{
|
|
||||||
CounterResetHint: histogram.GaugeType,
|
|
||||||
Schema: 0,
|
|
||||||
Count: 25,
|
|
||||||
Sum: 1234.5,
|
|
||||||
ZeroThreshold: 0.001,
|
|
||||||
ZeroCount: 4,
|
|
||||||
PositiveSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 2},
|
|
||||||
{Offset: 1, Length: 2},
|
|
||||||
},
|
|
||||||
PositiveBuckets: []int64{1, 1, -1, 0},
|
|
||||||
NegativeSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 2},
|
|
||||||
{Offset: 2, Length: 2},
|
|
||||||
},
|
|
||||||
NegativeBuckets: []int64{2, 2, -3, 8},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
CounterResetHint: histogram.GaugeType,
|
|
||||||
Schema: 0,
|
|
||||||
Count: 41,
|
|
||||||
Sum: 2345.6,
|
|
||||||
ZeroThreshold: 0.001,
|
|
||||||
ZeroCount: 5,
|
|
||||||
PositiveSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 4},
|
|
||||||
{Offset: 0, Length: 0},
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
PositiveBuckets: []int64{1, 2, -2, 1, -1, 0, 0},
|
|
||||||
NegativeSpans: []histogram.Span{
|
|
||||||
{Offset: 1, Length: 4},
|
|
||||||
{Offset: 2, Length: 0},
|
|
||||||
{Offset: 2, Length: 3},
|
|
||||||
},
|
|
||||||
NegativeBuckets: []int64{1, 3, -2, 5, -2, 0, -3},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
CounterResetHint: histogram.GaugeType,
|
|
||||||
Schema: 0,
|
|
||||||
Count: 41,
|
|
||||||
Sum: 1111.1,
|
|
||||||
ZeroThreshold: 0.001,
|
|
||||||
ZeroCount: 5,
|
|
||||||
PositiveSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 4},
|
|
||||||
{Offset: 0, Length: 0},
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
PositiveBuckets: []int64{1, 2, -2, 1, -1, 0, 0},
|
|
||||||
NegativeSpans: []histogram.Span{
|
|
||||||
{Offset: 1, Length: 4},
|
|
||||||
{Offset: 2, Length: 0},
|
|
||||||
{Offset: 2, Length: 3},
|
|
||||||
},
|
|
||||||
NegativeBuckets: []int64{1, 3, -2, 5, -2, 0, -3},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
CounterResetHint: histogram.GaugeType,
|
|
||||||
Schema: 1, // Everything is 0 just to make the count 4 so avg has nicer numbers.
|
|
||||||
},
|
|
||||||
},
|
|
||||||
expected: histogram.FloatHistogram{
|
|
||||||
CounterResetHint: histogram.GaugeType,
|
|
||||||
Schema: 0,
|
|
||||||
ZeroThreshold: 0.001,
|
|
||||||
ZeroCount: 14,
|
|
||||||
Count: 107,
|
|
||||||
Sum: 4691.2,
|
|
||||||
PositiveSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 7},
|
|
||||||
},
|
|
||||||
PositiveBuckets: []float64{3, 8, 2, 5, 3, 2, 2},
|
|
||||||
NegativeSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 6},
|
|
||||||
{Offset: 3, Length: 3},
|
|
||||||
},
|
|
||||||
NegativeBuckets: []float64{2, 6, 8, 4, 15, 9, 10, 10, 4},
|
|
||||||
},
|
|
||||||
expectedAvg: histogram.FloatHistogram{
|
|
||||||
CounterResetHint: histogram.GaugeType,
|
|
||||||
Schema: 0,
|
|
||||||
ZeroThreshold: 0.001,
|
|
||||||
ZeroCount: 3.5,
|
|
||||||
Count: 26.75,
|
|
||||||
Sum: 1172.8,
|
|
||||||
PositiveSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 7},
|
|
||||||
},
|
|
||||||
PositiveBuckets: []float64{0.75, 2, 0.5, 1.25, 0.75, 0.5, 0.5},
|
|
||||||
NegativeSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 6},
|
|
||||||
{Offset: 3, Length: 3},
|
|
||||||
},
|
|
||||||
NegativeBuckets: []float64{0.5, 1.5, 2, 1, 3.75, 2.25, 2.5, 2.5, 1},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
idx0 := int64(0)
|
|
||||||
for _, c := range cases {
|
|
||||||
for _, floatHisto := range []bool{true, false} {
|
|
||||||
t.Run(fmt.Sprintf("floatHistogram=%t %d", floatHisto, idx0), func(t *testing.T) {
|
|
||||||
storage := teststorage.New(t)
|
|
||||||
t.Cleanup(func() { storage.Close() })
|
|
||||||
|
|
||||||
seriesName := "sparse_histogram_series"
|
|
||||||
seriesNameOverTime := "sparse_histogram_series_over_time"
|
|
||||||
|
|
||||||
engine := newTestEngine()
|
|
||||||
|
|
||||||
ts := idx0 * int64(10*time.Minute/time.Millisecond)
|
|
||||||
app := storage.Appender(context.Background())
|
|
||||||
_, err := app.Append(0, labels.FromStrings("__name__", "float_series", "idx", "0"), ts, 42)
|
|
||||||
require.NoError(t, err)
|
|
||||||
for idx1, h := range c.histograms {
|
|
||||||
lbls := labels.FromStrings("__name__", seriesName, "idx", strconv.Itoa(idx1))
|
|
||||||
// Since we mutate h later, we need to create a copy here.
|
|
||||||
var err error
|
|
||||||
if floatHisto {
|
|
||||||
_, err = app.AppendHistogram(0, lbls, ts, nil, h.Copy().ToFloat(nil))
|
|
||||||
} else {
|
|
||||||
_, err = app.AppendHistogram(0, lbls, ts, h.Copy(), nil)
|
|
||||||
}
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
lbls = labels.FromStrings("__name__", seriesNameOverTime)
|
|
||||||
newTs := ts + int64(idx1)*int64(time.Minute/time.Millisecond)
|
|
||||||
// Since we mutate h later, we need to create a copy here.
|
|
||||||
if floatHisto {
|
|
||||||
_, err = app.AppendHistogram(0, lbls, newTs, nil, h.Copy().ToFloat(nil))
|
|
||||||
} else {
|
|
||||||
_, err = app.AppendHistogram(0, lbls, newTs, h.Copy(), nil)
|
|
||||||
}
|
|
||||||
require.NoError(t, err)
|
|
||||||
}
|
|
||||||
require.NoError(t, app.Commit())
|
|
||||||
|
|
||||||
queryAndCheck := func(queryString string, ts int64, exp promql.Vector) {
|
|
||||||
qry, err := engine.NewInstantQuery(context.Background(), storage, nil, queryString, timestamp.Time(ts))
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
res := qry.Exec(context.Background())
|
|
||||||
require.NoError(t, res.Err)
|
|
||||||
require.Empty(t, res.Warnings)
|
|
||||||
|
|
||||||
vector, err := res.Vector()
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
testutil.RequireEqual(t, exp, vector)
|
|
||||||
}
|
|
||||||
queryAndCheckAnnotations := func(queryString string, ts int64, expWarnings annotations.Annotations) {
|
|
||||||
qry, err := engine.NewInstantQuery(context.Background(), storage, nil, queryString, timestamp.Time(ts))
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
res := qry.Exec(context.Background())
|
|
||||||
require.NoError(t, res.Err)
|
|
||||||
require.Equal(t, expWarnings, res.Warnings)
|
|
||||||
}
|
|
||||||
|
|
||||||
// sum().
|
|
||||||
queryString := fmt.Sprintf("sum(%s)", seriesName)
|
|
||||||
queryAndCheck(queryString, ts, []promql.Sample{{T: ts, H: &c.expected, Metric: labels.EmptyLabels()}})
|
|
||||||
|
|
||||||
queryString = `sum({idx="0"})`
|
|
||||||
var annos annotations.Annotations
|
|
||||||
annos.Add(annotations.NewMixedFloatsHistogramsAggWarning(posrange.PositionRange{Start: 4, End: 13}))
|
|
||||||
queryAndCheckAnnotations(queryString, ts, annos)
|
|
||||||
|
|
||||||
// + operator.
|
|
||||||
queryString = fmt.Sprintf(`%s{idx="0"}`, seriesName)
|
|
||||||
for idx := 1; idx < len(c.histograms); idx++ {
|
|
||||||
queryString += fmt.Sprintf(` + ignoring(idx) %s{idx="%d"}`, seriesName, idx)
|
|
||||||
}
|
|
||||||
queryAndCheck(queryString, ts, []promql.Sample{{T: ts, H: &c.expected, Metric: labels.EmptyLabels()}})
|
|
||||||
|
|
||||||
// count().
|
|
||||||
queryString = fmt.Sprintf("count(%s)", seriesName)
|
|
||||||
queryAndCheck(queryString, ts, []promql.Sample{{T: ts, F: 4, Metric: labels.EmptyLabels()}})
|
|
||||||
|
|
||||||
// avg().
|
|
||||||
queryString = fmt.Sprintf("avg(%s)", seriesName)
|
|
||||||
queryAndCheck(queryString, ts, []promql.Sample{{T: ts, H: &c.expectedAvg, Metric: labels.EmptyLabels()}})
|
|
||||||
|
|
||||||
offset := int64(len(c.histograms) - 1)
|
|
||||||
newTs := ts + offset*int64(time.Minute/time.Millisecond)
|
|
||||||
|
|
||||||
// sum_over_time().
|
|
||||||
queryString = fmt.Sprintf("sum_over_time(%s[%dm:1m])", seriesNameOverTime, offset)
|
|
||||||
queryAndCheck(queryString, newTs, []promql.Sample{{T: newTs, H: &c.expected, Metric: labels.EmptyLabels()}})
|
|
||||||
|
|
||||||
// avg_over_time().
|
|
||||||
queryString = fmt.Sprintf("avg_over_time(%s[%dm:1m])", seriesNameOverTime, offset)
|
|
||||||
queryAndCheck(queryString, newTs, []promql.Sample{{T: newTs, H: &c.expectedAvg, Metric: labels.EmptyLabels()}})
|
|
||||||
})
|
|
||||||
idx0++
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestNativeHistogram_SubOperator(t *testing.T) {
|
func TestNativeHistogram_SubOperator(t *testing.T) {
|
||||||
// TODO(codesome): Integrate histograms into the PromQL testing framework
|
// TODO(codesome): Integrate histograms into the PromQL testing framework
|
||||||
// and write more tests there.
|
// and write more tests there.
|
||||||
|
@ -3543,171 +3336,6 @@ func TestNativeHistogram_SubOperator(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestNativeHistogram_MulDivOperator(t *testing.T) {
|
|
||||||
// TODO(codesome): Integrate histograms into the PromQL testing framework
|
|
||||||
// and write more tests there.
|
|
||||||
originalHistogram := histogram.Histogram{
|
|
||||||
Schema: 0,
|
|
||||||
Count: 21,
|
|
||||||
Sum: 33,
|
|
||||||
ZeroThreshold: 0.001,
|
|
||||||
ZeroCount: 3,
|
|
||||||
PositiveSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
PositiveBuckets: []int64{3, 0, 0},
|
|
||||||
NegativeSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
NegativeBuckets: []int64{3, 0, 0},
|
|
||||||
}
|
|
||||||
|
|
||||||
cases := []struct {
|
|
||||||
scalar float64
|
|
||||||
histogram histogram.Histogram
|
|
||||||
expectedMul histogram.FloatHistogram
|
|
||||||
expectedDiv histogram.FloatHistogram
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
scalar: 3,
|
|
||||||
histogram: originalHistogram,
|
|
||||||
expectedMul: histogram.FloatHistogram{
|
|
||||||
Schema: 0,
|
|
||||||
Count: 63,
|
|
||||||
Sum: 99,
|
|
||||||
ZeroThreshold: 0.001,
|
|
||||||
ZeroCount: 9,
|
|
||||||
PositiveSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
PositiveBuckets: []float64{9, 9, 9},
|
|
||||||
NegativeSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
NegativeBuckets: []float64{9, 9, 9},
|
|
||||||
},
|
|
||||||
expectedDiv: histogram.FloatHistogram{
|
|
||||||
Schema: 0,
|
|
||||||
Count: 7,
|
|
||||||
Sum: 11,
|
|
||||||
ZeroThreshold: 0.001,
|
|
||||||
ZeroCount: 1,
|
|
||||||
PositiveSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
PositiveBuckets: []float64{1, 1, 1},
|
|
||||||
NegativeSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
NegativeBuckets: []float64{1, 1, 1},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
scalar: 0,
|
|
||||||
histogram: originalHistogram,
|
|
||||||
expectedMul: histogram.FloatHistogram{
|
|
||||||
Schema: 0,
|
|
||||||
Count: 0,
|
|
||||||
Sum: 0,
|
|
||||||
ZeroThreshold: 0.001,
|
|
||||||
ZeroCount: 0,
|
|
||||||
PositiveSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
PositiveBuckets: []float64{0, 0, 0},
|
|
||||||
NegativeSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
NegativeBuckets: []float64{0, 0, 0},
|
|
||||||
},
|
|
||||||
expectedDiv: histogram.FloatHistogram{
|
|
||||||
Schema: 0,
|
|
||||||
Count: math.Inf(1),
|
|
||||||
Sum: math.Inf(1),
|
|
||||||
ZeroThreshold: 0.001,
|
|
||||||
ZeroCount: math.Inf(1),
|
|
||||||
PositiveSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
PositiveBuckets: []float64{math.Inf(1), math.Inf(1), math.Inf(1)},
|
|
||||||
NegativeSpans: []histogram.Span{
|
|
||||||
{Offset: 0, Length: 3},
|
|
||||||
},
|
|
||||||
NegativeBuckets: []float64{math.Inf(1), math.Inf(1), math.Inf(1)},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
idx0 := int64(0)
|
|
||||||
for _, c := range cases {
|
|
||||||
for _, floatHisto := range []bool{true, false} {
|
|
||||||
t.Run(fmt.Sprintf("floatHistogram=%t %d", floatHisto, idx0), func(t *testing.T) {
|
|
||||||
storage := teststorage.New(t)
|
|
||||||
t.Cleanup(func() { storage.Close() })
|
|
||||||
|
|
||||||
seriesName := "sparse_histogram_series"
|
|
||||||
floatSeriesName := "float_series"
|
|
||||||
|
|
||||||
engine := newTestEngine()
|
|
||||||
|
|
||||||
ts := idx0 * int64(10*time.Minute/time.Millisecond)
|
|
||||||
app := storage.Appender(context.Background())
|
|
||||||
h := c.histogram
|
|
||||||
lbls := labels.FromStrings("__name__", seriesName)
|
|
||||||
// Since we mutate h later, we need to create a copy here.
|
|
||||||
var err error
|
|
||||||
if floatHisto {
|
|
||||||
_, err = app.AppendHistogram(0, lbls, ts, nil, h.Copy().ToFloat(nil))
|
|
||||||
} else {
|
|
||||||
_, err = app.AppendHistogram(0, lbls, ts, h.Copy(), nil)
|
|
||||||
}
|
|
||||||
require.NoError(t, err)
|
|
||||||
_, err = app.Append(0, labels.FromStrings("__name__", floatSeriesName), ts, c.scalar)
|
|
||||||
require.NoError(t, err)
|
|
||||||
require.NoError(t, app.Commit())
|
|
||||||
|
|
||||||
queryAndCheck := func(queryString string, exp promql.Vector) {
|
|
||||||
qry, err := engine.NewInstantQuery(context.Background(), storage, nil, queryString, timestamp.Time(ts))
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
res := qry.Exec(context.Background())
|
|
||||||
require.NoError(t, res.Err)
|
|
||||||
|
|
||||||
vector, err := res.Vector()
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
testutil.RequireEqual(t, exp, vector)
|
|
||||||
}
|
|
||||||
|
|
||||||
// histogram * scalar.
|
|
||||||
queryString := fmt.Sprintf(`%s * %f`, seriesName, c.scalar)
|
|
||||||
queryAndCheck(queryString, []promql.Sample{{T: ts, H: &c.expectedMul, Metric: labels.EmptyLabels()}})
|
|
||||||
|
|
||||||
// scalar * histogram.
|
|
||||||
queryString = fmt.Sprintf(`%f * %s`, c.scalar, seriesName)
|
|
||||||
queryAndCheck(queryString, []promql.Sample{{T: ts, H: &c.expectedMul, Metric: labels.EmptyLabels()}})
|
|
||||||
|
|
||||||
// histogram * float.
|
|
||||||
queryString = fmt.Sprintf(`%s * %s`, seriesName, floatSeriesName)
|
|
||||||
queryAndCheck(queryString, []promql.Sample{{T: ts, H: &c.expectedMul, Metric: labels.EmptyLabels()}})
|
|
||||||
|
|
||||||
// float * histogram.
|
|
||||||
queryString = fmt.Sprintf(`%s * %s`, floatSeriesName, seriesName)
|
|
||||||
queryAndCheck(queryString, []promql.Sample{{T: ts, H: &c.expectedMul, Metric: labels.EmptyLabels()}})
|
|
||||||
|
|
||||||
// histogram / scalar.
|
|
||||||
queryString = fmt.Sprintf(`%s / %f`, seriesName, c.scalar)
|
|
||||||
queryAndCheck(queryString, []promql.Sample{{T: ts, H: &c.expectedDiv, Metric: labels.EmptyLabels()}})
|
|
||||||
|
|
||||||
// histogram / float.
|
|
||||||
queryString = fmt.Sprintf(`%s / %s`, seriesName, floatSeriesName)
|
|
||||||
queryAndCheck(queryString, []promql.Sample{{T: ts, H: &c.expectedDiv, Metric: labels.EmptyLabels()}})
|
|
||||||
})
|
|
||||||
idx0++
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestQueryLookbackDelta(t *testing.T) {
|
func TestQueryLookbackDelta(t *testing.T) {
|
||||||
var (
|
var (
|
||||||
load = `load 5m
|
load = `load 5m
|
||||||
|
|
|
@ -483,9 +483,13 @@ func funcClamp(vals []parser.Value, args parser.Expressions, enh *EvalNodeHelper
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
}
|
}
|
||||||
for _, el := range vec {
|
for _, el := range vec {
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
el.Metric = el.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: el.Metric.DropMetricName(),
|
Metric: el.Metric,
|
||||||
F: math.Max(minVal, math.Min(maxVal, el.F)),
|
F: math.Max(minVal, math.Min(maxVal, el.F)),
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
|
@ -496,9 +500,13 @@ func funcClampMax(vals []parser.Value, args parser.Expressions, enh *EvalNodeHel
|
||||||
vec := vals[0].(Vector)
|
vec := vals[0].(Vector)
|
||||||
maxVal := vals[1].(Vector)[0].F
|
maxVal := vals[1].(Vector)[0].F
|
||||||
for _, el := range vec {
|
for _, el := range vec {
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
el.Metric = el.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: el.Metric.DropMetricName(),
|
Metric: el.Metric,
|
||||||
F: math.Min(maxVal, el.F),
|
F: math.Min(maxVal, el.F),
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
|
@ -509,9 +517,13 @@ func funcClampMin(vals []parser.Value, args parser.Expressions, enh *EvalNodeHel
|
||||||
vec := vals[0].(Vector)
|
vec := vals[0].(Vector)
|
||||||
minVal := vals[1].(Vector)[0].F
|
minVal := vals[1].(Vector)[0].F
|
||||||
for _, el := range vec {
|
for _, el := range vec {
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
el.Metric = el.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: el.Metric.DropMetricName(),
|
Metric: el.Metric,
|
||||||
F: math.Max(minVal, el.F),
|
F: math.Max(minVal, el.F),
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
|
@ -532,8 +544,9 @@ func funcRound(vals []parser.Value, args parser.Expressions, enh *EvalNodeHelper
|
||||||
for _, el := range vec {
|
for _, el := range vec {
|
||||||
f := math.Floor(el.F*toNearestInverse+0.5) / toNearestInverse
|
f := math.Floor(el.F*toNearestInverse+0.5) / toNearestInverse
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: el.Metric.DropMetricName(),
|
Metric: el.Metric,
|
||||||
F: f,
|
F: f,
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
|
@ -882,9 +895,13 @@ func funcPresentOverTime(vals []parser.Value, args parser.Expressions, enh *Eval
|
||||||
func simpleFunc(vals []parser.Value, enh *EvalNodeHelper, f func(float64) float64) Vector {
|
func simpleFunc(vals []parser.Value, enh *EvalNodeHelper, f func(float64) float64) Vector {
|
||||||
for _, el := range vals[0].(Vector) {
|
for _, el := range vals[0].(Vector) {
|
||||||
if el.H == nil { // Process only float samples.
|
if el.H == nil { // Process only float samples.
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
el.Metric = el.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: el.Metric.DropMetricName(),
|
Metric: el.Metric,
|
||||||
F: f(el.F),
|
F: f(el.F),
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1028,9 +1045,13 @@ func funcSgn(vals []parser.Value, args parser.Expressions, enh *EvalNodeHelper)
|
||||||
func funcTimestamp(vals []parser.Value, args parser.Expressions, enh *EvalNodeHelper) (Vector, annotations.Annotations) {
|
func funcTimestamp(vals []parser.Value, args parser.Expressions, enh *EvalNodeHelper) (Vector, annotations.Annotations) {
|
||||||
vec := vals[0].(Vector)
|
vec := vals[0].(Vector)
|
||||||
for _, el := range vec {
|
for _, el := range vec {
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
el.Metric = el.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: el.Metric.DropMetricName(),
|
Metric: el.Metric,
|
||||||
F: float64(el.T) / 1000,
|
F: float64(el.T) / 1000,
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
|
@ -1137,9 +1158,13 @@ func funcHistogramCount(vals []parser.Value, args parser.Expressions, enh *EvalN
|
||||||
if sample.H == nil {
|
if sample.H == nil {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
sample.Metric = sample.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: sample.Metric.DropMetricName(),
|
Metric: sample.Metric,
|
||||||
F: sample.H.Count,
|
F: sample.H.Count,
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
|
@ -1154,9 +1179,13 @@ func funcHistogramSum(vals []parser.Value, args parser.Expressions, enh *EvalNod
|
||||||
if sample.H == nil {
|
if sample.H == nil {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
sample.Metric = sample.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: sample.Metric.DropMetricName(),
|
Metric: sample.Metric,
|
||||||
F: sample.H.Sum,
|
F: sample.H.Sum,
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
|
@ -1171,9 +1200,13 @@ func funcHistogramAvg(vals []parser.Value, args parser.Expressions, enh *EvalNod
|
||||||
if sample.H == nil {
|
if sample.H == nil {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
sample.Metric = sample.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: sample.Metric.DropMetricName(),
|
Metric: sample.Metric,
|
||||||
F: sample.H.Sum / sample.H.Count,
|
F: sample.H.Sum / sample.H.Count,
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
|
@ -1210,9 +1243,13 @@ func funcHistogramStdDev(vals []parser.Value, args parser.Expressions, enh *Eval
|
||||||
}
|
}
|
||||||
variance += cVariance
|
variance += cVariance
|
||||||
variance /= sample.H.Count
|
variance /= sample.H.Count
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
sample.Metric = sample.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: sample.Metric.DropMetricName(),
|
Metric: sample.Metric,
|
||||||
F: math.Sqrt(variance),
|
F: math.Sqrt(variance),
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
|
@ -1249,9 +1286,13 @@ func funcHistogramStdVar(vals []parser.Value, args parser.Expressions, enh *Eval
|
||||||
}
|
}
|
||||||
variance += cVariance
|
variance += cVariance
|
||||||
variance /= sample.H.Count
|
variance /= sample.H.Count
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
sample.Metric = sample.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: sample.Metric.DropMetricName(),
|
Metric: sample.Metric,
|
||||||
F: variance,
|
F: variance,
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
|
@ -1268,9 +1309,13 @@ func funcHistogramFraction(vals []parser.Value, args parser.Expressions, enh *Ev
|
||||||
if sample.H == nil {
|
if sample.H == nil {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
sample.Metric = sample.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: sample.Metric.DropMetricName(),
|
Metric: sample.Metric,
|
||||||
F: histogramFraction(lower, upper, sample.H),
|
F: histogramFraction(lower, upper, sample.H),
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out, nil
|
return enh.Out, nil
|
||||||
|
@ -1338,9 +1383,13 @@ func funcHistogramQuantile(vals []parser.Value, args parser.Expressions, enh *Ev
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
sample.Metric = sample.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: sample.Metric.DropMetricName(),
|
Metric: sample.Metric,
|
||||||
F: histogramQuantile(q, sample.H),
|
F: histogramQuantile(q, sample.H),
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1442,6 +1491,11 @@ func (ev *evaluator) evalLabelReplace(args parser.Expressions) (parser.Value, an
|
||||||
lb.Reset(el.Metric)
|
lb.Reset(el.Metric)
|
||||||
lb.Set(dst, string(res))
|
lb.Set(dst, string(res))
|
||||||
matrix[i].Metric = lb.Labels()
|
matrix[i].Metric = lb.Labels()
|
||||||
|
if dst == model.MetricNameLabel {
|
||||||
|
matrix[i].DropName = false
|
||||||
|
} else {
|
||||||
|
matrix[i].DropName = el.DropName
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if matrix.ContainsSameLabelset() {
|
if matrix.ContainsSameLabelset() {
|
||||||
|
@ -1496,6 +1550,12 @@ func (ev *evaluator) evalLabelJoin(args parser.Expressions) (parser.Value, annot
|
||||||
lb.Reset(el.Metric)
|
lb.Reset(el.Metric)
|
||||||
lb.Set(dst, strval)
|
lb.Set(dst, strval)
|
||||||
matrix[i].Metric = lb.Labels()
|
matrix[i].Metric = lb.Labels()
|
||||||
|
|
||||||
|
if dst == model.MetricNameLabel {
|
||||||
|
matrix[i].DropName = false
|
||||||
|
} else {
|
||||||
|
matrix[i].DropName = el.DropName
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return matrix, ws
|
return matrix, ws
|
||||||
|
@ -1518,9 +1578,13 @@ func dateWrapper(vals []parser.Value, enh *EvalNodeHelper, f func(time.Time) flo
|
||||||
|
|
||||||
for _, el := range vals[0].(Vector) {
|
for _, el := range vals[0].(Vector) {
|
||||||
t := time.Unix(int64(el.F), 0).UTC()
|
t := time.Unix(int64(el.F), 0).UTC()
|
||||||
|
if !enh.enableDelayedNameRemoval {
|
||||||
|
el.Metric = el.Metric.DropMetricName()
|
||||||
|
}
|
||||||
enh.Out = append(enh.Out, Sample{
|
enh.Out = append(enh.Out, Sample{
|
||||||
Metric: el.Metric.DropMetricName(),
|
Metric: el.Metric,
|
||||||
F: f(t),
|
F: f(t),
|
||||||
|
DropName: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return enh.Out
|
return enh.Out
|
||||||
|
|
|
@ -617,6 +617,16 @@ func lexBuckets(l *Lexer) stateFn {
|
||||||
l.bracketOpen = false
|
l.bracketOpen = false
|
||||||
l.emit(RIGHT_BRACKET)
|
l.emit(RIGHT_BRACKET)
|
||||||
return lexHistogram
|
return lexHistogram
|
||||||
|
case isAlpha(r):
|
||||||
|
// Current word is Inf or NaN.
|
||||||
|
word := l.input[l.start:l.pos]
|
||||||
|
if desc, ok := key[strings.ToLower(word)]; ok {
|
||||||
|
if desc == NUMBER {
|
||||||
|
l.emit(desc)
|
||||||
|
return lexStatements
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return lexBuckets
|
||||||
default:
|
default:
|
||||||
return l.errorf("invalid character in buckets description: %q", r)
|
return l.errorf("invalid character in buckets description: %q", r)
|
||||||
}
|
}
|
||||||
|
|
|
@ -639,6 +639,29 @@ var tests = []struct {
|
||||||
},
|
},
|
||||||
seriesDesc: true,
|
seriesDesc: true,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
input: `{} {{buckets: [Inf NaN] schema:1}}`,
|
||||||
|
expected: []Item{
|
||||||
|
{LEFT_BRACE, 0, `{`},
|
||||||
|
{RIGHT_BRACE, 1, `}`},
|
||||||
|
{SPACE, 2, ` `},
|
||||||
|
{OPEN_HIST, 3, `{{`},
|
||||||
|
{BUCKETS_DESC, 5, `buckets`},
|
||||||
|
{COLON, 12, `:`},
|
||||||
|
{SPACE, 13, ` `},
|
||||||
|
{LEFT_BRACKET, 14, `[`},
|
||||||
|
{NUMBER, 15, `Inf`},
|
||||||
|
{SPACE, 18, ` `},
|
||||||
|
{NUMBER, 19, `NaN`},
|
||||||
|
{RIGHT_BRACKET, 22, `]`},
|
||||||
|
{SPACE, 23, ` `},
|
||||||
|
{SCHEMA_DESC, 24, `schema`},
|
||||||
|
{COLON, 30, `:`},
|
||||||
|
{NUMBER, 31, `1`},
|
||||||
|
{CLOSE_HIST, 32, `}}`},
|
||||||
|
},
|
||||||
|
seriesDesc: true,
|
||||||
|
},
|
||||||
{ // Series with sum as -Inf and count as NaN.
|
{ // Series with sum as -Inf and count as NaN.
|
||||||
input: `{} {{buckets: [5 10 7] sum:Inf count:NaN}}`,
|
input: `{} {{buckets: [5 10 7] sum:Inf count:NaN}}`,
|
||||||
expected: []Item{
|
expected: []Item{
|
||||||
|
|
|
@ -88,7 +88,7 @@ func (node *AggregateExpr) getAggOpStr() string {
|
||||||
func joinLabels(ss []string) string {
|
func joinLabels(ss []string) string {
|
||||||
for i, s := range ss {
|
for i, s := range ss {
|
||||||
// If the label is already quoted, don't quote it again.
|
// If the label is already quoted, don't quote it again.
|
||||||
if s[0] != '"' && s[0] != '\'' && s[0] != '`' && !model.IsValidLegacyMetricName(model.LabelValue(s)) {
|
if s[0] != '"' && s[0] != '\'' && s[0] != '`' && !model.IsValidLegacyMetricName(string(model.LabelValue(s))) {
|
||||||
ss[i] = fmt.Sprintf("\"%s\"", s)
|
ss[i] = fmt.Sprintf("\"%s\"", s)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -90,6 +90,7 @@ func NewTestEngine(enablePerStepStats bool, lookbackDelta time.Duration, maxSamp
|
||||||
EnableNegativeOffset: true,
|
EnableNegativeOffset: true,
|
||||||
EnablePerStepStats: enablePerStepStats,
|
EnablePerStepStats: enablePerStepStats,
|
||||||
LookbackDelta: lookbackDelta,
|
LookbackDelta: lookbackDelta,
|
||||||
|
EnableDelayedNameRemoval: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -769,7 +770,7 @@ func (ev *evalCmd) compareResult(result parser.Value) error {
|
||||||
return fmt.Errorf("expected histogram value at index %v for %s to have timestamp %v, but it had timestamp %v (result has %s)", i, ev.metrics[hash], expected.T, actual.T, formatSeriesResult(s))
|
return fmt.Errorf("expected histogram value at index %v for %s to have timestamp %v, but it had timestamp %v (result has %s)", i, ev.metrics[hash], expected.T, actual.T, formatSeriesResult(s))
|
||||||
}
|
}
|
||||||
|
|
||||||
if !actual.H.Equals(expected.H.Compact(0)) {
|
if !compareNativeHistogram(expected.H.Compact(0), actual.H.Compact(0)) {
|
||||||
return fmt.Errorf("expected histogram value at index %v (t=%v) for %s to be %v, but got %v (result has %s)", i, actual.T, ev.metrics[hash], expected.H, actual.H, formatSeriesResult(s))
|
return fmt.Errorf("expected histogram value at index %v (t=%v) for %s to be %v, but got %v (result has %s)", i, actual.T, ev.metrics[hash], expected.H, actual.H, formatSeriesResult(s))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -804,7 +805,7 @@ func (ev *evalCmd) compareResult(result parser.Value) error {
|
||||||
if expH != nil && v.H == nil {
|
if expH != nil && v.H == nil {
|
||||||
return fmt.Errorf("expected histogram %s for %s but got float value %v", HistogramTestExpression(expH), v.Metric, v.F)
|
return fmt.Errorf("expected histogram %s for %s but got float value %v", HistogramTestExpression(expH), v.Metric, v.F)
|
||||||
}
|
}
|
||||||
if expH != nil && !expH.Compact(0).Equals(v.H) {
|
if expH != nil && !compareNativeHistogram(expH.Compact(0), v.H.Compact(0)) {
|
||||||
return fmt.Errorf("expected %v for %s but got %s", HistogramTestExpression(expH), v.Metric, HistogramTestExpression(v.H))
|
return fmt.Errorf("expected %v for %s but got %s", HistogramTestExpression(expH), v.Metric, HistogramTestExpression(v.H))
|
||||||
}
|
}
|
||||||
if !almost.Equal(exp0.Value, v.F, defaultEpsilon) {
|
if !almost.Equal(exp0.Value, v.F, defaultEpsilon) {
|
||||||
|
@ -837,6 +838,121 @@ func (ev *evalCmd) compareResult(result parser.Value) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// compareNativeHistogram is helper function to compare two native histograms
|
||||||
|
// which can tolerate some differ in the field of float type, such as Count, Sum.
|
||||||
|
func compareNativeHistogram(exp, cur *histogram.FloatHistogram) bool {
|
||||||
|
if exp == nil || cur == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if exp.Schema != cur.Schema ||
|
||||||
|
!almost.Equal(exp.Count, cur.Count, defaultEpsilon) ||
|
||||||
|
!almost.Equal(exp.Sum, cur.Sum, defaultEpsilon) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if exp.UsesCustomBuckets() {
|
||||||
|
if !histogram.FloatBucketsMatch(exp.CustomValues, cur.CustomValues) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if exp.ZeroThreshold != cur.ZeroThreshold ||
|
||||||
|
!almost.Equal(exp.ZeroCount, cur.ZeroCount, defaultEpsilon) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if !spansMatch(exp.NegativeSpans, cur.NegativeSpans) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if !floatBucketsMatch(exp.NegativeBuckets, cur.NegativeBuckets) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if !spansMatch(exp.PositiveSpans, cur.PositiveSpans) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if !floatBucketsMatch(exp.PositiveBuckets, cur.PositiveBuckets) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func floatBucketsMatch(b1, b2 []float64) bool {
|
||||||
|
if len(b1) != len(b2) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
for i, b := range b1 {
|
||||||
|
if !almost.Equal(b, b2[i], defaultEpsilon) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func spansMatch(s1, s2 []histogram.Span) bool {
|
||||||
|
if len(s1) == 0 && len(s2) == 0 {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
s1idx, s2idx := 0, 0
|
||||||
|
for {
|
||||||
|
if s1idx >= len(s1) {
|
||||||
|
return allEmptySpans(s2[s2idx:])
|
||||||
|
}
|
||||||
|
if s2idx >= len(s2) {
|
||||||
|
return allEmptySpans(s1[s1idx:])
|
||||||
|
}
|
||||||
|
|
||||||
|
currS1, currS2 := s1[s1idx], s2[s2idx]
|
||||||
|
s1idx++
|
||||||
|
s2idx++
|
||||||
|
if currS1.Length == 0 {
|
||||||
|
// This span is zero length, so we add consecutive such spans
|
||||||
|
// until we find a non-zero span.
|
||||||
|
for ; s1idx < len(s1) && s1[s1idx].Length == 0; s1idx++ {
|
||||||
|
currS1.Offset += s1[s1idx].Offset
|
||||||
|
}
|
||||||
|
if s1idx < len(s1) {
|
||||||
|
currS1.Offset += s1[s1idx].Offset
|
||||||
|
currS1.Length = s1[s1idx].Length
|
||||||
|
s1idx++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if currS2.Length == 0 {
|
||||||
|
// This span is zero length, so we add consecutive such spans
|
||||||
|
// until we find a non-zero span.
|
||||||
|
for ; s2idx < len(s2) && s2[s2idx].Length == 0; s2idx++ {
|
||||||
|
currS2.Offset += s2[s2idx].Offset
|
||||||
|
}
|
||||||
|
if s2idx < len(s2) {
|
||||||
|
currS2.Offset += s2[s2idx].Offset
|
||||||
|
currS2.Length = s2[s2idx].Length
|
||||||
|
s2idx++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if currS1.Length == 0 && currS2.Length == 0 {
|
||||||
|
// The last spans of both set are zero length. Previous spans match.
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if currS1.Offset != currS2.Offset || currS1.Length != currS2.Length {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func allEmptySpans(s []histogram.Span) bool {
|
||||||
|
for _, ss := range s {
|
||||||
|
if ss.Length > 0 {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
func (ev *evalCmd) checkExpectedFailure(actual error) error {
|
func (ev *evalCmd) checkExpectedFailure(actual error) error {
|
||||||
if ev.expectedFailMessage != "" {
|
if ev.expectedFailMessage != "" {
|
||||||
if ev.expectedFailMessage != actual.Error() {
|
if ev.expectedFailMessage != actual.Error() {
|
||||||
|
@ -1247,6 +1363,7 @@ func (ll *LazyLoader) clear() error {
|
||||||
NoStepSubqueryIntervalFn: func(int64) int64 { return durationMilliseconds(ll.SubqueryInterval) },
|
NoStepSubqueryIntervalFn: func(int64) int64 { return durationMilliseconds(ll.SubqueryInterval) },
|
||||||
EnableAtModifier: ll.opts.EnableAtModifier,
|
EnableAtModifier: ll.opts.EnableAtModifier,
|
||||||
EnableNegativeOffset: ll.opts.EnableNegativeOffset,
|
EnableNegativeOffset: ll.opts.EnableNegativeOffset,
|
||||||
|
EnableDelayedNameRemoval: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
ll.queryEngine = promql.NewEngine(opts)
|
ll.queryEngine = promql.NewEngine(opts)
|
||||||
|
|
84
promql/promqltest/testdata/name_label_dropping.test
vendored
Normal file
84
promql/promqltest/testdata/name_label_dropping.test
vendored
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
# Test for __name__ label drop.
|
||||||
|
load 5m
|
||||||
|
metric{env="1"} 0 60 120
|
||||||
|
another_metric{env="1"} 60 120 180
|
||||||
|
|
||||||
|
# Does not drop __name__ for vector selector
|
||||||
|
eval instant at 15m metric{env="1"}
|
||||||
|
metric{env="1"} 120
|
||||||
|
|
||||||
|
# Drops __name__ for unary operators
|
||||||
|
eval instant at 15m -metric
|
||||||
|
{env="1"} -120
|
||||||
|
|
||||||
|
# Drops __name__ for binary operators
|
||||||
|
eval instant at 15m metric + another_metric
|
||||||
|
{env="1"} 300
|
||||||
|
|
||||||
|
# Does not drop __name__ for binary comparison operators
|
||||||
|
eval instant at 15m metric <= another_metric
|
||||||
|
metric{env="1"} 120
|
||||||
|
|
||||||
|
# Drops __name__ for binary comparison operators with "bool" modifier
|
||||||
|
eval instant at 15m metric <= bool another_metric
|
||||||
|
{env="1"} 1
|
||||||
|
|
||||||
|
# Drops __name__ for vector-scalar operations
|
||||||
|
eval instant at 15m metric * 2
|
||||||
|
{env="1"} 240
|
||||||
|
|
||||||
|
# Drops __name__ for instant-vector functions
|
||||||
|
eval instant at 15m clamp(metric, 0, 100)
|
||||||
|
{env="1"} 100
|
||||||
|
|
||||||
|
# Drops __name__ for range-vector functions
|
||||||
|
eval instant at 15m rate(metric{env="1"}[10m])
|
||||||
|
{env="1"} 0.2
|
||||||
|
|
||||||
|
# Does not drop __name__ for last_over_time function
|
||||||
|
eval instant at 15m last_over_time(metric{env="1"}[10m])
|
||||||
|
metric{env="1"} 120
|
||||||
|
|
||||||
|
# Drops name for other _over_time functions
|
||||||
|
eval instant at 15m max_over_time(metric{env="1"}[10m])
|
||||||
|
{env="1"} 120
|
||||||
|
|
||||||
|
# Allows relabeling (to-be-dropped) __name__ via label_replace
|
||||||
|
eval instant at 15m label_replace(rate({env="1"}[10m]), "my_name", "rate_$1", "__name__", "(.+)")
|
||||||
|
{my_name="rate_metric", env="1"} 0.2
|
||||||
|
{my_name="rate_another_metric", env="1"} 0.2
|
||||||
|
|
||||||
|
# Allows preserving __name__ via label_replace
|
||||||
|
eval instant at 15m label_replace(rate({env="1"}[10m]), "__name__", "rate_$1", "__name__", "(.+)")
|
||||||
|
rate_metric{env="1"} 0.2
|
||||||
|
rate_another_metric{env="1"} 0.2
|
||||||
|
|
||||||
|
# Allows relabeling (to-be-dropped) __name__ via label_join
|
||||||
|
eval instant at 15m label_join(rate({env="1"}[10m]), "my_name", "_", "__name__")
|
||||||
|
{my_name="metric", env="1"} 0.2
|
||||||
|
{my_name="another_metric", env="1"} 0.2
|
||||||
|
|
||||||
|
# Allows preserving __name__ via label_join
|
||||||
|
eval instant at 15m label_join(rate({env="1"}[10m]), "__name__", "_", "__name__", "env")
|
||||||
|
metric_1{env="1"} 0.2
|
||||||
|
another_metric_1{env="1"} 0.2
|
||||||
|
|
||||||
|
# Does not drop metric names fro aggregation operators
|
||||||
|
eval instant at 15m sum by (__name__, env) (metric{env="1"})
|
||||||
|
metric{env="1"} 120
|
||||||
|
|
||||||
|
# Aggregation operators by __name__ lead to duplicate labelset errors (aggregation is partitioned by not yet removed __name__ label)
|
||||||
|
# This is an accidental side effect of delayed __name__ label dropping
|
||||||
|
eval_fail instant at 15m sum by (__name__) (rate({env="1"}[10m]))
|
||||||
|
|
||||||
|
# Aggregation operators aggregate metrics with same labelset and to-be-dropped names
|
||||||
|
# This is an accidental side effect of delayed __name__ label dropping
|
||||||
|
eval instant at 15m sum(rate({env="1"}[10m])) by (env)
|
||||||
|
{env="1"} 0.4
|
||||||
|
|
||||||
|
# Aggregationk operators propagate __name__ label dropping information
|
||||||
|
eval instant at 15m topk(10, sum by (__name__, env) (metric{env="1"}))
|
||||||
|
metric{env="1"} 120
|
||||||
|
|
||||||
|
eval instant at 15m topk(10, sum by (__name__, env) (rate(metric{env="1"}[10m])))
|
||||||
|
{env="1"} 0.2
|
|
@ -718,6 +718,52 @@ eval instant at 10m histogram_fraction(-Inf, +Inf, histogram_fraction_4)
|
||||||
eval instant at 10m histogram_sum(scalar(histogram_fraction(-Inf, +Inf, sum(histogram_fraction_4))) * histogram_fraction_4)
|
eval instant at 10m histogram_sum(scalar(histogram_fraction(-Inf, +Inf, sum(histogram_fraction_4))) * histogram_fraction_4)
|
||||||
{} 100
|
{} 100
|
||||||
|
|
||||||
|
# Apply multiplication and division operator to histogram.
|
||||||
|
load 10m
|
||||||
|
histogram_mul_div {{schema:0 count:21 sum:33 z_bucket:3 z_bucket_w:0.001 buckets:[3 3 3] n_buckets:[3 3 3]}}x1
|
||||||
|
float_series_3 3+0x1
|
||||||
|
float_series_0 0+0x1
|
||||||
|
|
||||||
|
eval instant at 10m histogram_mul_div*3
|
||||||
|
{} {{schema:0 count:63 sum:99 z_bucket:9 z_bucket_w:0.001 buckets:[9 9 9] n_buckets:[9 9 9]}}
|
||||||
|
|
||||||
|
eval instant at 10m 3*histogram_mul_div
|
||||||
|
{} {{schema:0 count:63 sum:99 z_bucket:9 z_bucket_w:0.001 buckets:[9 9 9] n_buckets:[9 9 9]}}
|
||||||
|
|
||||||
|
eval instant at 10m histogram_mul_div*float_series_3
|
||||||
|
{} {{schema:0 count:63 sum:99 z_bucket:9 z_bucket_w:0.001 buckets:[9 9 9] n_buckets:[9 9 9]}}
|
||||||
|
|
||||||
|
eval instant at 10m float_series_3*histogram_mul_div
|
||||||
|
{} {{schema:0 count:63 sum:99 z_bucket:9 z_bucket_w:0.001 buckets:[9 9 9] n_buckets:[9 9 9]}}
|
||||||
|
|
||||||
|
eval instant at 10m histogram_mul_div/3
|
||||||
|
{} {{schema:0 count:7 sum:11 z_bucket:1 z_bucket_w:0.001 buckets:[1 1 1] n_buckets:[1 1 1]}}
|
||||||
|
|
||||||
|
eval instant at 10m histogram_mul_div/float_series_3
|
||||||
|
{} {{schema:0 count:7 sum:11 z_bucket:1 z_bucket_w:0.001 buckets:[1 1 1] n_buckets:[1 1 1]}}
|
||||||
|
|
||||||
|
eval instant at 10m histogram_mul_div*0
|
||||||
|
{} {{schema:0 count:0 sum:0 z_bucket:0 z_bucket_w:0.001 buckets:[0 0 0] n_buckets:[0 0 0]}}
|
||||||
|
|
||||||
|
eval instant at 10m 0*histogram_mul_div
|
||||||
|
{} {{schema:0 count:0 sum:0 z_bucket:0 z_bucket_w:0.001 buckets:[0 0 0] n_buckets:[0 0 0]}}
|
||||||
|
|
||||||
|
eval instant at 10m histogram_mul_div*float_series_0
|
||||||
|
{} {{schema:0 count:0 sum:0 z_bucket:0 z_bucket_w:0.001 buckets:[0 0 0] n_buckets:[0 0 0]}}
|
||||||
|
|
||||||
|
eval instant at 10m float_series_0*histogram_mul_div
|
||||||
|
{} {{schema:0 count:0 sum:0 z_bucket:0 z_bucket_w:0.001 buckets:[0 0 0] n_buckets:[0 0 0]}}
|
||||||
|
|
||||||
|
# TODO: (NeerajGartia21) remove all the histogram buckets in case of division with zero. See: https://github.com/prometheus/prometheus/issues/13934
|
||||||
|
eval instant at 10m histogram_mul_div/0
|
||||||
|
{} {{schema:0 count:Inf sum:Inf z_bucket:Inf z_bucket_w:0.001 buckets:[Inf Inf Inf] n_buckets:[Inf Inf Inf]}}
|
||||||
|
|
||||||
|
eval instant at 10m histogram_mul_div/float_series_0
|
||||||
|
{} {{schema:0 count:Inf sum:Inf z_bucket:Inf z_bucket_w:0.001 buckets:[Inf Inf Inf] n_buckets:[Inf Inf Inf]}}
|
||||||
|
|
||||||
|
eval instant at 10m histogram_mul_div*0/0
|
||||||
|
{} {{schema:0 count:NaN sum:NaN z_bucket:NaN z_bucket_w:0.001 buckets:[NaN NaN NaN] n_buckets:[NaN NaN NaN]}}
|
||||||
|
|
||||||
clear
|
clear
|
||||||
|
|
||||||
# Counter reset only noticeable in a single bucket.
|
# Counter reset only noticeable in a single bucket.
|
||||||
|
@ -886,3 +932,39 @@ eval_warn instant at 0 sum by (group) (metric)
|
||||||
{group="just-floats"} 5
|
{group="just-floats"} 5
|
||||||
{group="just-exponential-histograms"} {{sum:5 count:7 buckets:[2 3 2]}}
|
{group="just-exponential-histograms"} {{sum:5 count:7 buckets:[2 3 2]}}
|
||||||
{group="just-custom-histograms"} {{schema:-53 sum:4 count:5 custom_values:[2] buckets:[8]}}
|
{group="just-custom-histograms"} {{schema:-53 sum:4 count:5 custom_values:[2] buckets:[8]}}
|
||||||
|
|
||||||
|
clear
|
||||||
|
|
||||||
|
# Test native histograms with sum, count, avg.
|
||||||
|
load 10m
|
||||||
|
histogram_sum{idx="0"} {{schema:0 count:25 sum:1234.5 z_bucket:4 z_bucket_w:0.001 buckets:[1 2 0 1 1] n_buckets:[2 4 0 0 1 9]}}x1
|
||||||
|
histogram_sum{idx="1"} {{schema:0 count:41 sum:2345.6 z_bucket:5 z_bucket_w:0.001 buckets:[1 3 1 2 1 1 1] n_buckets:[0 1 4 2 7 0 0 0 0 5 5 2]}}x1
|
||||||
|
histogram_sum{idx="2"} {{schema:0 count:41 sum:1111.1 z_bucket:5 z_bucket_w:0.001 buckets:[1 3 1 2 1 1 1] n_buckets:[0 1 4 2 7 0 0 0 0 5 5 2]}}x1
|
||||||
|
histogram_sum{idx="3"} {{schema:1 count:0}}x1
|
||||||
|
histogram_sum_float{idx="0"} 42.0x1
|
||||||
|
|
||||||
|
eval instant at 10m sum(histogram_sum)
|
||||||
|
{} {{schema:0 count:107 sum:4691.2 z_bucket:14 z_bucket_w:0.001 buckets:[3 8 2 5 3 2 2] n_buckets:[2 6 8 4 15 9 0 0 0 10 10 4]}}
|
||||||
|
|
||||||
|
eval_warn instant at 10m sum({idx="0"})
|
||||||
|
|
||||||
|
eval instant at 10m sum(histogram_sum{idx="0"} + ignoring(idx) histogram_sum{idx="1"} + ignoring(idx) histogram_sum{idx="2"} + ignoring(idx) histogram_sum{idx="3"})
|
||||||
|
{} {{schema:0 count:107 sum:4691.2 z_bucket:14 z_bucket_w:0.001 buckets:[3 8 2 5 3 2 2] n_buckets:[2 6 8 4 15 9 0 0 0 10 10 4]}}
|
||||||
|
|
||||||
|
eval instant at 10m count(histogram_sum)
|
||||||
|
{} 4
|
||||||
|
|
||||||
|
eval instant at 10m avg(histogram_sum)
|
||||||
|
{} {{schema:0 count:26.75 sum:1172.8 z_bucket:3.5 z_bucket_w:0.001 buckets:[0.75 2 0.5 1.25 0.75 0.5 0.5] n_buckets:[0.5 1.5 2 1 3.75 2.25 0 0 0 2.5 2.5 1]}}
|
||||||
|
|
||||||
|
clear
|
||||||
|
|
||||||
|
# Test native histograms with sum_over_time, avg_over_time.
|
||||||
|
load 1m
|
||||||
|
histogram_sum_over_time {{schema:0 count:25 sum:1234.5 z_bucket:4 z_bucket_w:0.001 buckets:[1 2 0 1 1] n_buckets:[2 4 0 0 1 9]}} {{schema:0 count:41 sum:2345.6 z_bucket:5 z_bucket_w:0.001 buckets:[1 3 1 2 1 1 1] n_buckets:[0 1 4 2 7 0 0 0 0 5 5 2]}} {{schema:0 count:41 sum:1111.1 z_bucket:5 z_bucket_w:0.001 buckets:[1 3 1 2 1 1 1] n_buckets:[0 1 4 2 7 0 0 0 0 5 5 2]}} {{schema:1 count:0}}
|
||||||
|
|
||||||
|
eval instant at 3m sum_over_time(histogram_sum_over_time[3m:1m])
|
||||||
|
{} {{schema:0 count:107 sum:4691.2 z_bucket:14 z_bucket_w:0.001 buckets:[3 8 2 5 3 2 2] n_buckets:[2 6 8 4 15 9 0 0 0 10 10 4]}}
|
||||||
|
|
||||||
|
eval instant at 3m avg_over_time(histogram_sum_over_time[3m:1m])
|
||||||
|
{} {{schema:0 count:26.75 sum:1172.8 z_bucket:3.5 z_bucket_w:0.001 buckets:[0.75 2 0.5 1.25 0.75 0.5 0.5] n_buckets:[0.5 1.5 2 1 3.75 2.25 0 0 0 2.5 2.5 1]}}
|
||||||
|
|
|
@ -68,6 +68,9 @@ type Series struct {
|
||||||
Metric labels.Labels `json:"metric"`
|
Metric labels.Labels `json:"metric"`
|
||||||
Floats []FPoint `json:"values,omitempty"`
|
Floats []FPoint `json:"values,omitempty"`
|
||||||
Histograms []HPoint `json:"histograms,omitempty"`
|
Histograms []HPoint `json:"histograms,omitempty"`
|
||||||
|
// DropName is used to indicate whether the __name__ label should be dropped
|
||||||
|
// as part of the query evaluation.
|
||||||
|
DropName bool `json:"-"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s Series) String() string {
|
func (s Series) String() string {
|
||||||
|
@ -194,6 +197,9 @@ type Sample struct {
|
||||||
H *histogram.FloatHistogram
|
H *histogram.FloatHistogram
|
||||||
|
|
||||||
Metric labels.Labels
|
Metric labels.Labels
|
||||||
|
// DropName is used to indicate whether the __name__ label should be dropped
|
||||||
|
// as part of the query evaluation.
|
||||||
|
DropName bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s Sample) String() string {
|
func (s Sample) String() string {
|
||||||
|
|
|
@ -648,7 +648,7 @@ func TestAlertingRuleLimit(t *testing.T) {
|
||||||
case err != nil:
|
case err != nil:
|
||||||
require.EqualError(t, err, test.err)
|
require.EqualError(t, err, test.err)
|
||||||
case test.err != "":
|
case test.err != "":
|
||||||
t.Errorf("Expected errror %s, got none", test.err)
|
t.Errorf("Expected error %s, got none", test.err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -142,7 +142,7 @@ func (m *Manager) UnregisterMetrics() {
|
||||||
|
|
||||||
func (m *Manager) reloader() {
|
func (m *Manager) reloader() {
|
||||||
reloadIntervalDuration := m.opts.DiscoveryReloadInterval
|
reloadIntervalDuration := m.opts.DiscoveryReloadInterval
|
||||||
if reloadIntervalDuration < model.Duration(5*time.Second) {
|
if reloadIntervalDuration == model.Duration(0) {
|
||||||
reloadIntervalDuration = model.Duration(5 * time.Second)
|
reloadIntervalDuration = model.Duration(5 * time.Second)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -20,6 +20,7 @@ import (
|
||||||
"net/http/httptest"
|
"net/http/httptest"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
"sync"
|
"sync"
|
||||||
"testing"
|
"testing"
|
||||||
|
@ -36,6 +37,7 @@ import (
|
||||||
|
|
||||||
"github.com/prometheus/prometheus/config"
|
"github.com/prometheus/prometheus/config"
|
||||||
"github.com/prometheus/prometheus/discovery"
|
"github.com/prometheus/prometheus/discovery"
|
||||||
|
_ "github.com/prometheus/prometheus/discovery/file"
|
||||||
"github.com/prometheus/prometheus/discovery/targetgroup"
|
"github.com/prometheus/prometheus/discovery/targetgroup"
|
||||||
"github.com/prometheus/prometheus/model/labels"
|
"github.com/prometheus/prometheus/model/labels"
|
||||||
"github.com/prometheus/prometheus/model/relabel"
|
"github.com/prometheus/prometheus/model/relabel"
|
||||||
|
@ -722,8 +724,6 @@ func TestManagerCTZeroIngestion(t *testing.T) {
|
||||||
name string
|
name string
|
||||||
counterSample *dto.Counter
|
counterSample *dto.Counter
|
||||||
enableCTZeroIngestion bool
|
enableCTZeroIngestion bool
|
||||||
|
|
||||||
expectedValues []float64
|
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
name: "disabled with CT on counter",
|
name: "disabled with CT on counter",
|
||||||
|
@ -732,7 +732,6 @@ func TestManagerCTZeroIngestion(t *testing.T) {
|
||||||
// Timestamp does not matter as long as it exists in this test.
|
// Timestamp does not matter as long as it exists in this test.
|
||||||
CreatedTimestamp: timestamppb.Now(),
|
CreatedTimestamp: timestamppb.Now(),
|
||||||
},
|
},
|
||||||
expectedValues: []float64{1.0},
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "enabled with CT on counter",
|
name: "enabled with CT on counter",
|
||||||
|
@ -742,7 +741,6 @@ func TestManagerCTZeroIngestion(t *testing.T) {
|
||||||
CreatedTimestamp: timestamppb.Now(),
|
CreatedTimestamp: timestamppb.Now(),
|
||||||
},
|
},
|
||||||
enableCTZeroIngestion: true,
|
enableCTZeroIngestion: true,
|
||||||
expectedValues: []float64{0.0, 1.0},
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "enabled without CT on counter",
|
name: "enabled without CT on counter",
|
||||||
|
@ -750,7 +748,6 @@ func TestManagerCTZeroIngestion(t *testing.T) {
|
||||||
Value: proto.Float64(1.0),
|
Value: proto.Float64(1.0),
|
||||||
},
|
},
|
||||||
enableCTZeroIngestion: true,
|
enableCTZeroIngestion: true,
|
||||||
expectedValues: []float64{1.0},
|
|
||||||
},
|
},
|
||||||
} {
|
} {
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
@ -817,46 +814,44 @@ func TestManagerCTZeroIngestion(t *testing.T) {
|
||||||
})
|
})
|
||||||
scrapeManager.reload()
|
scrapeManager.reload()
|
||||||
|
|
||||||
|
var got []float64
|
||||||
// Wait for one scrape.
|
// Wait for one scrape.
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Minute)
|
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Minute)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
require.NoError(t, runutil.Retry(100*time.Millisecond, ctx.Done(), func() error {
|
require.NoError(t, runutil.Retry(100*time.Millisecond, ctx.Done(), func() error {
|
||||||
if countFloatSamples(app, mName) != len(tc.expectedValues) {
|
app.mtx.Lock()
|
||||||
return fmt.Errorf("expected %v samples", tc.expectedValues)
|
defer app.mtx.Unlock()
|
||||||
|
|
||||||
|
// Check if scrape happened and grab the relevant samples, they have to be there - or it's a bug
|
||||||
|
// and it's not worth waiting.
|
||||||
|
for _, f := range app.resultFloats {
|
||||||
|
if f.metric.Get(model.MetricNameLabel) == mName {
|
||||||
|
got = append(got, f.f)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return nil
|
if len(app.resultFloats) > 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return fmt.Errorf("expected some samples, got none")
|
||||||
}), "after 1 minute")
|
}), "after 1 minute")
|
||||||
scrapeManager.Stop()
|
scrapeManager.Stop()
|
||||||
|
|
||||||
require.Equal(t, tc.expectedValues, getResultFloats(app, mName))
|
// Check for zero samples, assuming we only injected always one sample.
|
||||||
|
// Did it contain CT to inject? If yes, was CT zero enabled?
|
||||||
|
if tc.counterSample.CreatedTimestamp.IsValid() && tc.enableCTZeroIngestion {
|
||||||
|
require.Len(t, got, 2)
|
||||||
|
require.Equal(t, 0.0, got[0])
|
||||||
|
require.Equal(t, tc.counterSample.GetValue(), got[1])
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Expect only one, valid sample.
|
||||||
|
require.Len(t, got, 1)
|
||||||
|
require.Equal(t, tc.counterSample.GetValue(), got[0])
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func countFloatSamples(a *collectResultAppender, expectedMetricName string) (count int) {
|
|
||||||
a.mtx.Lock()
|
|
||||||
defer a.mtx.Unlock()
|
|
||||||
|
|
||||||
for _, f := range a.resultFloats {
|
|
||||||
if f.metric.Get(model.MetricNameLabel) == expectedMetricName {
|
|
||||||
count++
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return count
|
|
||||||
}
|
|
||||||
|
|
||||||
func getResultFloats(app *collectResultAppender, expectedMetricName string) (result []float64) {
|
|
||||||
app.mtx.Lock()
|
|
||||||
defer app.mtx.Unlock()
|
|
||||||
|
|
||||||
for _, f := range app.resultFloats {
|
|
||||||
if f.metric.Get(model.MetricNameLabel) == expectedMetricName {
|
|
||||||
result = append(result, f.f)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestUnregisterMetrics(t *testing.T) {
|
func TestUnregisterMetrics(t *testing.T) {
|
||||||
reg := prometheus.NewRegistry()
|
reg := prometheus.NewRegistry()
|
||||||
// Check that all metrics can be unregistered, allowing a second manager to be created.
|
// Check that all metrics can be unregistered, allowing a second manager to be created.
|
||||||
|
@ -869,3 +864,414 @@ func TestUnregisterMetrics(t *testing.T) {
|
||||||
manager.UnregisterMetrics()
|
manager.UnregisterMetrics()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func applyConfig(
|
||||||
|
t *testing.T,
|
||||||
|
config string,
|
||||||
|
scrapeManager *Manager,
|
||||||
|
discoveryManager *discovery.Manager,
|
||||||
|
) {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
cfg := loadConfiguration(t, config)
|
||||||
|
require.NoError(t, scrapeManager.ApplyConfig(cfg))
|
||||||
|
|
||||||
|
c := make(map[string]discovery.Configs)
|
||||||
|
scfgs, err := cfg.GetScrapeConfigs()
|
||||||
|
require.NoError(t, err)
|
||||||
|
for _, v := range scfgs {
|
||||||
|
c[v.JobName] = v.ServiceDiscoveryConfigs
|
||||||
|
}
|
||||||
|
require.NoError(t, discoveryManager.ApplyConfig(c))
|
||||||
|
}
|
||||||
|
|
||||||
|
func runManagers(t *testing.T, ctx context.Context) (*discovery.Manager, *Manager) {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
reg := prometheus.NewRegistry()
|
||||||
|
sdMetrics, err := discovery.RegisterSDMetrics(reg, discovery.NewRefreshMetrics(reg))
|
||||||
|
require.NoError(t, err)
|
||||||
|
discoveryManager := discovery.NewManager(
|
||||||
|
ctx,
|
||||||
|
log.NewNopLogger(),
|
||||||
|
reg,
|
||||||
|
sdMetrics,
|
||||||
|
discovery.Updatert(100*time.Millisecond),
|
||||||
|
)
|
||||||
|
scrapeManager, err := NewManager(
|
||||||
|
&Options{DiscoveryReloadInterval: model.Duration(100 * time.Millisecond)},
|
||||||
|
nil,
|
||||||
|
nopAppendable{},
|
||||||
|
prometheus.NewRegistry(),
|
||||||
|
)
|
||||||
|
require.NoError(t, err)
|
||||||
|
go discoveryManager.Run()
|
||||||
|
go scrapeManager.Run(discoveryManager.SyncCh())
|
||||||
|
return discoveryManager, scrapeManager
|
||||||
|
}
|
||||||
|
|
||||||
|
func writeIntoFile(t *testing.T, content, filePattern string) *os.File {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
file, err := os.CreateTemp("", filePattern)
|
||||||
|
require.NoError(t, err)
|
||||||
|
_, err = file.WriteString(content)
|
||||||
|
require.NoError(t, err)
|
||||||
|
return file
|
||||||
|
}
|
||||||
|
|
||||||
|
func requireTargets(
|
||||||
|
t *testing.T,
|
||||||
|
scrapeManager *Manager,
|
||||||
|
jobName string,
|
||||||
|
waitToAppear bool,
|
||||||
|
expectedTargets []string,
|
||||||
|
) {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
require.Eventually(t, func() bool {
|
||||||
|
targets, ok := scrapeManager.TargetsActive()[jobName]
|
||||||
|
if !ok {
|
||||||
|
if waitToAppear {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
t.Fatalf("job %s shouldn't be dropped", jobName)
|
||||||
|
}
|
||||||
|
if expectedTargets == nil {
|
||||||
|
return targets == nil
|
||||||
|
}
|
||||||
|
if len(targets) != len(expectedTargets) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
sTargets := []string{}
|
||||||
|
for _, t := range targets {
|
||||||
|
sTargets = append(sTargets, t.String())
|
||||||
|
}
|
||||||
|
sort.Strings(expectedTargets)
|
||||||
|
sort.Strings(sTargets)
|
||||||
|
for i, t := range sTargets {
|
||||||
|
if t != expectedTargets[i] {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}, 1*time.Second, 100*time.Millisecond)
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestTargetDisappearsAfterProviderRemoved makes sure that when a provider is dropped, (only) its targets are dropped.
|
||||||
|
func TestTargetDisappearsAfterProviderRemoved(t *testing.T) {
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
myJob := "my-job"
|
||||||
|
myJobSDTargetURL := "my:9876"
|
||||||
|
myJobStaticTargetURL := "my:5432"
|
||||||
|
|
||||||
|
sdFileContent := fmt.Sprintf(`[{"targets": ["%s"]}]`, myJobSDTargetURL)
|
||||||
|
sDFile := writeIntoFile(t, sdFileContent, "*targets.json")
|
||||||
|
|
||||||
|
baseConfig := `
|
||||||
|
scrape_configs:
|
||||||
|
- job_name: %s
|
||||||
|
static_configs:
|
||||||
|
- targets: ['%s']
|
||||||
|
file_sd_configs:
|
||||||
|
- files: ['%s']
|
||||||
|
`
|
||||||
|
|
||||||
|
discoveryManager, scrapeManager := runManagers(t, ctx)
|
||||||
|
defer scrapeManager.Stop()
|
||||||
|
|
||||||
|
applyConfig(
|
||||||
|
t,
|
||||||
|
fmt.Sprintf(
|
||||||
|
baseConfig,
|
||||||
|
myJob,
|
||||||
|
myJobStaticTargetURL,
|
||||||
|
sDFile.Name(),
|
||||||
|
),
|
||||||
|
scrapeManager,
|
||||||
|
discoveryManager,
|
||||||
|
)
|
||||||
|
// Make sure the jobs targets are taken into account
|
||||||
|
requireTargets(
|
||||||
|
t,
|
||||||
|
scrapeManager,
|
||||||
|
myJob,
|
||||||
|
true,
|
||||||
|
[]string{
|
||||||
|
fmt.Sprintf("http://%s/metrics", myJobSDTargetURL),
|
||||||
|
fmt.Sprintf("http://%s/metrics", myJobStaticTargetURL),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
// Apply a new config where a provider is removed
|
||||||
|
baseConfig = `
|
||||||
|
scrape_configs:
|
||||||
|
- job_name: %s
|
||||||
|
static_configs:
|
||||||
|
- targets: ['%s']
|
||||||
|
`
|
||||||
|
applyConfig(
|
||||||
|
t,
|
||||||
|
fmt.Sprintf(
|
||||||
|
baseConfig,
|
||||||
|
myJob,
|
||||||
|
myJobStaticTargetURL,
|
||||||
|
),
|
||||||
|
scrapeManager,
|
||||||
|
discoveryManager,
|
||||||
|
)
|
||||||
|
// Make sure the corresponding target was dropped
|
||||||
|
requireTargets(
|
||||||
|
t,
|
||||||
|
scrapeManager,
|
||||||
|
myJob,
|
||||||
|
false,
|
||||||
|
[]string{
|
||||||
|
fmt.Sprintf("http://%s/metrics", myJobStaticTargetURL),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
// Apply a new config with no providers
|
||||||
|
baseConfig = `
|
||||||
|
scrape_configs:
|
||||||
|
- job_name: %s
|
||||||
|
`
|
||||||
|
applyConfig(
|
||||||
|
t,
|
||||||
|
fmt.Sprintf(
|
||||||
|
baseConfig,
|
||||||
|
myJob,
|
||||||
|
),
|
||||||
|
scrapeManager,
|
||||||
|
discoveryManager,
|
||||||
|
)
|
||||||
|
// Make sure the corresponding target was dropped
|
||||||
|
requireTargets(
|
||||||
|
t,
|
||||||
|
scrapeManager,
|
||||||
|
myJob,
|
||||||
|
false,
|
||||||
|
nil,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestOnlyProviderStaleTargetsAreDropped makes sure that when a job has only one provider with multiple targets
|
||||||
|
// and when the provider can no longer discover some of those targets, only those stale targets are dropped.
|
||||||
|
func TestOnlyProviderStaleTargetsAreDropped(t *testing.T) {
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
jobName := "my-job"
|
||||||
|
jobTarget1URL := "foo:9876"
|
||||||
|
jobTarget2URL := "foo:5432"
|
||||||
|
|
||||||
|
sdFile1Content := fmt.Sprintf(`[{"targets": ["%s"]}]`, jobTarget1URL)
|
||||||
|
sdFile2Content := fmt.Sprintf(`[{"targets": ["%s"]}]`, jobTarget2URL)
|
||||||
|
sDFile1 := writeIntoFile(t, sdFile1Content, "*targets.json")
|
||||||
|
sDFile2 := writeIntoFile(t, sdFile2Content, "*targets.json")
|
||||||
|
|
||||||
|
baseConfig := `
|
||||||
|
scrape_configs:
|
||||||
|
- job_name: %s
|
||||||
|
file_sd_configs:
|
||||||
|
- files: ['%s', '%s']
|
||||||
|
`
|
||||||
|
discoveryManager, scrapeManager := runManagers(t, ctx)
|
||||||
|
defer scrapeManager.Stop()
|
||||||
|
|
||||||
|
applyConfig(
|
||||||
|
t,
|
||||||
|
fmt.Sprintf(baseConfig, jobName, sDFile1.Name(), sDFile2.Name()),
|
||||||
|
scrapeManager,
|
||||||
|
discoveryManager,
|
||||||
|
)
|
||||||
|
|
||||||
|
// Make sure the job's targets are taken into account
|
||||||
|
requireTargets(
|
||||||
|
t,
|
||||||
|
scrapeManager,
|
||||||
|
jobName,
|
||||||
|
true,
|
||||||
|
[]string{
|
||||||
|
fmt.Sprintf("http://%s/metrics", jobTarget1URL),
|
||||||
|
fmt.Sprintf("http://%s/metrics", jobTarget2URL),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
// Apply the same config for the same job but with a non existing file to make the provider
|
||||||
|
// unable to discover some targets
|
||||||
|
applyConfig(
|
||||||
|
t,
|
||||||
|
fmt.Sprintf(baseConfig, jobName, sDFile1.Name(), "/idontexistdoi.json"),
|
||||||
|
scrapeManager,
|
||||||
|
discoveryManager,
|
||||||
|
)
|
||||||
|
|
||||||
|
// The old target should get dropped
|
||||||
|
requireTargets(
|
||||||
|
t,
|
||||||
|
scrapeManager,
|
||||||
|
jobName,
|
||||||
|
false,
|
||||||
|
[]string{fmt.Sprintf("http://%s/metrics", jobTarget1URL)},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestProviderStaleTargetsAreDropped makes sure that when a job has only one provider and when that provider
|
||||||
|
// should no longer discover targets, the targets of that provider are dropped.
|
||||||
|
// See: https://github.com/prometheus/prometheus/issues/12858
|
||||||
|
func TestProviderStaleTargetsAreDropped(t *testing.T) {
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
jobName := "my-job"
|
||||||
|
jobTargetURL := "foo:9876"
|
||||||
|
|
||||||
|
sdFileContent := fmt.Sprintf(`[{"targets": ["%s"]}]`, jobTargetURL)
|
||||||
|
sDFile := writeIntoFile(t, sdFileContent, "*targets.json")
|
||||||
|
|
||||||
|
baseConfig := `
|
||||||
|
scrape_configs:
|
||||||
|
- job_name: %s
|
||||||
|
file_sd_configs:
|
||||||
|
- files: ['%s']
|
||||||
|
`
|
||||||
|
discoveryManager, scrapeManager := runManagers(t, ctx)
|
||||||
|
defer scrapeManager.Stop()
|
||||||
|
|
||||||
|
applyConfig(
|
||||||
|
t,
|
||||||
|
fmt.Sprintf(baseConfig, jobName, sDFile.Name()),
|
||||||
|
scrapeManager,
|
||||||
|
discoveryManager,
|
||||||
|
)
|
||||||
|
|
||||||
|
// Make sure the job's targets are taken into account
|
||||||
|
requireTargets(
|
||||||
|
t,
|
||||||
|
scrapeManager,
|
||||||
|
jobName,
|
||||||
|
true,
|
||||||
|
[]string{
|
||||||
|
fmt.Sprintf("http://%s/metrics", jobTargetURL),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
// Apply the same config for the same job but with a non existing file to make the provider
|
||||||
|
// unable to discover some targets
|
||||||
|
applyConfig(
|
||||||
|
t,
|
||||||
|
fmt.Sprintf(baseConfig, jobName, "/idontexistdoi.json"),
|
||||||
|
scrapeManager,
|
||||||
|
discoveryManager,
|
||||||
|
)
|
||||||
|
|
||||||
|
// The old target should get dropped
|
||||||
|
requireTargets(
|
||||||
|
t,
|
||||||
|
scrapeManager,
|
||||||
|
jobName,
|
||||||
|
false,
|
||||||
|
nil,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestOnlyStaleTargetsAreDropped makes sure that when a job has multiple providers, when aone of them should no,
|
||||||
|
// longer discover targets, only the stale targets of that provier are dropped.
|
||||||
|
func TestOnlyStaleTargetsAreDropped(t *testing.T) {
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
myJob := "my-job"
|
||||||
|
myJobSDTargetURL := "my:9876"
|
||||||
|
myJobStaticTargetURL := "my:5432"
|
||||||
|
otherJob := "other-job"
|
||||||
|
otherJobTargetURL := "other:1234"
|
||||||
|
|
||||||
|
sdFileContent := fmt.Sprintf(`[{"targets": ["%s"]}]`, myJobSDTargetURL)
|
||||||
|
sDFile := writeIntoFile(t, sdFileContent, "*targets.json")
|
||||||
|
|
||||||
|
baseConfig := `
|
||||||
|
scrape_configs:
|
||||||
|
- job_name: %s
|
||||||
|
static_configs:
|
||||||
|
- targets: ['%s']
|
||||||
|
file_sd_configs:
|
||||||
|
- files: ['%s']
|
||||||
|
- job_name: %s
|
||||||
|
static_configs:
|
||||||
|
- targets: ['%s']
|
||||||
|
`
|
||||||
|
|
||||||
|
discoveryManager, scrapeManager := runManagers(t, ctx)
|
||||||
|
defer scrapeManager.Stop()
|
||||||
|
|
||||||
|
// Apply the initial config with an existing file
|
||||||
|
applyConfig(
|
||||||
|
t,
|
||||||
|
fmt.Sprintf(
|
||||||
|
baseConfig,
|
||||||
|
myJob,
|
||||||
|
myJobStaticTargetURL,
|
||||||
|
sDFile.Name(),
|
||||||
|
otherJob,
|
||||||
|
otherJobTargetURL,
|
||||||
|
),
|
||||||
|
scrapeManager,
|
||||||
|
discoveryManager,
|
||||||
|
)
|
||||||
|
// Make sure the jobs targets are taken into account
|
||||||
|
requireTargets(
|
||||||
|
t,
|
||||||
|
scrapeManager,
|
||||||
|
myJob,
|
||||||
|
true,
|
||||||
|
[]string{
|
||||||
|
fmt.Sprintf("http://%s/metrics", myJobSDTargetURL),
|
||||||
|
fmt.Sprintf("http://%s/metrics", myJobStaticTargetURL),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
requireTargets(
|
||||||
|
t,
|
||||||
|
scrapeManager,
|
||||||
|
otherJob,
|
||||||
|
true,
|
||||||
|
[]string{fmt.Sprintf("http://%s/metrics", otherJobTargetURL)},
|
||||||
|
)
|
||||||
|
|
||||||
|
// Apply the same config with a non existing file for myJob
|
||||||
|
applyConfig(
|
||||||
|
t,
|
||||||
|
fmt.Sprintf(
|
||||||
|
baseConfig,
|
||||||
|
myJob,
|
||||||
|
myJobStaticTargetURL,
|
||||||
|
"/idontexistdoi.json",
|
||||||
|
otherJob,
|
||||||
|
otherJobTargetURL,
|
||||||
|
),
|
||||||
|
scrapeManager,
|
||||||
|
discoveryManager,
|
||||||
|
)
|
||||||
|
|
||||||
|
// Only the SD target should get dropped for myJob
|
||||||
|
requireTargets(
|
||||||
|
t,
|
||||||
|
scrapeManager,
|
||||||
|
myJob,
|
||||||
|
false,
|
||||||
|
[]string{
|
||||||
|
fmt.Sprintf("http://%s/metrics", myJobStaticTargetURL),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
// The otherJob should keep its target
|
||||||
|
requireTargets(
|
||||||
|
t,
|
||||||
|
scrapeManager,
|
||||||
|
otherJob,
|
||||||
|
false,
|
||||||
|
[]string{fmt.Sprintf("http://%s/metrics", otherJobTargetURL)},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
|
@ -111,6 +111,7 @@ type scrapeLoopOptions struct {
|
||||||
interval time.Duration
|
interval time.Duration
|
||||||
timeout time.Duration
|
timeout time.Duration
|
||||||
scrapeClassicHistograms bool
|
scrapeClassicHistograms bool
|
||||||
|
validationScheme model.ValidationScheme
|
||||||
|
|
||||||
mrc []*relabel.Config
|
mrc []*relabel.Config
|
||||||
cache *scrapeCache
|
cache *scrapeCache
|
||||||
|
@ -186,6 +187,7 @@ func newScrapePool(cfg *config.ScrapeConfig, app storage.Appendable, offsetSeed
|
||||||
options.PassMetadataInContext,
|
options.PassMetadataInContext,
|
||||||
metrics,
|
metrics,
|
||||||
options.skipOffsetting,
|
options.skipOffsetting,
|
||||||
|
opts.validationScheme,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
sp.metrics.targetScrapePoolTargetLimit.WithLabelValues(sp.config.JobName).Set(float64(sp.config.TargetLimit))
|
sp.metrics.targetScrapePoolTargetLimit.WithLabelValues(sp.config.JobName).Set(float64(sp.config.TargetLimit))
|
||||||
|
@ -346,6 +348,7 @@ func (sp *scrapePool) restartLoops(reuseCache bool) {
|
||||||
cache: cache,
|
cache: cache,
|
||||||
interval: interval,
|
interval: interval,
|
||||||
timeout: timeout,
|
timeout: timeout,
|
||||||
|
validationScheme: validationScheme,
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -853,6 +856,7 @@ type scrapeLoop struct {
|
||||||
interval time.Duration
|
interval time.Duration
|
||||||
timeout time.Duration
|
timeout time.Duration
|
||||||
scrapeClassicHistograms bool
|
scrapeClassicHistograms bool
|
||||||
|
validationScheme model.ValidationScheme
|
||||||
|
|
||||||
// Feature flagged options.
|
// Feature flagged options.
|
||||||
enableNativeHistogramIngestion bool
|
enableNativeHistogramIngestion bool
|
||||||
|
@ -1160,6 +1164,7 @@ func newScrapeLoop(ctx context.Context,
|
||||||
passMetadataInContext bool,
|
passMetadataInContext bool,
|
||||||
metrics *scrapeMetrics,
|
metrics *scrapeMetrics,
|
||||||
skipOffsetting bool,
|
skipOffsetting bool,
|
||||||
|
validationScheme model.ValidationScheme,
|
||||||
) *scrapeLoop {
|
) *scrapeLoop {
|
||||||
if l == nil {
|
if l == nil {
|
||||||
l = log.NewNopLogger()
|
l = log.NewNopLogger()
|
||||||
|
@ -1211,6 +1216,7 @@ func newScrapeLoop(ctx context.Context,
|
||||||
appendMetadataToWAL: appendMetadataToWAL,
|
appendMetadataToWAL: appendMetadataToWAL,
|
||||||
metrics: metrics,
|
metrics: metrics,
|
||||||
skipOffsetting: skipOffsetting,
|
skipOffsetting: skipOffsetting,
|
||||||
|
validationScheme: validationScheme,
|
||||||
}
|
}
|
||||||
sl.ctx, sl.cancel = context.WithCancel(ctx)
|
sl.ctx, sl.cancel = context.WithCancel(ctx)
|
||||||
|
|
||||||
|
@ -1631,7 +1637,7 @@ loop:
|
||||||
err = errNameLabelMandatory
|
err = errNameLabelMandatory
|
||||||
break loop
|
break loop
|
||||||
}
|
}
|
||||||
if !lset.IsValid() {
|
if !lset.IsValid(sl.validationScheme) {
|
||||||
err = fmt.Errorf("invalid metric name or label names: %s", lset.String())
|
err = fmt.Errorf("invalid metric name or label names: %s", lset.String())
|
||||||
break loop
|
break loop
|
||||||
}
|
}
|
||||||
|
@ -1646,7 +1652,7 @@ loop:
|
||||||
updateMetadata(lset, true)
|
updateMetadata(lset, true)
|
||||||
}
|
}
|
||||||
|
|
||||||
if seriesAlreadyScraped {
|
if seriesAlreadyScraped && parsedTimestamp == nil {
|
||||||
err = storage.ErrDuplicateSampleForTimestamp
|
err = storage.ErrDuplicateSampleForTimestamp
|
||||||
} else {
|
} else {
|
||||||
if ctMs := p.CreatedTimestamp(); sl.enableCTZeroIngestion && ctMs != nil {
|
if ctMs := p.CreatedTimestamp(); sl.enableCTZeroIngestion && ctMs != nil {
|
||||||
|
|
|
@ -35,6 +35,7 @@ import (
|
||||||
"github.com/gogo/protobuf/proto"
|
"github.com/gogo/protobuf/proto"
|
||||||
"github.com/google/go-cmp/cmp"
|
"github.com/google/go-cmp/cmp"
|
||||||
"github.com/prometheus/client_golang/prometheus"
|
"github.com/prometheus/client_golang/prometheus"
|
||||||
|
prom_testutil "github.com/prometheus/client_golang/prometheus/testutil"
|
||||||
dto "github.com/prometheus/client_model/go"
|
dto "github.com/prometheus/client_model/go"
|
||||||
config_util "github.com/prometheus/common/config"
|
config_util "github.com/prometheus/common/config"
|
||||||
"github.com/prometheus/common/model"
|
"github.com/prometheus/common/model"
|
||||||
|
@ -683,6 +684,7 @@ func newBasicScrapeLoop(t testing.TB, ctx context.Context, scraper scraper, app
|
||||||
false,
|
false,
|
||||||
newTestScrapeMetrics(t),
|
newTestScrapeMetrics(t),
|
||||||
false,
|
false,
|
||||||
|
model.LegacyValidation,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -825,6 +827,7 @@ func TestScrapeLoopRun(t *testing.T) {
|
||||||
false,
|
false,
|
||||||
scrapeMetrics,
|
scrapeMetrics,
|
||||||
false,
|
false,
|
||||||
|
model.LegacyValidation,
|
||||||
)
|
)
|
||||||
|
|
||||||
// The loop must terminate during the initial offset if the context
|
// The loop must terminate during the initial offset if the context
|
||||||
|
@ -969,6 +972,7 @@ func TestScrapeLoopMetadata(t *testing.T) {
|
||||||
false,
|
false,
|
||||||
scrapeMetrics,
|
scrapeMetrics,
|
||||||
false,
|
false,
|
||||||
|
model.LegacyValidation,
|
||||||
)
|
)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
|
@ -1064,6 +1068,40 @@ func TestScrapeLoopFailWithInvalidLabelsAfterRelabel(t *testing.T) {
|
||||||
require.Equal(t, 0, seriesAdded)
|
require.Equal(t, 0, seriesAdded)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestScrapeLoopFailLegacyUnderUTF8(t *testing.T) {
|
||||||
|
// Test that scrapes fail when default validation is utf8 but scrape config is
|
||||||
|
// legacy.
|
||||||
|
model.NameValidationScheme = model.UTF8Validation
|
||||||
|
defer func() {
|
||||||
|
model.NameValidationScheme = model.LegacyValidation
|
||||||
|
}()
|
||||||
|
s := teststorage.New(t)
|
||||||
|
defer s.Close()
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
sl := newBasicScrapeLoop(t, ctx, &testScraper{}, s.Appender, 0)
|
||||||
|
sl.validationScheme = model.LegacyValidation
|
||||||
|
|
||||||
|
slApp := sl.appender(ctx)
|
||||||
|
total, added, seriesAdded, err := sl.append(slApp, []byte("{\"test.metric\"} 1\n"), "", time.Time{})
|
||||||
|
require.ErrorContains(t, err, "invalid metric name or label names")
|
||||||
|
require.NoError(t, slApp.Rollback())
|
||||||
|
require.Equal(t, 1, total)
|
||||||
|
require.Equal(t, 0, added)
|
||||||
|
require.Equal(t, 0, seriesAdded)
|
||||||
|
|
||||||
|
// When scrapeloop has validation set to UTF-8, the metric is allowed.
|
||||||
|
sl.validationScheme = model.UTF8Validation
|
||||||
|
|
||||||
|
slApp = sl.appender(ctx)
|
||||||
|
total, added, seriesAdded, err = sl.append(slApp, []byte("{\"test.metric\"} 1\n"), "", time.Time{})
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, 1, total)
|
||||||
|
require.Equal(t, 1, added)
|
||||||
|
require.Equal(t, 1, seriesAdded)
|
||||||
|
}
|
||||||
|
|
||||||
func makeTestMetrics(n int) []byte {
|
func makeTestMetrics(n int) []byte {
|
||||||
// Construct a metrics string to parse
|
// Construct a metrics string to parse
|
||||||
sb := bytes.Buffer{}
|
sb := bytes.Buffer{}
|
||||||
|
@ -3644,6 +3682,7 @@ func TestScrapeLoopSeriesAddedDuplicates(t *testing.T) {
|
||||||
require.Equal(t, 3, total)
|
require.Equal(t, 3, total)
|
||||||
require.Equal(t, 3, added)
|
require.Equal(t, 3, added)
|
||||||
require.Equal(t, 1, seriesAdded)
|
require.Equal(t, 1, seriesAdded)
|
||||||
|
require.Equal(t, 2.0, prom_testutil.ToFloat64(sl.metrics.targetScrapeSampleDuplicate))
|
||||||
|
|
||||||
slApp = sl.appender(ctx)
|
slApp = sl.appender(ctx)
|
||||||
total, added, seriesAdded, err = sl.append(slApp, []byte("test_metric 1\ntest_metric 1\ntest_metric 1\n"), "", time.Time{})
|
total, added, seriesAdded, err = sl.append(slApp, []byte("test_metric 1\ntest_metric 1\ntest_metric 1\n"), "", time.Time{})
|
||||||
|
@ -3652,12 +3691,18 @@ func TestScrapeLoopSeriesAddedDuplicates(t *testing.T) {
|
||||||
require.Equal(t, 3, total)
|
require.Equal(t, 3, total)
|
||||||
require.Equal(t, 3, added)
|
require.Equal(t, 3, added)
|
||||||
require.Equal(t, 0, seriesAdded)
|
require.Equal(t, 0, seriesAdded)
|
||||||
|
require.Equal(t, 4.0, prom_testutil.ToFloat64(sl.metrics.targetScrapeSampleDuplicate))
|
||||||
|
|
||||||
metric := dto.Metric{}
|
// When different timestamps are supplied, multiple samples are accepted.
|
||||||
err = sl.metrics.targetScrapeSampleDuplicate.Write(&metric)
|
slApp = sl.appender(ctx)
|
||||||
|
total, added, seriesAdded, err = sl.append(slApp, []byte("test_metric 1 1001\ntest_metric 1 1002\ntest_metric 1 1003\n"), "", time.Time{})
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
value := metric.GetCounter().GetValue()
|
require.NoError(t, slApp.Commit())
|
||||||
require.Equal(t, 4.0, value)
|
require.Equal(t, 3, total)
|
||||||
|
require.Equal(t, 3, added)
|
||||||
|
require.Equal(t, 0, seriesAdded)
|
||||||
|
// Metric is not higher than last time.
|
||||||
|
require.Equal(t, 4.0, prom_testutil.ToFloat64(sl.metrics.targetScrapeSampleDuplicate))
|
||||||
}
|
}
|
||||||
|
|
||||||
// This tests running a full scrape loop and checking that the scrape option
|
// This tests running a full scrape loop and checking that the scrape option
|
||||||
|
|
|
@ -26,10 +26,6 @@ import (
|
||||||
"github.com/gogo/protobuf/proto"
|
"github.com/gogo/protobuf/proto"
|
||||||
)
|
)
|
||||||
|
|
||||||
// DefaultChunkedReadLimit is the default value for the maximum size of the protobuf frame client allows.
|
|
||||||
// 50MB is the default. This is equivalent to ~100k full XOR chunks and average labelset.
|
|
||||||
const DefaultChunkedReadLimit = 5e+7
|
|
||||||
|
|
||||||
// The table gets initialized with sync.Once but may still cause a race
|
// The table gets initialized with sync.Once but may still cause a race
|
||||||
// with any other use of the crc32 package anywhere. Thus we initialize it
|
// with any other use of the crc32 package anywhere. Thus we initialize it
|
||||||
// before.
|
// before.
|
||||||
|
|
|
@ -16,6 +16,7 @@ package remote
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
@ -36,13 +37,14 @@ import (
|
||||||
|
|
||||||
"github.com/prometheus/prometheus/config"
|
"github.com/prometheus/prometheus/config"
|
||||||
"github.com/prometheus/prometheus/prompb"
|
"github.com/prometheus/prometheus/prompb"
|
||||||
|
"github.com/prometheus/prometheus/storage"
|
||||||
"github.com/prometheus/prometheus/storage/remote/azuread"
|
"github.com/prometheus/prometheus/storage/remote/azuread"
|
||||||
"github.com/prometheus/prometheus/storage/remote/googleiam"
|
"github.com/prometheus/prometheus/storage/remote/googleiam"
|
||||||
)
|
)
|
||||||
|
|
||||||
const maxErrMsgLen = 1024
|
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
maxErrMsgLen = 1024
|
||||||
|
|
||||||
RemoteWriteVersionHeader = "X-Prometheus-Remote-Write-Version"
|
RemoteWriteVersionHeader = "X-Prometheus-Remote-Write-Version"
|
||||||
RemoteWriteVersion1HeaderValue = "0.1.0"
|
RemoteWriteVersion1HeaderValue = "0.1.0"
|
||||||
RemoteWriteVersion20HeaderValue = "2.0.0"
|
RemoteWriteVersion20HeaderValue = "2.0.0"
|
||||||
|
@ -68,9 +70,12 @@ var (
|
||||||
config.RemoteWriteProtoMsgV1: appProtoContentType, // Also application/x-protobuf;proto=prometheus.WriteRequest but simplified for compatibility with 1.x spec.
|
config.RemoteWriteProtoMsgV1: appProtoContentType, // Also application/x-protobuf;proto=prometheus.WriteRequest but simplified for compatibility with 1.x spec.
|
||||||
config.RemoteWriteProtoMsgV2: appProtoContentType + ";proto=io.prometheus.write.v2.Request",
|
config.RemoteWriteProtoMsgV2: appProtoContentType + ";proto=io.prometheus.write.v2.Request",
|
||||||
}
|
}
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
AcceptedResponseTypes = []prompb.ReadRequest_ResponseType{
|
||||||
|
prompb.ReadRequest_STREAMED_XOR_CHUNKS,
|
||||||
|
prompb.ReadRequest_SAMPLES,
|
||||||
|
}
|
||||||
|
|
||||||
remoteReadQueriesTotal = prometheus.NewCounterVec(
|
remoteReadQueriesTotal = prometheus.NewCounterVec(
|
||||||
prometheus.CounterOpts{
|
prometheus.CounterOpts{
|
||||||
Namespace: namespace,
|
Namespace: namespace,
|
||||||
|
@ -78,7 +83,7 @@ var (
|
||||||
Name: "read_queries_total",
|
Name: "read_queries_total",
|
||||||
Help: "The total number of remote read queries.",
|
Help: "The total number of remote read queries.",
|
||||||
},
|
},
|
||||||
[]string{remoteName, endpoint, "code"},
|
[]string{remoteName, endpoint, "response_type", "code"},
|
||||||
)
|
)
|
||||||
remoteReadQueries = prometheus.NewGaugeVec(
|
remoteReadQueries = prometheus.NewGaugeVec(
|
||||||
prometheus.GaugeOpts{
|
prometheus.GaugeOpts{
|
||||||
|
@ -94,13 +99,13 @@ var (
|
||||||
Namespace: namespace,
|
Namespace: namespace,
|
||||||
Subsystem: subsystem,
|
Subsystem: subsystem,
|
||||||
Name: "read_request_duration_seconds",
|
Name: "read_request_duration_seconds",
|
||||||
Help: "Histogram of the latency for remote read requests.",
|
Help: "Histogram of the latency for remote read requests. Note that for streamed responses this is only the duration of the initial call and does not include the processing of the stream.",
|
||||||
Buckets: append(prometheus.DefBuckets, 25, 60),
|
Buckets: append(prometheus.DefBuckets, 25, 60),
|
||||||
NativeHistogramBucketFactor: 1.1,
|
NativeHistogramBucketFactor: 1.1,
|
||||||
NativeHistogramMaxBucketNumber: 100,
|
NativeHistogramMaxBucketNumber: 100,
|
||||||
NativeHistogramMinResetDuration: 1 * time.Hour,
|
NativeHistogramMinResetDuration: 1 * time.Hour,
|
||||||
},
|
},
|
||||||
[]string{remoteName, endpoint},
|
[]string{remoteName, endpoint, "response_type"},
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -116,10 +121,11 @@ type Client struct {
|
||||||
timeout time.Duration
|
timeout time.Duration
|
||||||
|
|
||||||
retryOnRateLimit bool
|
retryOnRateLimit bool
|
||||||
|
chunkedReadLimit uint64
|
||||||
|
|
||||||
readQueries prometheus.Gauge
|
readQueries prometheus.Gauge
|
||||||
readQueriesTotal *prometheus.CounterVec
|
readQueriesTotal *prometheus.CounterVec
|
||||||
readQueriesDuration prometheus.Observer
|
readQueriesDuration prometheus.ObserverVec
|
||||||
|
|
||||||
writeProtoMsg config.RemoteWriteProtoMsg
|
writeProtoMsg config.RemoteWriteProtoMsg
|
||||||
writeCompression Compression // Not exposed by ClientConfig for now.
|
writeCompression Compression // Not exposed by ClientConfig for now.
|
||||||
|
@ -136,12 +142,13 @@ type ClientConfig struct {
|
||||||
Headers map[string]string
|
Headers map[string]string
|
||||||
RetryOnRateLimit bool
|
RetryOnRateLimit bool
|
||||||
WriteProtoMsg config.RemoteWriteProtoMsg
|
WriteProtoMsg config.RemoteWriteProtoMsg
|
||||||
|
ChunkedReadLimit uint64
|
||||||
}
|
}
|
||||||
|
|
||||||
// ReadClient uses the SAMPLES method of remote read to read series samples from remote server.
|
// ReadClient will request the STREAMED_XOR_CHUNKS method of remote read but can
|
||||||
// TODO(bwplotka): Add streamed chunked remote read method as well (https://github.com/prometheus/prometheus/issues/5926).
|
// also fall back to the SAMPLES method if necessary.
|
||||||
type ReadClient interface {
|
type ReadClient interface {
|
||||||
Read(ctx context.Context, query *prompb.Query) (*prompb.QueryResult, error)
|
Read(ctx context.Context, query *prompb.Query, sortSeries bool) (storage.SeriesSet, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewReadClient creates a new client for remote read.
|
// NewReadClient creates a new client for remote read.
|
||||||
|
@ -162,9 +169,10 @@ func NewReadClient(name string, conf *ClientConfig) (ReadClient, error) {
|
||||||
urlString: conf.URL.String(),
|
urlString: conf.URL.String(),
|
||||||
Client: httpClient,
|
Client: httpClient,
|
||||||
timeout: time.Duration(conf.Timeout),
|
timeout: time.Duration(conf.Timeout),
|
||||||
|
chunkedReadLimit: conf.ChunkedReadLimit,
|
||||||
readQueries: remoteReadQueries.WithLabelValues(name, conf.URL.String()),
|
readQueries: remoteReadQueries.WithLabelValues(name, conf.URL.String()),
|
||||||
readQueriesTotal: remoteReadQueriesTotal.MustCurryWith(prometheus.Labels{remoteName: name, endpoint: conf.URL.String()}),
|
readQueriesTotal: remoteReadQueriesTotal.MustCurryWith(prometheus.Labels{remoteName: name, endpoint: conf.URL.String()}),
|
||||||
readQueriesDuration: remoteReadQueryDuration.WithLabelValues(name, conf.URL.String()),
|
readQueriesDuration: remoteReadQueryDuration.MustCurryWith(prometheus.Labels{remoteName: name, endpoint: conf.URL.String()}),
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -278,8 +286,8 @@ func (c *Client) Store(ctx context.Context, req []byte, attempt int) (WriteRespo
|
||||||
return WriteResponseStats{}, RecoverableError{err, defaultBackoff}
|
return WriteResponseStats{}, RecoverableError{err, defaultBackoff}
|
||||||
}
|
}
|
||||||
defer func() {
|
defer func() {
|
||||||
io.Copy(io.Discard, httpResp.Body)
|
_, _ = io.Copy(io.Discard, httpResp.Body)
|
||||||
httpResp.Body.Close()
|
_ = httpResp.Body.Close()
|
||||||
}()
|
}()
|
||||||
|
|
||||||
// TODO(bwplotka): Pass logger and emit debug on error?
|
// TODO(bwplotka): Pass logger and emit debug on error?
|
||||||
|
@ -329,17 +337,17 @@ func (c *Client) Endpoint() string {
|
||||||
return c.urlString
|
return c.urlString
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read reads from a remote endpoint.
|
// Read reads from a remote endpoint. The sortSeries parameter is only respected in the case of a sampled response;
|
||||||
func (c *Client) Read(ctx context.Context, query *prompb.Query) (*prompb.QueryResult, error) {
|
// chunked responses arrive already sorted by the server.
|
||||||
|
func (c *Client) Read(ctx context.Context, query *prompb.Query, sortSeries bool) (storage.SeriesSet, error) {
|
||||||
c.readQueries.Inc()
|
c.readQueries.Inc()
|
||||||
defer c.readQueries.Dec()
|
defer c.readQueries.Dec()
|
||||||
|
|
||||||
req := &prompb.ReadRequest{
|
req := &prompb.ReadRequest{
|
||||||
// TODO: Support batching multiple queries into one read request,
|
// TODO: Support batching multiple queries into one read request,
|
||||||
// as the protobuf interface allows for it.
|
// as the protobuf interface allows for it.
|
||||||
Queries: []*prompb.Query{
|
Queries: []*prompb.Query{query},
|
||||||
query,
|
AcceptedResponseTypes: AcceptedResponseTypes,
|
||||||
},
|
|
||||||
}
|
}
|
||||||
data, err := proto.Marshal(req)
|
data, err := proto.Marshal(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -358,7 +366,6 @@ func (c *Client) Read(ctx context.Context, query *prompb.Query) (*prompb.QueryRe
|
||||||
httpReq.Header.Set("X-Prometheus-Remote-Read-Version", "0.1.0")
|
httpReq.Header.Set("X-Prometheus-Remote-Read-Version", "0.1.0")
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(ctx, c.timeout)
|
ctx, cancel := context.WithTimeout(ctx, c.timeout)
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
ctx, span := otel.Tracer("").Start(ctx, "Remote Read", trace.WithSpanKind(trace.SpanKindClient))
|
ctx, span := otel.Tracer("").Start(ctx, "Remote Read", trace.WithSpanKind(trace.SpanKindClient))
|
||||||
defer span.End()
|
defer span.End()
|
||||||
|
@ -366,24 +373,58 @@ func (c *Client) Read(ctx context.Context, query *prompb.Query) (*prompb.QueryRe
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
httpResp, err := c.Client.Do(httpReq.WithContext(ctx))
|
httpResp, err := c.Client.Do(httpReq.WithContext(ctx))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
cancel()
|
||||||
return nil, fmt.Errorf("error sending request: %w", err)
|
return nil, fmt.Errorf("error sending request: %w", err)
|
||||||
}
|
}
|
||||||
defer func() {
|
|
||||||
io.Copy(io.Discard, httpResp.Body)
|
|
||||||
httpResp.Body.Close()
|
|
||||||
}()
|
|
||||||
c.readQueriesDuration.Observe(time.Since(start).Seconds())
|
|
||||||
c.readQueriesTotal.WithLabelValues(strconv.Itoa(httpResp.StatusCode)).Inc()
|
|
||||||
|
|
||||||
compressed, err = io.ReadAll(httpResp.Body)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("error reading response. HTTP status code: %s: %w", httpResp.Status, err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if httpResp.StatusCode/100 != 2 {
|
if httpResp.StatusCode/100 != 2 {
|
||||||
return nil, fmt.Errorf("remote server %s returned HTTP status %s: %s", c.urlString, httpResp.Status, strings.TrimSpace(string(compressed)))
|
// Make an attempt at getting an error message.
|
||||||
|
body, _ := io.ReadAll(httpResp.Body)
|
||||||
|
_ = httpResp.Body.Close()
|
||||||
|
|
||||||
|
cancel()
|
||||||
|
return nil, fmt.Errorf("remote server %s returned http status %s: %s", c.urlString, httpResp.Status, string(body))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
contentType := httpResp.Header.Get("Content-Type")
|
||||||
|
|
||||||
|
switch {
|
||||||
|
case strings.HasPrefix(contentType, "application/x-protobuf"):
|
||||||
|
c.readQueriesDuration.WithLabelValues("sampled").Observe(time.Since(start).Seconds())
|
||||||
|
c.readQueriesTotal.WithLabelValues("sampled", strconv.Itoa(httpResp.StatusCode)).Inc()
|
||||||
|
ss, err := c.handleSampledResponse(req, httpResp, sortSeries)
|
||||||
|
cancel()
|
||||||
|
return ss, err
|
||||||
|
case strings.HasPrefix(contentType, "application/x-streamed-protobuf; proto=prometheus.ChunkedReadResponse"):
|
||||||
|
c.readQueriesDuration.WithLabelValues("chunked").Observe(time.Since(start).Seconds())
|
||||||
|
|
||||||
|
s := NewChunkedReader(httpResp.Body, c.chunkedReadLimit, nil)
|
||||||
|
return NewChunkedSeriesSet(s, httpResp.Body, query.StartTimestampMs, query.EndTimestampMs, func(err error) {
|
||||||
|
code := strconv.Itoa(httpResp.StatusCode)
|
||||||
|
if !errors.Is(err, io.EOF) {
|
||||||
|
code = "aborted_stream"
|
||||||
|
}
|
||||||
|
c.readQueriesTotal.WithLabelValues("chunked", code).Inc()
|
||||||
|
cancel()
|
||||||
|
}), nil
|
||||||
|
default:
|
||||||
|
c.readQueriesDuration.WithLabelValues("unsupported").Observe(time.Since(start).Seconds())
|
||||||
|
c.readQueriesTotal.WithLabelValues("unsupported", strconv.Itoa(httpResp.StatusCode)).Inc()
|
||||||
|
cancel()
|
||||||
|
return nil, fmt.Errorf("unsupported content type: %s", contentType)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) handleSampledResponse(req *prompb.ReadRequest, httpResp *http.Response, sortSeries bool) (storage.SeriesSet, error) {
|
||||||
|
compressed, err := io.ReadAll(httpResp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("error reading response. HTTP status code: %s: %w", httpResp.Status, err)
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
_, _ = io.Copy(io.Discard, httpResp.Body)
|
||||||
|
_ = httpResp.Body.Close()
|
||||||
|
}()
|
||||||
|
|
||||||
uncompressed, err := snappy.Decode(nil, compressed)
|
uncompressed, err := snappy.Decode(nil, compressed)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("error reading response: %w", err)
|
return nil, fmt.Errorf("error reading response: %w", err)
|
||||||
|
@ -399,5 +440,8 @@ func (c *Client) Read(ctx context.Context, query *prompb.Query) (*prompb.QueryRe
|
||||||
return nil, fmt.Errorf("responses: want %d, got %d", len(req.Queries), len(resp.Results))
|
return nil, fmt.Errorf("responses: want %d, got %d", len(req.Queries), len(resp.Results))
|
||||||
}
|
}
|
||||||
|
|
||||||
return resp.Results[0], nil
|
// This client does not batch queries so there's always only 1 result.
|
||||||
|
res := resp.Results[0]
|
||||||
|
|
||||||
|
return FromQueryResult(sortSeries, res), nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,9 +23,15 @@ import (
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/gogo/protobuf/proto"
|
||||||
|
"github.com/golang/snappy"
|
||||||
config_util "github.com/prometheus/common/config"
|
config_util "github.com/prometheus/common/config"
|
||||||
"github.com/prometheus/common/model"
|
"github.com/prometheus/common/model"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
|
|
||||||
|
"github.com/prometheus/prometheus/config"
|
||||||
|
"github.com/prometheus/prometheus/prompb"
|
||||||
|
"github.com/prometheus/prometheus/tsdb/chunkenc"
|
||||||
)
|
)
|
||||||
|
|
||||||
var longErrMessage = strings.Repeat("error message", maxErrMsgLen)
|
var longErrMessage = strings.Repeat("error message", maxErrMsgLen)
|
||||||
|
@ -208,3 +214,226 @@ func TestClientCustomHeaders(t *testing.T) {
|
||||||
|
|
||||||
require.True(t, called, "The remote server wasn't called")
|
require.True(t, called, "The remote server wasn't called")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestReadClient(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
query *prompb.Query
|
||||||
|
httpHandler http.HandlerFunc
|
||||||
|
expectedLabels []map[string]string
|
||||||
|
expectedSamples [][]model.SamplePair
|
||||||
|
expectedErrorContains string
|
||||||
|
sortSeries bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "sorted sampled response",
|
||||||
|
httpHandler: sampledResponseHTTPHandler(t),
|
||||||
|
expectedLabels: []map[string]string{
|
||||||
|
{"foo1": "bar"},
|
||||||
|
{"foo2": "bar"},
|
||||||
|
},
|
||||||
|
expectedSamples: [][]model.SamplePair{
|
||||||
|
{
|
||||||
|
{Timestamp: model.Time(0), Value: model.SampleValue(3)},
|
||||||
|
{Timestamp: model.Time(5), Value: model.SampleValue(4)},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{Timestamp: model.Time(0), Value: model.SampleValue(1)},
|
||||||
|
{Timestamp: model.Time(5), Value: model.SampleValue(2)},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
expectedErrorContains: "",
|
||||||
|
sortSeries: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "unsorted sampled response",
|
||||||
|
httpHandler: sampledResponseHTTPHandler(t),
|
||||||
|
expectedLabels: []map[string]string{
|
||||||
|
{"foo2": "bar"},
|
||||||
|
{"foo1": "bar"},
|
||||||
|
},
|
||||||
|
expectedSamples: [][]model.SamplePair{
|
||||||
|
{
|
||||||
|
{Timestamp: model.Time(0), Value: model.SampleValue(1)},
|
||||||
|
{Timestamp: model.Time(5), Value: model.SampleValue(2)},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{Timestamp: model.Time(0), Value: model.SampleValue(3)},
|
||||||
|
{Timestamp: model.Time(5), Value: model.SampleValue(4)},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
expectedErrorContains: "",
|
||||||
|
sortSeries: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "chunked response",
|
||||||
|
query: &prompb.Query{
|
||||||
|
StartTimestampMs: 4000,
|
||||||
|
EndTimestampMs: 12000,
|
||||||
|
},
|
||||||
|
httpHandler: http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.Header().Set("Content-Type", "application/x-streamed-protobuf; proto=prometheus.ChunkedReadResponse")
|
||||||
|
|
||||||
|
flusher, ok := w.(http.Flusher)
|
||||||
|
require.True(t, ok)
|
||||||
|
|
||||||
|
cw := NewChunkedWriter(w, flusher)
|
||||||
|
l := []prompb.Label{
|
||||||
|
{Name: "foo", Value: "bar"},
|
||||||
|
}
|
||||||
|
|
||||||
|
chunks := buildTestChunks(t)
|
||||||
|
for i, c := range chunks {
|
||||||
|
cSeries := prompb.ChunkedSeries{Labels: l, Chunks: []prompb.Chunk{c}}
|
||||||
|
readResp := prompb.ChunkedReadResponse{
|
||||||
|
ChunkedSeries: []*prompb.ChunkedSeries{&cSeries},
|
||||||
|
QueryIndex: int64(i),
|
||||||
|
}
|
||||||
|
|
||||||
|
b, err := proto.Marshal(&readResp)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
_, err = cw.Write(b)
|
||||||
|
require.NoError(t, err)
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
expectedLabels: []map[string]string{
|
||||||
|
{"foo": "bar"},
|
||||||
|
{"foo": "bar"},
|
||||||
|
{"foo": "bar"},
|
||||||
|
},
|
||||||
|
// This is the output of buildTestChunks minus the samples outside the query range.
|
||||||
|
expectedSamples: [][]model.SamplePair{
|
||||||
|
{
|
||||||
|
{Timestamp: model.Time(4000), Value: model.SampleValue(4)},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{Timestamp: model.Time(5000), Value: model.SampleValue(1)},
|
||||||
|
{Timestamp: model.Time(6000), Value: model.SampleValue(2)},
|
||||||
|
{Timestamp: model.Time(7000), Value: model.SampleValue(3)},
|
||||||
|
{Timestamp: model.Time(8000), Value: model.SampleValue(4)},
|
||||||
|
{Timestamp: model.Time(9000), Value: model.SampleValue(5)},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{Timestamp: model.Time(10000), Value: model.SampleValue(2)},
|
||||||
|
{Timestamp: model.Time(11000), Value: model.SampleValue(3)},
|
||||||
|
{Timestamp: model.Time(12000), Value: model.SampleValue(4)},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
expectedErrorContains: "",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "unsupported content type",
|
||||||
|
httpHandler: http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.Header().Set("Content-Type", "foobar")
|
||||||
|
}),
|
||||||
|
expectedErrorContains: "unsupported content type",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, test := range tests {
|
||||||
|
t.Run(test.name, func(t *testing.T) {
|
||||||
|
server := httptest.NewServer(test.httpHandler)
|
||||||
|
defer server.Close()
|
||||||
|
|
||||||
|
u, err := url.Parse(server.URL)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
conf := &ClientConfig{
|
||||||
|
URL: &config_util.URL{URL: u},
|
||||||
|
Timeout: model.Duration(5 * time.Second),
|
||||||
|
ChunkedReadLimit: config.DefaultChunkedReadLimit,
|
||||||
|
}
|
||||||
|
c, err := NewReadClient("test", conf)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
query := &prompb.Query{}
|
||||||
|
if test.query != nil {
|
||||||
|
query = test.query
|
||||||
|
}
|
||||||
|
|
||||||
|
ss, err := c.Read(context.Background(), query, test.sortSeries)
|
||||||
|
if test.expectedErrorContains != "" {
|
||||||
|
require.ErrorContains(t, err, test.expectedErrorContains)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
i := 0
|
||||||
|
|
||||||
|
for ss.Next() {
|
||||||
|
require.NoError(t, ss.Err())
|
||||||
|
s := ss.At()
|
||||||
|
|
||||||
|
l := s.Labels()
|
||||||
|
require.Len(t, test.expectedLabels[i], l.Len())
|
||||||
|
for k, v := range test.expectedLabels[i] {
|
||||||
|
require.True(t, l.Has(k))
|
||||||
|
require.Equal(t, v, l.Get(k))
|
||||||
|
}
|
||||||
|
|
||||||
|
it := s.Iterator(nil)
|
||||||
|
j := 0
|
||||||
|
|
||||||
|
for valType := it.Next(); valType != chunkenc.ValNone; valType = it.Next() {
|
||||||
|
require.NoError(t, it.Err())
|
||||||
|
|
||||||
|
ts, v := it.At()
|
||||||
|
expectedSample := test.expectedSamples[i][j]
|
||||||
|
|
||||||
|
require.Equal(t, int64(expectedSample.Timestamp), ts)
|
||||||
|
require.Equal(t, float64(expectedSample.Value), v)
|
||||||
|
|
||||||
|
j++
|
||||||
|
}
|
||||||
|
|
||||||
|
require.Len(t, test.expectedSamples[i], j)
|
||||||
|
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
|
||||||
|
require.NoError(t, ss.Err())
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func sampledResponseHTTPHandler(t *testing.T) http.HandlerFunc {
|
||||||
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.Header().Set("Content-Type", "application/x-protobuf")
|
||||||
|
|
||||||
|
resp := prompb.ReadResponse{
|
||||||
|
Results: []*prompb.QueryResult{
|
||||||
|
{
|
||||||
|
Timeseries: []*prompb.TimeSeries{
|
||||||
|
{
|
||||||
|
Labels: []prompb.Label{
|
||||||
|
{Name: "foo2", Value: "bar"},
|
||||||
|
},
|
||||||
|
Samples: []prompb.Sample{
|
||||||
|
{Value: float64(1), Timestamp: int64(0)},
|
||||||
|
{Value: float64(2), Timestamp: int64(5)},
|
||||||
|
},
|
||||||
|
Exemplars: []prompb.Exemplar{},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Labels: []prompb.Label{
|
||||||
|
{Name: "foo1", Value: "bar"},
|
||||||
|
},
|
||||||
|
Samples: []prompb.Sample{
|
||||||
|
{Value: float64(3), Timestamp: int64(0)},
|
||||||
|
{Value: float64(4), Timestamp: int64(5)},
|
||||||
|
},
|
||||||
|
Exemplars: []prompb.Exemplar{},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
b, err := proto.Marshal(&resp)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
_, err = w.Write(snappy.Encode(nil, b))
|
||||||
|
require.NoError(t, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -540,6 +540,220 @@ func (c *concreteSeriesIterator) Err() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// chunkedSeriesSet implements storage.SeriesSet.
|
||||||
|
type chunkedSeriesSet struct {
|
||||||
|
chunkedReader *ChunkedReader
|
||||||
|
respBody io.ReadCloser
|
||||||
|
mint, maxt int64
|
||||||
|
cancel func(error)
|
||||||
|
|
||||||
|
current storage.Series
|
||||||
|
err error
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewChunkedSeriesSet(chunkedReader *ChunkedReader, respBody io.ReadCloser, mint, maxt int64, cancel func(error)) storage.SeriesSet {
|
||||||
|
return &chunkedSeriesSet{
|
||||||
|
chunkedReader: chunkedReader,
|
||||||
|
respBody: respBody,
|
||||||
|
mint: mint,
|
||||||
|
maxt: maxt,
|
||||||
|
cancel: cancel,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Next return true if there is a next series and false otherwise. It will
|
||||||
|
// block until the next series is available.
|
||||||
|
func (s *chunkedSeriesSet) Next() bool {
|
||||||
|
res := &prompb.ChunkedReadResponse{}
|
||||||
|
|
||||||
|
err := s.chunkedReader.NextProto(res)
|
||||||
|
if err != nil {
|
||||||
|
if !errors.Is(err, io.EOF) {
|
||||||
|
s.err = err
|
||||||
|
_, _ = io.Copy(io.Discard, s.respBody)
|
||||||
|
}
|
||||||
|
|
||||||
|
_ = s.respBody.Close()
|
||||||
|
s.cancel(err)
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
s.current = &chunkedSeries{
|
||||||
|
ChunkedSeries: prompb.ChunkedSeries{
|
||||||
|
Labels: res.ChunkedSeries[0].Labels,
|
||||||
|
Chunks: res.ChunkedSeries[0].Chunks,
|
||||||
|
},
|
||||||
|
mint: s.mint,
|
||||||
|
maxt: s.maxt,
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *chunkedSeriesSet) At() storage.Series {
|
||||||
|
return s.current
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *chunkedSeriesSet) Err() error {
|
||||||
|
return s.err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *chunkedSeriesSet) Warnings() annotations.Annotations {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type chunkedSeries struct {
|
||||||
|
prompb.ChunkedSeries
|
||||||
|
mint, maxt int64
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ storage.Series = &chunkedSeries{}
|
||||||
|
|
||||||
|
func (s *chunkedSeries) Labels() labels.Labels {
|
||||||
|
b := labels.NewScratchBuilder(0)
|
||||||
|
return s.ToLabels(&b, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *chunkedSeries) Iterator(it chunkenc.Iterator) chunkenc.Iterator {
|
||||||
|
csIt, ok := it.(*chunkedSeriesIterator)
|
||||||
|
if ok {
|
||||||
|
csIt.reset(s.Chunks, s.mint, s.maxt)
|
||||||
|
return csIt
|
||||||
|
}
|
||||||
|
return newChunkedSeriesIterator(s.Chunks, s.mint, s.maxt)
|
||||||
|
}
|
||||||
|
|
||||||
|
type chunkedSeriesIterator struct {
|
||||||
|
chunks []prompb.Chunk
|
||||||
|
idx int
|
||||||
|
cur chunkenc.Iterator
|
||||||
|
valType chunkenc.ValueType
|
||||||
|
mint, maxt int64
|
||||||
|
|
||||||
|
err error
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ chunkenc.Iterator = &chunkedSeriesIterator{}
|
||||||
|
|
||||||
|
func newChunkedSeriesIterator(chunks []prompb.Chunk, mint, maxt int64) *chunkedSeriesIterator {
|
||||||
|
it := &chunkedSeriesIterator{}
|
||||||
|
it.reset(chunks, mint, maxt)
|
||||||
|
return it
|
||||||
|
}
|
||||||
|
|
||||||
|
func (it *chunkedSeriesIterator) Next() chunkenc.ValueType {
|
||||||
|
if it.err != nil {
|
||||||
|
return chunkenc.ValNone
|
||||||
|
}
|
||||||
|
if len(it.chunks) == 0 {
|
||||||
|
return chunkenc.ValNone
|
||||||
|
}
|
||||||
|
|
||||||
|
for it.valType = it.cur.Next(); it.valType != chunkenc.ValNone; it.valType = it.cur.Next() {
|
||||||
|
atT := it.AtT()
|
||||||
|
if atT > it.maxt {
|
||||||
|
it.chunks = nil // Exhaust this iterator so follow-up calls to Next or Seek return fast.
|
||||||
|
return chunkenc.ValNone
|
||||||
|
}
|
||||||
|
if atT >= it.mint {
|
||||||
|
return it.valType
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if it.idx >= len(it.chunks)-1 {
|
||||||
|
it.valType = chunkenc.ValNone
|
||||||
|
} else {
|
||||||
|
it.idx++
|
||||||
|
it.resetIterator()
|
||||||
|
it.valType = it.Next()
|
||||||
|
}
|
||||||
|
|
||||||
|
return it.valType
|
||||||
|
}
|
||||||
|
|
||||||
|
func (it *chunkedSeriesIterator) Seek(t int64) chunkenc.ValueType {
|
||||||
|
if it.err != nil {
|
||||||
|
return chunkenc.ValNone
|
||||||
|
}
|
||||||
|
if len(it.chunks) == 0 {
|
||||||
|
return chunkenc.ValNone
|
||||||
|
}
|
||||||
|
|
||||||
|
startIdx := it.idx
|
||||||
|
it.idx += sort.Search(len(it.chunks)-startIdx, func(i int) bool {
|
||||||
|
return it.chunks[startIdx+i].MaxTimeMs >= t
|
||||||
|
})
|
||||||
|
if it.idx > startIdx {
|
||||||
|
it.resetIterator()
|
||||||
|
} else {
|
||||||
|
ts := it.cur.AtT()
|
||||||
|
if ts >= t {
|
||||||
|
return it.valType
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for it.valType = it.cur.Next(); it.valType != chunkenc.ValNone; it.valType = it.cur.Next() {
|
||||||
|
ts := it.cur.AtT()
|
||||||
|
if ts > it.maxt {
|
||||||
|
it.chunks = nil // Exhaust this iterator so follow-up calls to Next or Seek return fast.
|
||||||
|
return chunkenc.ValNone
|
||||||
|
}
|
||||||
|
if ts >= t && ts >= it.mint {
|
||||||
|
return it.valType
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
it.valType = chunkenc.ValNone
|
||||||
|
return it.valType
|
||||||
|
}
|
||||||
|
|
||||||
|
func (it *chunkedSeriesIterator) resetIterator() {
|
||||||
|
if it.idx < len(it.chunks) {
|
||||||
|
chunk := it.chunks[it.idx]
|
||||||
|
|
||||||
|
decodedChunk, err := chunkenc.FromData(chunkenc.Encoding(chunk.Type), chunk.Data)
|
||||||
|
if err != nil {
|
||||||
|
it.err = err
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
it.cur = decodedChunk.Iterator(nil)
|
||||||
|
} else {
|
||||||
|
it.cur = chunkenc.NewNopIterator()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (it *chunkedSeriesIterator) reset(chunks []prompb.Chunk, mint, maxt int64) {
|
||||||
|
it.chunks = chunks
|
||||||
|
it.mint = mint
|
||||||
|
it.maxt = maxt
|
||||||
|
it.idx = 0
|
||||||
|
if len(chunks) > 0 {
|
||||||
|
it.resetIterator()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (it *chunkedSeriesIterator) At() (ts int64, v float64) {
|
||||||
|
return it.cur.At()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (it *chunkedSeriesIterator) AtHistogram(h *histogram.Histogram) (int64, *histogram.Histogram) {
|
||||||
|
return it.cur.AtHistogram(h)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (it *chunkedSeriesIterator) AtFloatHistogram(fh *histogram.FloatHistogram) (int64, *histogram.FloatHistogram) {
|
||||||
|
return it.cur.AtFloatHistogram(fh)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (it *chunkedSeriesIterator) AtT() int64 {
|
||||||
|
return it.cur.AtT()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (it *chunkedSeriesIterator) Err() error {
|
||||||
|
return it.err
|
||||||
|
}
|
||||||
|
|
||||||
// validateLabelsAndMetricName validates the label names/values and metric names returned from remote read,
|
// validateLabelsAndMetricName validates the label names/values and metric names returned from remote read,
|
||||||
// also making sure that there are no labels with duplicate names.
|
// also making sure that there are no labels with duplicate names.
|
||||||
func validateLabelsAndMetricName(ls []prompb.Label) error {
|
func validateLabelsAndMetricName(ls []prompb.Label) error {
|
||||||
|
@ -612,15 +826,6 @@ func FromLabelMatchers(matchers []*prompb.LabelMatcher) ([]*labels.Matcher, erro
|
||||||
return result, nil
|
return result, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// LabelProtosToMetric unpack a []*prompb.Label to a model.Metric.
|
|
||||||
func LabelProtosToMetric(labelPairs []*prompb.Label) model.Metric {
|
|
||||||
metric := make(model.Metric, len(labelPairs))
|
|
||||||
for _, l := range labelPairs {
|
|
||||||
metric[model.LabelName(l.Name)] = model.LabelValue(l.Value)
|
|
||||||
}
|
|
||||||
return metric
|
|
||||||
}
|
|
||||||
|
|
||||||
// DecodeWriteRequest from an io.Reader into a prompb.WriteRequest, handling
|
// DecodeWriteRequest from an io.Reader into a prompb.WriteRequest, handling
|
||||||
// snappy decompression.
|
// snappy decompression.
|
||||||
// Used also by documentation/examples/remote_storage.
|
// Used also by documentation/examples/remote_storage.
|
||||||
|
|
|
@ -16,6 +16,7 @@ package remote
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io"
|
||||||
"sync"
|
"sync"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
@ -24,6 +25,7 @@ import (
|
||||||
"github.com/prometheus/common/model"
|
"github.com/prometheus/common/model"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
|
|
||||||
|
"github.com/prometheus/prometheus/config"
|
||||||
"github.com/prometheus/prometheus/model/histogram"
|
"github.com/prometheus/prometheus/model/histogram"
|
||||||
"github.com/prometheus/prometheus/model/labels"
|
"github.com/prometheus/prometheus/model/labels"
|
||||||
"github.com/prometheus/prometheus/model/metadata"
|
"github.com/prometheus/prometheus/model/metadata"
|
||||||
|
@ -705,3 +707,270 @@ func (c *mockChunkIterator) Next() bool {
|
||||||
func (c *mockChunkIterator) Err() error {
|
func (c *mockChunkIterator) Err() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestChunkedSeriesIterator(t *testing.T) {
|
||||||
|
t.Run("happy path", func(t *testing.T) {
|
||||||
|
chks := buildTestChunks(t)
|
||||||
|
|
||||||
|
it := newChunkedSeriesIterator(chks, 2000, 12000)
|
||||||
|
|
||||||
|
require.NoError(t, it.err)
|
||||||
|
require.NotNil(t, it.cur)
|
||||||
|
|
||||||
|
// Initial next; advance to first valid sample of first chunk.
|
||||||
|
res := it.Next()
|
||||||
|
require.Equal(t, chunkenc.ValFloat, res)
|
||||||
|
require.NoError(t, it.Err())
|
||||||
|
|
||||||
|
ts, v := it.At()
|
||||||
|
require.Equal(t, int64(2000), ts)
|
||||||
|
require.Equal(t, float64(2), v)
|
||||||
|
|
||||||
|
// Next to the second sample of the first chunk.
|
||||||
|
res = it.Next()
|
||||||
|
require.Equal(t, chunkenc.ValFloat, res)
|
||||||
|
require.NoError(t, it.Err())
|
||||||
|
|
||||||
|
ts, v = it.At()
|
||||||
|
require.Equal(t, int64(3000), ts)
|
||||||
|
require.Equal(t, float64(3), v)
|
||||||
|
|
||||||
|
// Attempt to seek to the first sample of the first chunk (should return current sample).
|
||||||
|
res = it.Seek(0)
|
||||||
|
require.Equal(t, chunkenc.ValFloat, res)
|
||||||
|
|
||||||
|
ts, v = it.At()
|
||||||
|
require.Equal(t, int64(3000), ts)
|
||||||
|
require.Equal(t, float64(3), v)
|
||||||
|
|
||||||
|
// Seek to the end of the first chunk.
|
||||||
|
res = it.Seek(4000)
|
||||||
|
require.Equal(t, chunkenc.ValFloat, res)
|
||||||
|
|
||||||
|
ts, v = it.At()
|
||||||
|
require.Equal(t, int64(4000), ts)
|
||||||
|
require.Equal(t, float64(4), v)
|
||||||
|
|
||||||
|
// Next to the first sample of the second chunk.
|
||||||
|
res = it.Next()
|
||||||
|
require.Equal(t, chunkenc.ValFloat, res)
|
||||||
|
require.NoError(t, it.Err())
|
||||||
|
|
||||||
|
ts, v = it.At()
|
||||||
|
require.Equal(t, int64(5000), ts)
|
||||||
|
require.Equal(t, float64(1), v)
|
||||||
|
|
||||||
|
// Seek to the second sample of the third chunk.
|
||||||
|
res = it.Seek(10999)
|
||||||
|
require.Equal(t, chunkenc.ValFloat, res)
|
||||||
|
require.NoError(t, it.Err())
|
||||||
|
|
||||||
|
ts, v = it.At()
|
||||||
|
require.Equal(t, int64(11000), ts)
|
||||||
|
require.Equal(t, float64(3), v)
|
||||||
|
|
||||||
|
// Attempt to seek to something past the last sample (should return false and exhaust the iterator).
|
||||||
|
res = it.Seek(99999)
|
||||||
|
require.Equal(t, chunkenc.ValNone, res)
|
||||||
|
require.NoError(t, it.Err())
|
||||||
|
|
||||||
|
// Attempt to next past the last sample (should return false as the iterator is exhausted).
|
||||||
|
res = it.Next()
|
||||||
|
require.Equal(t, chunkenc.ValNone, res)
|
||||||
|
require.NoError(t, it.Err())
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("invalid chunk encoding error", func(t *testing.T) {
|
||||||
|
chks := buildTestChunks(t)
|
||||||
|
|
||||||
|
// Set chunk type to an invalid value.
|
||||||
|
chks[0].Type = 8
|
||||||
|
|
||||||
|
it := newChunkedSeriesIterator(chks, 0, 14000)
|
||||||
|
|
||||||
|
res := it.Next()
|
||||||
|
require.Equal(t, chunkenc.ValNone, res)
|
||||||
|
|
||||||
|
res = it.Seek(1000)
|
||||||
|
require.Equal(t, chunkenc.ValNone, res)
|
||||||
|
|
||||||
|
require.ErrorContains(t, it.err, "invalid chunk encoding")
|
||||||
|
require.Nil(t, it.cur)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("empty chunks", func(t *testing.T) {
|
||||||
|
emptyChunks := make([]prompb.Chunk, 0)
|
||||||
|
|
||||||
|
it1 := newChunkedSeriesIterator(emptyChunks, 0, 1000)
|
||||||
|
require.Equal(t, chunkenc.ValNone, it1.Next())
|
||||||
|
require.Equal(t, chunkenc.ValNone, it1.Seek(1000))
|
||||||
|
require.NoError(t, it1.Err())
|
||||||
|
|
||||||
|
var nilChunks []prompb.Chunk
|
||||||
|
|
||||||
|
it2 := newChunkedSeriesIterator(nilChunks, 0, 1000)
|
||||||
|
require.Equal(t, chunkenc.ValNone, it2.Next())
|
||||||
|
require.Equal(t, chunkenc.ValNone, it2.Seek(1000))
|
||||||
|
require.NoError(t, it2.Err())
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestChunkedSeries(t *testing.T) {
|
||||||
|
t.Run("happy path", func(t *testing.T) {
|
||||||
|
chks := buildTestChunks(t)
|
||||||
|
|
||||||
|
s := chunkedSeries{
|
||||||
|
ChunkedSeries: prompb.ChunkedSeries{
|
||||||
|
Labels: []prompb.Label{
|
||||||
|
{Name: "foo", Value: "bar"},
|
||||||
|
{Name: "asdf", Value: "zxcv"},
|
||||||
|
},
|
||||||
|
Chunks: chks,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
require.Equal(t, labels.FromStrings("asdf", "zxcv", "foo", "bar"), s.Labels())
|
||||||
|
|
||||||
|
it := s.Iterator(nil)
|
||||||
|
res := it.Next() // Behavior is undefined w/o the initial call to Next.
|
||||||
|
|
||||||
|
require.Equal(t, chunkenc.ValFloat, res)
|
||||||
|
require.NoError(t, it.Err())
|
||||||
|
|
||||||
|
ts, v := it.At()
|
||||||
|
require.Equal(t, int64(0), ts)
|
||||||
|
require.Equal(t, float64(0), v)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestChunkedSeriesSet(t *testing.T) {
|
||||||
|
t.Run("happy path", func(t *testing.T) {
|
||||||
|
buf := &bytes.Buffer{}
|
||||||
|
flusher := &mockFlusher{}
|
||||||
|
|
||||||
|
w := NewChunkedWriter(buf, flusher)
|
||||||
|
r := NewChunkedReader(buf, config.DefaultChunkedReadLimit, nil)
|
||||||
|
|
||||||
|
chks := buildTestChunks(t)
|
||||||
|
l := []prompb.Label{
|
||||||
|
{Name: "foo", Value: "bar"},
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, c := range chks {
|
||||||
|
cSeries := prompb.ChunkedSeries{Labels: l, Chunks: []prompb.Chunk{c}}
|
||||||
|
readResp := prompb.ChunkedReadResponse{
|
||||||
|
ChunkedSeries: []*prompb.ChunkedSeries{&cSeries},
|
||||||
|
QueryIndex: int64(i),
|
||||||
|
}
|
||||||
|
|
||||||
|
b, err := proto.Marshal(&readResp)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
_, err = w.Write(b)
|
||||||
|
require.NoError(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ss := NewChunkedSeriesSet(r, io.NopCloser(buf), 0, 14000, func(error) {})
|
||||||
|
require.NoError(t, ss.Err())
|
||||||
|
require.Nil(t, ss.Warnings())
|
||||||
|
|
||||||
|
res := ss.Next()
|
||||||
|
require.True(t, res)
|
||||||
|
require.NoError(t, ss.Err())
|
||||||
|
|
||||||
|
s := ss.At()
|
||||||
|
require.Equal(t, 1, s.Labels().Len())
|
||||||
|
require.True(t, s.Labels().Has("foo"))
|
||||||
|
require.Equal(t, "bar", s.Labels().Get("foo"))
|
||||||
|
|
||||||
|
it := s.Iterator(nil)
|
||||||
|
it.Next()
|
||||||
|
ts, v := it.At()
|
||||||
|
require.Equal(t, int64(0), ts)
|
||||||
|
require.Equal(t, float64(0), v)
|
||||||
|
|
||||||
|
numResponses := 1
|
||||||
|
for ss.Next() {
|
||||||
|
numResponses++
|
||||||
|
}
|
||||||
|
require.Equal(t, numTestChunks, numResponses)
|
||||||
|
require.NoError(t, ss.Err())
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("chunked reader error", func(t *testing.T) {
|
||||||
|
buf := &bytes.Buffer{}
|
||||||
|
flusher := &mockFlusher{}
|
||||||
|
|
||||||
|
w := NewChunkedWriter(buf, flusher)
|
||||||
|
r := NewChunkedReader(buf, config.DefaultChunkedReadLimit, nil)
|
||||||
|
|
||||||
|
chks := buildTestChunks(t)
|
||||||
|
l := []prompb.Label{
|
||||||
|
{Name: "foo", Value: "bar"},
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, c := range chks {
|
||||||
|
cSeries := prompb.ChunkedSeries{Labels: l, Chunks: []prompb.Chunk{c}}
|
||||||
|
readResp := prompb.ChunkedReadResponse{
|
||||||
|
ChunkedSeries: []*prompb.ChunkedSeries{&cSeries},
|
||||||
|
QueryIndex: int64(i),
|
||||||
|
}
|
||||||
|
|
||||||
|
b, err := proto.Marshal(&readResp)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
b[0] = 0xFF // Corruption!
|
||||||
|
|
||||||
|
_, err = w.Write(b)
|
||||||
|
require.NoError(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ss := NewChunkedSeriesSet(r, io.NopCloser(buf), 0, 14000, func(error) {})
|
||||||
|
require.NoError(t, ss.Err())
|
||||||
|
require.Nil(t, ss.Warnings())
|
||||||
|
|
||||||
|
res := ss.Next()
|
||||||
|
require.False(t, res)
|
||||||
|
require.ErrorContains(t, ss.Err(), "proto: illegal wireType 7")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// mockFlusher implements http.Flusher.
|
||||||
|
type mockFlusher struct{}
|
||||||
|
|
||||||
|
func (f *mockFlusher) Flush() {}
|
||||||
|
|
||||||
|
const (
|
||||||
|
numTestChunks = 3
|
||||||
|
numSamplesPerTestChunk = 5
|
||||||
|
)
|
||||||
|
|
||||||
|
func buildTestChunks(t *testing.T) []prompb.Chunk {
|
||||||
|
startTime := int64(0)
|
||||||
|
chks := make([]prompb.Chunk, 0, numTestChunks)
|
||||||
|
|
||||||
|
time := startTime
|
||||||
|
|
||||||
|
for i := 0; i < numTestChunks; i++ {
|
||||||
|
c := chunkenc.NewXORChunk()
|
||||||
|
|
||||||
|
a, err := c.Appender()
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
minTimeMs := time
|
||||||
|
|
||||||
|
for j := 0; j < numSamplesPerTestChunk; j++ {
|
||||||
|
a.Append(time, float64(i+j))
|
||||||
|
time += int64(1000)
|
||||||
|
}
|
||||||
|
|
||||||
|
chks = append(chks, prompb.Chunk{
|
||||||
|
MinTimeMs: minTimeMs,
|
||||||
|
MaxTimeMs: time,
|
||||||
|
Type: prompb.Chunk_XOR,
|
||||||
|
Data: c.Bytes(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return chks
|
||||||
|
}
|
||||||
|
|
|
@ -24,7 +24,6 @@ import (
|
||||||
"slices"
|
"slices"
|
||||||
"sort"
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
"time"
|
|
||||||
"unicode/utf8"
|
"unicode/utf8"
|
||||||
|
|
||||||
"github.com/cespare/xxhash/v2"
|
"github.com/cespare/xxhash/v2"
|
||||||
|
@ -594,5 +593,5 @@ func addResourceTargetInfo(resource pcommon.Resource, settings Settings, timesta
|
||||||
|
|
||||||
// convertTimeStamp converts OTLP timestamp in ns to timestamp in ms
|
// convertTimeStamp converts OTLP timestamp in ns to timestamp in ms
|
||||||
func convertTimeStamp(timestamp pcommon.Timestamp) int64 {
|
func convertTimeStamp(timestamp pcommon.Timestamp) int64 {
|
||||||
return timestamp.AsTime().UnixNano() / (int64(time.Millisecond) / int64(time.Nanosecond))
|
return int64(timestamp) / 1_000_000
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,6 +14,7 @@ package prometheusremotewrite
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"go.opentelemetry.io/collector/pdata/pcommon"
|
"go.opentelemetry.io/collector/pdata/pcommon"
|
||||||
|
@ -159,3 +160,21 @@ func TestCreateAttributes(t *testing.T) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func Test_convertTimeStamp(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
arg pcommon.Timestamp
|
||||||
|
want int64
|
||||||
|
}{
|
||||||
|
{"zero", 0, 0},
|
||||||
|
{"1ms", 1_000_000, 1},
|
||||||
|
{"1s", pcommon.Timestamp(time.Unix(1, 0).UnixNano()), 1000},
|
||||||
|
}
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
got := convertTimeStamp(tt.arg)
|
||||||
|
assert.Equal(t, tt.want, got)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -165,11 +165,11 @@ func (q *querier) Select(ctx context.Context, sortSeries bool, hints *storage.Se
|
||||||
return storage.ErrSeriesSet(fmt.Errorf("toQuery: %w", err))
|
return storage.ErrSeriesSet(fmt.Errorf("toQuery: %w", err))
|
||||||
}
|
}
|
||||||
|
|
||||||
res, err := q.client.Read(ctx, query)
|
res, err := q.client.Read(ctx, query, sortSeries)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return storage.ErrSeriesSet(fmt.Errorf("remote_read: %w", err))
|
return storage.ErrSeriesSet(fmt.Errorf("remote_read: %w", err))
|
||||||
}
|
}
|
||||||
return newSeriesSetFilter(FromQueryResult(sortSeries, res), added)
|
return newSeriesSetFilter(res, added)
|
||||||
}
|
}
|
||||||
|
|
||||||
// addExternalLabels adds matchers for each external label. External labels
|
// addExternalLabels adds matchers for each external label. External labels
|
||||||
|
|
|
@ -179,7 +179,7 @@ func BenchmarkStreamReadEndpoint(b *testing.B) {
|
||||||
require.Equal(b, 2, recorder.Code/100)
|
require.Equal(b, 2, recorder.Code/100)
|
||||||
|
|
||||||
var results []*prompb.ChunkedReadResponse
|
var results []*prompb.ChunkedReadResponse
|
||||||
stream := NewChunkedReader(recorder.Result().Body, DefaultChunkedReadLimit, nil)
|
stream := NewChunkedReader(recorder.Result().Body, config.DefaultChunkedReadLimit, nil)
|
||||||
|
|
||||||
for {
|
for {
|
||||||
res := &prompb.ChunkedReadResponse{}
|
res := &prompb.ChunkedReadResponse{}
|
||||||
|
@ -280,7 +280,7 @@ func TestStreamReadEndpoint(t *testing.T) {
|
||||||
require.Equal(t, "", recorder.Result().Header.Get("Content-Encoding"))
|
require.Equal(t, "", recorder.Result().Header.Get("Content-Encoding"))
|
||||||
|
|
||||||
var results []*prompb.ChunkedReadResponse
|
var results []*prompb.ChunkedReadResponse
|
||||||
stream := NewChunkedReader(recorder.Result().Body, DefaultChunkedReadLimit, nil)
|
stream := NewChunkedReader(recorder.Result().Body, config.DefaultChunkedReadLimit, nil)
|
||||||
for {
|
for {
|
||||||
res := &prompb.ChunkedReadResponse{}
|
res := &prompb.ChunkedReadResponse{}
|
||||||
err := stream.NextProto(res)
|
err := stream.NextProto(res)
|
||||||
|
|
|
@ -27,6 +27,7 @@ import (
|
||||||
"github.com/prometheus/prometheus/config"
|
"github.com/prometheus/prometheus/config"
|
||||||
"github.com/prometheus/prometheus/model/labels"
|
"github.com/prometheus/prometheus/model/labels"
|
||||||
"github.com/prometheus/prometheus/prompb"
|
"github.com/prometheus/prometheus/prompb"
|
||||||
|
"github.com/prometheus/prometheus/storage"
|
||||||
"github.com/prometheus/prometheus/util/annotations"
|
"github.com/prometheus/prometheus/util/annotations"
|
||||||
"github.com/prometheus/prometheus/util/testutil"
|
"github.com/prometheus/prometheus/util/testutil"
|
||||||
)
|
)
|
||||||
|
@ -198,7 +199,7 @@ type mockedRemoteClient struct {
|
||||||
b labels.ScratchBuilder
|
b labels.ScratchBuilder
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *mockedRemoteClient) Read(_ context.Context, query *prompb.Query) (*prompb.QueryResult, error) {
|
func (c *mockedRemoteClient) Read(_ context.Context, query *prompb.Query, sortSeries bool) (storage.SeriesSet, error) {
|
||||||
if c.got != nil {
|
if c.got != nil {
|
||||||
return nil, fmt.Errorf("expected only one call to remote client got: %v", query)
|
return nil, fmt.Errorf("expected only one call to remote client got: %v", query)
|
||||||
}
|
}
|
||||||
|
@ -227,7 +228,7 @@ func (c *mockedRemoteClient) Read(_ context.Context, query *prompb.Query) (*prom
|
||||||
q.Timeseries = append(q.Timeseries, &prompb.TimeSeries{Labels: s.Labels})
|
q.Timeseries = append(q.Timeseries, &prompb.TimeSeries{Labels: s.Labels})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return q, nil
|
return FromQueryResult(sortSeries, q), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *mockedRemoteClient) reset() {
|
func (c *mockedRemoteClient) reset() {
|
||||||
|
|
|
@ -115,6 +115,7 @@ func (s *Storage) ApplyConfig(conf *config.Config) error {
|
||||||
c, err := NewReadClient(name, &ClientConfig{
|
c, err := NewReadClient(name, &ClientConfig{
|
||||||
URL: rrConf.URL,
|
URL: rrConf.URL,
|
||||||
Timeout: rrConf.RemoteTimeout,
|
Timeout: rrConf.RemoteTimeout,
|
||||||
|
ChunkedReadLimit: rrConf.ChunkedReadLimit,
|
||||||
HTTPClientConfig: rrConf.HTTPClientConfig,
|
HTTPClientConfig: rrConf.HTTPClientConfig,
|
||||||
Headers: rrConf.Headers,
|
Headers: rrConf.Headers,
|
||||||
})
|
})
|
||||||
|
|
|
@ -28,6 +28,7 @@ import (
|
||||||
"github.com/golang/snappy"
|
"github.com/golang/snappy"
|
||||||
"github.com/prometheus/client_golang/prometheus"
|
"github.com/prometheus/client_golang/prometheus"
|
||||||
"github.com/prometheus/client_golang/prometheus/promauto"
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
||||||
|
"github.com/prometheus/common/model"
|
||||||
|
|
||||||
"github.com/prometheus/prometheus/config"
|
"github.com/prometheus/prometheus/config"
|
||||||
"github.com/prometheus/prometheus/model/exemplar"
|
"github.com/prometheus/prometheus/model/exemplar"
|
||||||
|
@ -239,7 +240,7 @@ func (h *writeHandler) write(ctx context.Context, req *prompb.WriteRequest) (err
|
||||||
|
|
||||||
// TODO(bwplotka): Even as per 1.0 spec, this should be a 400 error, while other samples are
|
// TODO(bwplotka): Even as per 1.0 spec, this should be a 400 error, while other samples are
|
||||||
// potentially written. Perhaps unify with fixed writeV2 implementation a bit.
|
// potentially written. Perhaps unify with fixed writeV2 implementation a bit.
|
||||||
if !ls.Has(labels.MetricName) || !ls.IsValid() {
|
if !ls.Has(labels.MetricName) || !ls.IsValid(model.NameValidationScheme) {
|
||||||
level.Warn(h.logger).Log("msg", "Invalid metric names or labels", "got", ls.String())
|
level.Warn(h.logger).Log("msg", "Invalid metric names or labels", "got", ls.String())
|
||||||
samplesWithInvalidLabels++
|
samplesWithInvalidLabels++
|
||||||
continue
|
continue
|
||||||
|
@ -380,7 +381,7 @@ func (h *writeHandler) appendV2(app storage.Appender, req *writev2.Request, rs *
|
||||||
// Validate series labels early.
|
// Validate series labels early.
|
||||||
// NOTE(bwplotka): While spec allows UTF-8, Prometheus Receiver may impose
|
// NOTE(bwplotka): While spec allows UTF-8, Prometheus Receiver may impose
|
||||||
// specific limits and follow https://prometheus.io/docs/specs/remote_write_spec_2_0/#invalid-samples case.
|
// specific limits and follow https://prometheus.io/docs/specs/remote_write_spec_2_0/#invalid-samples case.
|
||||||
if !ls.Has(labels.MetricName) || !ls.IsValid() {
|
if !ls.Has(labels.MetricName) || !ls.IsValid(model.NameValidationScheme) {
|
||||||
badRequestErrs = append(badRequestErrs, fmt.Errorf("invalid metric name or labels, got %v", ls.String()))
|
badRequestErrs = append(badRequestErrs, fmt.Errorf("invalid metric name or labels, got %v", ls.String()))
|
||||||
samplesWithInvalidLabels += len(ts.Samples) + len(ts.Histograms)
|
samplesWithInvalidLabels += len(ts.Samples) + len(ts.Histograms)
|
||||||
continue
|
continue
|
||||||
|
|
|
@ -133,9 +133,6 @@ type Meta struct {
|
||||||
// Time range the data covers.
|
// Time range the data covers.
|
||||||
// When MaxTime == math.MaxInt64 the chunk is still open and being appended to.
|
// When MaxTime == math.MaxInt64 the chunk is still open and being appended to.
|
||||||
MinTime, MaxTime int64
|
MinTime, MaxTime int64
|
||||||
|
|
||||||
// Flag to indicate that this meta needs merge with OOO data.
|
|
||||||
MergeOOO bool
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ChunkFromSamples requires all samples to have the same type.
|
// ChunkFromSamples requires all samples to have the same type.
|
||||||
|
|
|
@ -2018,7 +2018,7 @@ func TestDelayedCompaction(t *testing.T) {
|
||||||
// This implies that the compaction delay doesn't block or wait on the initial trigger.
|
// This implies that the compaction delay doesn't block or wait on the initial trigger.
|
||||||
// 3 is an arbitrary value because it's difficult to determine the precise value.
|
// 3 is an arbitrary value because it's difficult to determine the precise value.
|
||||||
require.GreaterOrEqual(t, prom_testutil.ToFloat64(db.metrics.compactionsTriggered)-prom_testutil.ToFloat64(db.metrics.compactionsSkipped), 3.0)
|
require.GreaterOrEqual(t, prom_testutil.ToFloat64(db.metrics.compactionsTriggered)-prom_testutil.ToFloat64(db.metrics.compactionsSkipped), 3.0)
|
||||||
// The delay doesn't change the head blocks alignement.
|
// The delay doesn't change the head blocks alignment.
|
||||||
require.Eventually(t, func() bool {
|
require.Eventually(t, func() bool {
|
||||||
return db.head.MinTime() == db.compactor.(*LeveledCompactor).ranges[0]+1
|
return db.head.MinTime() == db.compactor.(*LeveledCompactor).ranges[0]+1
|
||||||
}, 500*time.Millisecond, 10*time.Millisecond)
|
}, 500*time.Millisecond, 10*time.Millisecond)
|
||||||
|
|
46
tsdb/head.go
46
tsdb/head.go
|
@ -178,7 +178,6 @@ type HeadOptions struct {
|
||||||
WALReplayConcurrency int
|
WALReplayConcurrency int
|
||||||
|
|
||||||
// EnableSharding enables ShardedPostings() support in the Head.
|
// EnableSharding enables ShardedPostings() support in the Head.
|
||||||
// EnableSharding is temporarily disabled during Init().
|
|
||||||
EnableSharding bool
|
EnableSharding bool
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -610,7 +609,7 @@ const cardinalityCacheExpirationTime = time.Duration(30) * time.Second
|
||||||
// Init loads data from the write ahead log and prepares the head for writes.
|
// Init loads data from the write ahead log and prepares the head for writes.
|
||||||
// It should be called before using an appender so that it
|
// It should be called before using an appender so that it
|
||||||
// limits the ingested samples to the head min valid time.
|
// limits the ingested samples to the head min valid time.
|
||||||
func (h *Head) Init(minValidTime int64) (err error) {
|
func (h *Head) Init(minValidTime int64) error {
|
||||||
h.minValidTime.Store(minValidTime)
|
h.minValidTime.Store(minValidTime)
|
||||||
defer func() {
|
defer func() {
|
||||||
h.postings.EnsureOrder(h.opts.WALReplayConcurrency)
|
h.postings.EnsureOrder(h.opts.WALReplayConcurrency)
|
||||||
|
@ -624,24 +623,6 @@ func (h *Head) Init(minValidTime int64) (err error) {
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
// If sharding is enabled, disable it while initializing, and calculate the shards later.
|
|
||||||
// We're going to use that field for other purposes during WAL replay,
|
|
||||||
// so we don't want to waste time on calculating the shard that we're going to lose anyway.
|
|
||||||
if h.opts.EnableSharding {
|
|
||||||
h.opts.EnableSharding = false
|
|
||||||
defer func() {
|
|
||||||
h.opts.EnableSharding = true
|
|
||||||
if err == nil {
|
|
||||||
// No locking is needed here as nobody should be writing while we're in Init.
|
|
||||||
for _, stripe := range h.series.series {
|
|
||||||
for _, s := range stripe {
|
|
||||||
s.shardHashOrMemoryMappedMaxTime = labels.StableHash(s.lset)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
}
|
|
||||||
|
|
||||||
level.Info(h.logger).Log("msg", "Replaying on-disk memory mappable chunks if any")
|
level.Info(h.logger).Log("msg", "Replaying on-disk memory mappable chunks if any")
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
|
|
||||||
|
@ -702,6 +683,7 @@ func (h *Head) Init(minValidTime int64) (err error) {
|
||||||
mmappedChunks map[chunks.HeadSeriesRef][]*mmappedChunk
|
mmappedChunks map[chunks.HeadSeriesRef][]*mmappedChunk
|
||||||
oooMmappedChunks map[chunks.HeadSeriesRef][]*mmappedChunk
|
oooMmappedChunks map[chunks.HeadSeriesRef][]*mmappedChunk
|
||||||
lastMmapRef chunks.ChunkDiskMapperRef
|
lastMmapRef chunks.ChunkDiskMapperRef
|
||||||
|
err error
|
||||||
|
|
||||||
mmapChunkReplayDuration time.Duration
|
mmapChunkReplayDuration time.Duration
|
||||||
)
|
)
|
||||||
|
@ -2086,11 +2068,9 @@ type memSeries struct {
|
||||||
ref chunks.HeadSeriesRef
|
ref chunks.HeadSeriesRef
|
||||||
meta *metadata.Metadata
|
meta *metadata.Metadata
|
||||||
|
|
||||||
// Series labels hash to use for sharding purposes.
|
// Series labels hash to use for sharding purposes. The value is always 0 when sharding has not
|
||||||
// The value is always 0 when sharding has not been explicitly enabled in TSDB.
|
// been explicitly enabled in TSDB.
|
||||||
// While the WAL replay the value stored here is the max time of any mmapped chunk,
|
shardHash uint64
|
||||||
// and the shard hash is re-calculated after WAL replay is complete.
|
|
||||||
shardHashOrMemoryMappedMaxTime uint64
|
|
||||||
|
|
||||||
// Everything after here should only be accessed with the lock held.
|
// Everything after here should only be accessed with the lock held.
|
||||||
sync.Mutex
|
sync.Mutex
|
||||||
|
@ -2115,6 +2095,8 @@ type memSeries struct {
|
||||||
|
|
||||||
ooo *memSeriesOOOFields
|
ooo *memSeriesOOOFields
|
||||||
|
|
||||||
|
mmMaxTime int64 // Max time of any mmapped chunk, only used during WAL replay.
|
||||||
|
|
||||||
nextAt int64 // Timestamp at which to cut the next chunk.
|
nextAt int64 // Timestamp at which to cut the next chunk.
|
||||||
histogramChunkHasComputedEndTime bool // True if nextAt has been predicted for the current histograms chunk; false otherwise.
|
histogramChunkHasComputedEndTime bool // True if nextAt has been predicted for the current histograms chunk; false otherwise.
|
||||||
pendingCommit bool // Whether there are samples waiting to be committed to this series.
|
pendingCommit bool // Whether there are samples waiting to be committed to this series.
|
||||||
|
@ -2145,10 +2127,10 @@ type memSeriesOOOFields struct {
|
||||||
|
|
||||||
func newMemSeries(lset labels.Labels, id chunks.HeadSeriesRef, shardHash uint64, isolationDisabled bool) *memSeries {
|
func newMemSeries(lset labels.Labels, id chunks.HeadSeriesRef, shardHash uint64, isolationDisabled bool) *memSeries {
|
||||||
s := &memSeries{
|
s := &memSeries{
|
||||||
lset: lset,
|
lset: lset,
|
||||||
ref: id,
|
ref: id,
|
||||||
nextAt: math.MinInt64,
|
nextAt: math.MinInt64,
|
||||||
shardHashOrMemoryMappedMaxTime: shardHash,
|
shardHash: shardHash,
|
||||||
}
|
}
|
||||||
if !isolationDisabled {
|
if !isolationDisabled {
|
||||||
s.txs = newTxRing(0)
|
s.txs = newTxRing(0)
|
||||||
|
@ -2236,12 +2218,6 @@ func (s *memSeries) truncateChunksBefore(mint int64, minOOOMmapRef chunks.ChunkD
|
||||||
return removedInOrder + removedOOO
|
return removedInOrder + removedOOO
|
||||||
}
|
}
|
||||||
|
|
||||||
// shardHash returns the shard hash of the series, only available after WAL replay.
|
|
||||||
func (s *memSeries) shardHash() uint64 { return s.shardHashOrMemoryMappedMaxTime }
|
|
||||||
|
|
||||||
// mmMaxTime returns the max time of any mmapped chunk in the series, only available during WAL replay.
|
|
||||||
func (s *memSeries) mmMaxTime() int64 { return int64(s.shardHashOrMemoryMappedMaxTime) }
|
|
||||||
|
|
||||||
// cleanupAppendIDsBelow cleans up older appendIDs. Has to be called after
|
// cleanupAppendIDsBelow cleans up older appendIDs. Has to be called after
|
||||||
// acquiring lock.
|
// acquiring lock.
|
||||||
func (s *memSeries) cleanupAppendIDsBelow(bound uint64) {
|
func (s *memSeries) cleanupAppendIDsBelow(bound uint64) {
|
||||||
|
|
|
@ -170,7 +170,7 @@ func (h *headIndexReader) ShardedPostings(p index.Postings, shardIndex, shardCou
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if the series belong to the shard.
|
// Check if the series belong to the shard.
|
||||||
if s.shardHash()%shardCount != shardIndex {
|
if s.shardHash%shardCount != shardIndex {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -366,7 +366,7 @@ func (h *headChunkReader) ChunkOrIterableWithCopy(meta chunks.Meta) (chunkenc.Ch
|
||||||
// If copyLastChunk is true, then it makes a copy of the head chunk if asked for it.
|
// If copyLastChunk is true, then it makes a copy of the head chunk if asked for it.
|
||||||
// Also returns max time of the chunk.
|
// Also returns max time of the chunk.
|
||||||
func (h *headChunkReader) chunk(meta chunks.Meta, copyLastChunk bool) (chunkenc.Chunk, int64, error) {
|
func (h *headChunkReader) chunk(meta chunks.Meta, copyLastChunk bool) (chunkenc.Chunk, int64, error) {
|
||||||
sid, cid := chunks.HeadChunkRef(meta.Ref).Unpack()
|
sid, cid, isOOO := unpackHeadChunkRef(meta.Ref)
|
||||||
|
|
||||||
s := h.head.series.getByID(sid)
|
s := h.head.series.getByID(sid)
|
||||||
// This means that the series has been garbage collected.
|
// This means that the series has been garbage collected.
|
||||||
|
@ -376,12 +376,21 @@ func (h *headChunkReader) chunk(meta chunks.Meta, copyLastChunk bool) (chunkenc.
|
||||||
|
|
||||||
s.Lock()
|
s.Lock()
|
||||||
defer s.Unlock()
|
defer s.Unlock()
|
||||||
return h.chunkFromSeries(s, cid, copyLastChunk)
|
return h.head.chunkFromSeries(s, cid, isOOO, h.mint, h.maxt, h.isoState, copyLastChunk)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dumb thing to defeat chunk pool.
|
||||||
|
type wrapOOOHeadChunk struct {
|
||||||
|
chunkenc.Chunk
|
||||||
}
|
}
|
||||||
|
|
||||||
// Call with s locked.
|
// Call with s locked.
|
||||||
func (h *headChunkReader) chunkFromSeries(s *memSeries, cid chunks.HeadChunkID, copyLastChunk bool) (chunkenc.Chunk, int64, error) {
|
func (h *Head) chunkFromSeries(s *memSeries, cid chunks.HeadChunkID, isOOO bool, mint, maxt int64, isoState *isolationState, copyLastChunk bool) (chunkenc.Chunk, int64, error) {
|
||||||
c, headChunk, isOpen, err := s.chunk(cid, h.head.chunkDiskMapper, &h.head.memChunkPool)
|
if isOOO {
|
||||||
|
chk, maxTime, err := s.oooChunk(cid, h.chunkDiskMapper, &h.memChunkPool)
|
||||||
|
return wrapOOOHeadChunk{chk}, maxTime, err
|
||||||
|
}
|
||||||
|
c, headChunk, isOpen, err := s.chunk(cid, h.chunkDiskMapper, &h.memChunkPool)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, 0, err
|
return nil, 0, err
|
||||||
}
|
}
|
||||||
|
@ -390,12 +399,12 @@ func (h *headChunkReader) chunkFromSeries(s *memSeries, cid chunks.HeadChunkID,
|
||||||
// Set this to nil so that Go GC can collect it after it has been used.
|
// Set this to nil so that Go GC can collect it after it has been used.
|
||||||
c.chunk = nil
|
c.chunk = nil
|
||||||
c.prev = nil
|
c.prev = nil
|
||||||
h.head.memChunkPool.Put(c)
|
h.memChunkPool.Put(c)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
// This means that the chunk is outside the specified range.
|
// This means that the chunk is outside the specified range.
|
||||||
if !c.OverlapsClosedInterval(h.mint, h.maxt) {
|
if !c.OverlapsClosedInterval(mint, maxt) {
|
||||||
return nil, 0, storage.ErrNotFound
|
return nil, 0, storage.ErrNotFound
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -407,7 +416,7 @@ func (h *headChunkReader) chunkFromSeries(s *memSeries, cid chunks.HeadChunkID,
|
||||||
newB := make([]byte, len(b))
|
newB := make([]byte, len(b))
|
||||||
copy(newB, b) // TODO(codesome): Use bytes.Clone() when we upgrade to Go 1.20.
|
copy(newB, b) // TODO(codesome): Use bytes.Clone() when we upgrade to Go 1.20.
|
||||||
// TODO(codesome): Put back in the pool (non-trivial).
|
// TODO(codesome): Put back in the pool (non-trivial).
|
||||||
chk, err = h.head.opts.ChunkPool.Get(s.headChunks.chunk.Encoding(), newB)
|
chk, err = h.opts.ChunkPool.Get(s.headChunks.chunk.Encoding(), newB)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, 0, err
|
return nil, 0, err
|
||||||
}
|
}
|
||||||
|
@ -417,7 +426,7 @@ func (h *headChunkReader) chunkFromSeries(s *memSeries, cid chunks.HeadChunkID,
|
||||||
Chunk: chk,
|
Chunk: chk,
|
||||||
s: s,
|
s: s,
|
||||||
cid: cid,
|
cid: cid,
|
||||||
isoState: h.isoState,
|
isoState: isoState,
|
||||||
}, maxTime, nil
|
}, maxTime, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -430,7 +439,7 @@ func (s *memSeries) chunk(id chunks.HeadChunkID, chunkDiskMapper *chunks.ChunkDi
|
||||||
// incremented by 1 when new chunk is created, hence (id - firstChunkID) gives the slice index.
|
// incremented by 1 when new chunk is created, hence (id - firstChunkID) gives the slice index.
|
||||||
// The max index for the s.mmappedChunks slice can be len(s.mmappedChunks)-1, hence if the ix
|
// The max index for the s.mmappedChunks slice can be len(s.mmappedChunks)-1, hence if the ix
|
||||||
// is >= len(s.mmappedChunks), it represents one of the chunks on s.headChunks linked list.
|
// is >= len(s.mmappedChunks), it represents one of the chunks on s.headChunks linked list.
|
||||||
// The order of elemens is different for slice and linked list.
|
// The order of elements is different for slice and linked list.
|
||||||
// For s.mmappedChunks slice newer chunks are appended to it.
|
// For s.mmappedChunks slice newer chunks are appended to it.
|
||||||
// For s.headChunks list newer chunks are prepended to it.
|
// For s.headChunks list newer chunks are prepended to it.
|
||||||
//
|
//
|
||||||
|
@ -481,85 +490,19 @@ func (s *memSeries) chunk(id chunks.HeadChunkID, chunkDiskMapper *chunks.ChunkDi
|
||||||
return elem, true, offset == 0, nil
|
return elem, true, offset == 0, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// mergedChunks return an iterable over all chunks that overlap the
|
// oooChunk returns the chunk for the HeadChunkID by m-mapping it from the disk.
|
||||||
// time window [mint,maxt], plus meta.Chunk if populated.
|
// It never returns the head OOO chunk.
|
||||||
// If hr is non-nil then in-order chunks are included.
|
func (s *memSeries) oooChunk(id chunks.HeadChunkID, chunkDiskMapper *chunks.ChunkDiskMapper, memChunkPool *sync.Pool) (chunk chunkenc.Chunk, maxTime int64, err error) {
|
||||||
// This function is not thread safe unless the caller holds a lock.
|
// ix represents the index of chunk in the s.ooo.oooMmappedChunks slice. The chunk id's are
|
||||||
// The caller must ensure that s.ooo is not nil.
|
// incremented by 1 when new chunk is created, hence (id - firstOOOChunkID) gives the slice index.
|
||||||
func (s *memSeries) mergedChunks(meta chunks.Meta, cdm *chunks.ChunkDiskMapper, hr *headChunkReader, mint, maxt int64, maxMmapRef chunks.ChunkDiskMapperRef) (chunkenc.Iterable, error) {
|
ix := int(id) - int(s.ooo.firstOOOChunkID)
|
||||||
// We create a temporary slice of chunk metas to hold the information of all
|
|
||||||
// possible chunks that may overlap with the requested chunk.
|
|
||||||
tmpChks := make([]chunkMetaAndChunkDiskMapperRef, 0, len(s.ooo.oooMmappedChunks)+1)
|
|
||||||
|
|
||||||
for i, c := range s.ooo.oooMmappedChunks {
|
if ix < 0 || ix >= len(s.ooo.oooMmappedChunks) {
|
||||||
if maxMmapRef != 0 && c.ref > maxMmapRef {
|
return nil, 0, storage.ErrNotFound
|
||||||
break
|
|
||||||
}
|
|
||||||
if c.OverlapsClosedInterval(mint, maxt) {
|
|
||||||
tmpChks = append(tmpChks, chunkMetaAndChunkDiskMapperRef{
|
|
||||||
meta: chunks.Meta{
|
|
||||||
MinTime: c.minTime,
|
|
||||||
MaxTime: c.maxTime,
|
|
||||||
Ref: chunks.ChunkRef(chunks.NewHeadChunkRef(s.ref, s.oooHeadChunkID(i))),
|
|
||||||
},
|
|
||||||
ref: c.ref,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Add in data copied from the head OOO chunk.
|
|
||||||
if meta.Chunk != nil {
|
|
||||||
tmpChks = append(tmpChks, chunkMetaAndChunkDiskMapperRef{meta: meta})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if hr != nil { // Include in-order chunks.
|
chk, err := chunkDiskMapper.Chunk(s.ooo.oooMmappedChunks[ix].ref)
|
||||||
metas := appendSeriesChunks(s, max(meta.MinTime, mint), min(meta.MaxTime, maxt), nil)
|
return chk, s.ooo.oooMmappedChunks[ix].maxTime, err
|
||||||
for _, m := range metas {
|
|
||||||
tmpChks = append(tmpChks, chunkMetaAndChunkDiskMapperRef{
|
|
||||||
meta: m,
|
|
||||||
ref: 0, // This tells the loop below it's an in-order head chunk.
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Next we want to sort all the collected chunks by min time so we can find
|
|
||||||
// those that overlap and stop when we know the rest don't.
|
|
||||||
slices.SortFunc(tmpChks, refLessByMinTimeAndMinRef)
|
|
||||||
|
|
||||||
mc := &mergedOOOChunks{}
|
|
||||||
absoluteMax := int64(math.MinInt64)
|
|
||||||
for _, c := range tmpChks {
|
|
||||||
if c.meta.Ref != meta.Ref && (len(mc.chunkIterables) == 0 || c.meta.MinTime > absoluteMax) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
var iterable chunkenc.Iterable
|
|
||||||
switch {
|
|
||||||
case c.meta.Chunk != nil:
|
|
||||||
iterable = c.meta.Chunk
|
|
||||||
case c.ref == 0: // This is an in-order head chunk.
|
|
||||||
_, cid := chunks.HeadChunkRef(c.meta.Ref).Unpack()
|
|
||||||
var err error
|
|
||||||
iterable, _, err = hr.chunkFromSeries(s, cid, false)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("invalid head chunk: %w", err)
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
chk, err := cdm.Chunk(c.ref)
|
|
||||||
if err != nil {
|
|
||||||
var cerr *chunks.CorruptionErr
|
|
||||||
if errors.As(err, &cerr) {
|
|
||||||
return nil, fmt.Errorf("invalid ooo mmapped chunk: %w", err)
|
|
||||||
}
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
iterable = chk
|
|
||||||
}
|
|
||||||
mc.chunkIterables = append(mc.chunkIterables, iterable)
|
|
||||||
if c.meta.MaxTime > absoluteMax {
|
|
||||||
absoluteMax = c.meta.MaxTime
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return mc, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// safeHeadChunk makes sure that the chunk can be accessed without a race condition.
|
// safeHeadChunk makes sure that the chunk can be accessed without a race condition.
|
||||||
|
|
|
@ -23,7 +23,6 @@ import (
|
||||||
"path"
|
"path"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"reflect"
|
"reflect"
|
||||||
"runtime/pprof"
|
|
||||||
"sort"
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
@ -90,43 +89,6 @@ func newTestHeadWithOptions(t testing.TB, compressWAL wlog.CompressionType, opts
|
||||||
return h, wal
|
return h, wal
|
||||||
}
|
}
|
||||||
|
|
||||||
// BenchmarkLoadRealWLs will be skipped unless the BENCHMARK_LOAD_REAL_WLS_DIR environment variable is set.
|
|
||||||
// BENCHMARK_LOAD_REAL_WLS_DIR should be the folder where `wal` and `chunks_head` are located.
|
|
||||||
// Optionally, BENCHMARK_LOAD_REAL_WLS_PROFILE can be set to a file path to write a CPU profile.
|
|
||||||
func BenchmarkLoadRealWLs(b *testing.B) {
|
|
||||||
dir := os.Getenv("BENCHMARK_LOAD_REAL_WLS_DIR")
|
|
||||||
if dir == "" {
|
|
||||||
b.Skipped()
|
|
||||||
}
|
|
||||||
|
|
||||||
profileFile := os.Getenv("BENCHMARK_LOAD_REAL_WLS_PROFILE")
|
|
||||||
if profileFile != "" {
|
|
||||||
b.Logf("Will profile in %s", profileFile)
|
|
||||||
f, err := os.Create(profileFile)
|
|
||||||
require.NoError(b, err)
|
|
||||||
b.Cleanup(func() { f.Close() })
|
|
||||||
require.NoError(b, pprof.StartCPUProfile(f))
|
|
||||||
b.Cleanup(pprof.StopCPUProfile)
|
|
||||||
}
|
|
||||||
|
|
||||||
wal, err := wlog.New(nil, nil, filepath.Join(dir, "wal"), wlog.CompressionNone)
|
|
||||||
require.NoError(b, err)
|
|
||||||
b.Cleanup(func() { wal.Close() })
|
|
||||||
|
|
||||||
wbl, err := wlog.New(nil, nil, filepath.Join(dir, "wbl"), wlog.CompressionNone)
|
|
||||||
require.NoError(b, err)
|
|
||||||
b.Cleanup(func() { wbl.Close() })
|
|
||||||
|
|
||||||
// Load the WAL.
|
|
||||||
for i := 0; i < b.N; i++ {
|
|
||||||
opts := DefaultHeadOptions()
|
|
||||||
opts.ChunkDirRoot = dir
|
|
||||||
h, err := NewHead(nil, nil, wal, wbl, opts, nil)
|
|
||||||
require.NoError(b, err)
|
|
||||||
h.Init(0)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func BenchmarkCreateSeries(b *testing.B) {
|
func BenchmarkCreateSeries(b *testing.B) {
|
||||||
series := genSeries(b.N, 10, 0, 0)
|
series := genSeries(b.N, 10, 0, 0)
|
||||||
h, _ := newTestHead(b, 10000, wlog.CompressionNone, false)
|
h, _ := newTestHead(b, 10000, wlog.CompressionNone, false)
|
||||||
|
|
|
@ -435,8 +435,6 @@ Outer:
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func minInt64() int64 { return math.MinInt64 }
|
|
||||||
|
|
||||||
// resetSeriesWithMMappedChunks is only used during the WAL replay.
|
// resetSeriesWithMMappedChunks is only used during the WAL replay.
|
||||||
func (h *Head) resetSeriesWithMMappedChunks(mSeries *memSeries, mmc, oooMmc []*mmappedChunk, walSeriesRef chunks.HeadSeriesRef) (overlapped bool) {
|
func (h *Head) resetSeriesWithMMappedChunks(mSeries *memSeries, mmc, oooMmc []*mmappedChunk, walSeriesRef chunks.HeadSeriesRef) (overlapped bool) {
|
||||||
if mSeries.ref != walSeriesRef {
|
if mSeries.ref != walSeriesRef {
|
||||||
|
@ -483,11 +481,10 @@ func (h *Head) resetSeriesWithMMappedChunks(mSeries *memSeries, mmc, oooMmc []*m
|
||||||
}
|
}
|
||||||
// Cache the last mmapped chunk time, so we can skip calling append() for samples it will reject.
|
// Cache the last mmapped chunk time, so we can skip calling append() for samples it will reject.
|
||||||
if len(mmc) == 0 {
|
if len(mmc) == 0 {
|
||||||
mSeries.shardHashOrMemoryMappedMaxTime = uint64(minInt64())
|
mSeries.mmMaxTime = math.MinInt64
|
||||||
} else {
|
} else {
|
||||||
mmMaxTime := mmc[len(mmc)-1].maxTime
|
mSeries.mmMaxTime = mmc[len(mmc)-1].maxTime
|
||||||
mSeries.shardHashOrMemoryMappedMaxTime = uint64(mmMaxTime)
|
h.updateMinMaxTime(mmc[0].minTime, mSeries.mmMaxTime)
|
||||||
h.updateMinMaxTime(mmc[0].minTime, mmMaxTime)
|
|
||||||
}
|
}
|
||||||
if len(oooMmc) != 0 {
|
if len(oooMmc) != 0 {
|
||||||
// Mint and maxt can be in any chunk, they are not sorted.
|
// Mint and maxt can be in any chunk, they are not sorted.
|
||||||
|
@ -588,7 +585,7 @@ func (wp *walSubsetProcessor) processWALSamples(h *Head, mmappedChunks, oooMmapp
|
||||||
unknownRefs++
|
unknownRefs++
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if s.T <= ms.mmMaxTime() {
|
if s.T <= ms.mmMaxTime {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if _, chunkCreated := ms.append(s.T, s.V, 0, appendChunkOpts); chunkCreated {
|
if _, chunkCreated := ms.append(s.T, s.V, 0, appendChunkOpts); chunkCreated {
|
||||||
|
@ -617,7 +614,7 @@ func (wp *walSubsetProcessor) processWALSamples(h *Head, mmappedChunks, oooMmapp
|
||||||
unknownHistogramRefs++
|
unknownHistogramRefs++
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if s.t <= ms.mmMaxTime() {
|
if s.t <= ms.mmMaxTime {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
var chunkCreated bool
|
var chunkCreated bool
|
||||||
|
|
|
@ -16,6 +16,7 @@ package tsdb
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
|
"fmt"
|
||||||
"math"
|
"math"
|
||||||
"slices"
|
"slices"
|
||||||
|
|
||||||
|
@ -91,11 +92,10 @@ func getOOOSeriesChunks(s *memSeries, mint, maxt int64, lastGarbageCollectedMmap
|
||||||
|
|
||||||
addChunk := func(minT, maxT int64, ref chunks.ChunkRef, chunk chunkenc.Chunk) {
|
addChunk := func(minT, maxT int64, ref chunks.ChunkRef, chunk chunkenc.Chunk) {
|
||||||
tmpChks = append(tmpChks, chunks.Meta{
|
tmpChks = append(tmpChks, chunks.Meta{
|
||||||
MinTime: minT,
|
MinTime: minT,
|
||||||
MaxTime: maxT,
|
MaxTime: maxT,
|
||||||
Ref: ref,
|
Ref: ref,
|
||||||
Chunk: chunk,
|
Chunk: chunk,
|
||||||
MergeOOO: true,
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -140,34 +140,39 @@ func getOOOSeriesChunks(s *memSeries, mint, maxt int64, lastGarbageCollectedMmap
|
||||||
// those that overlap.
|
// those that overlap.
|
||||||
slices.SortFunc(tmpChks, lessByMinTimeAndMinRef)
|
slices.SortFunc(tmpChks, lessByMinTimeAndMinRef)
|
||||||
|
|
||||||
// Next we want to iterate the sorted collected chunks and only return the
|
// Next we want to iterate the sorted collected chunks and return composites for chunks that overlap with others.
|
||||||
// chunks Meta the first chunk that overlaps with others.
|
|
||||||
// Example chunks of a series: 5:(100, 200) 6:(500, 600) 7:(150, 250) 8:(550, 650)
|
// Example chunks of a series: 5:(100, 200) 6:(500, 600) 7:(150, 250) 8:(550, 650)
|
||||||
// In the example 5 overlaps with 7 and 6 overlaps with 8 so we only want to
|
// In the example 5 overlaps with 7 and 6 overlaps with 8 so we will return
|
||||||
// return chunk Metas for chunk 5 and chunk 6e
|
// [5,7], [6,8].
|
||||||
*chks = append(*chks, tmpChks[0])
|
toBeMerged := tmpChks[0]
|
||||||
maxTime := tmpChks[0].MaxTime // Tracks the maxTime of the previous "to be merged chunk".
|
|
||||||
for _, c := range tmpChks[1:] {
|
for _, c := range tmpChks[1:] {
|
||||||
switch {
|
if c.MinTime > toBeMerged.MaxTime {
|
||||||
case c.MinTime > maxTime:
|
// This chunk doesn't overlap. Send current toBeMerged to output and start a new one.
|
||||||
*chks = append(*chks, c)
|
*chks = append(*chks, toBeMerged)
|
||||||
maxTime = c.MaxTime
|
toBeMerged = c
|
||||||
case c.MaxTime > maxTime:
|
} else {
|
||||||
maxTime = c.MaxTime
|
// Merge this chunk with existing toBeMerged.
|
||||||
(*chks)[len(*chks)-1].MaxTime = c.MaxTime
|
if mm, ok := toBeMerged.Chunk.(*multiMeta); ok {
|
||||||
fallthrough
|
mm.metas = append(mm.metas, c)
|
||||||
default:
|
} else {
|
||||||
// If the head OOO chunk is part of an output chunk, copy the chunk pointer.
|
toBeMerged.Chunk = &multiMeta{metas: []chunks.Meta{toBeMerged, c}}
|
||||||
if c.Chunk != nil {
|
}
|
||||||
(*chks)[len(*chks)-1].Chunk = c.Chunk
|
if toBeMerged.MaxTime < c.MaxTime {
|
||||||
|
toBeMerged.MaxTime = c.MaxTime
|
||||||
}
|
}
|
||||||
(*chks)[len(*chks)-1].MergeOOO = (*chks)[len(*chks)-1].MergeOOO || c.MergeOOO
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
*chks = append(*chks, toBeMerged)
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Fake Chunk object to pass a set of Metas inside Meta.Chunk.
|
||||||
|
type multiMeta struct {
|
||||||
|
chunkenc.Chunk // We don't expect any of the methods to be called.
|
||||||
|
metas []chunks.Meta
|
||||||
|
}
|
||||||
|
|
||||||
// LabelValues needs to be overridden from the headIndexReader implementation
|
// LabelValues needs to be overridden from the headIndexReader implementation
|
||||||
// so we can return labels within either in-order range or ooo range.
|
// so we can return labels within either in-order range or ooo range.
|
||||||
func (oh *HeadAndOOOIndexReader) LabelValues(ctx context.Context, name string, matchers ...*labels.Matcher) ([]string, error) {
|
func (oh *HeadAndOOOIndexReader) LabelValues(ctx context.Context, name string, matchers ...*labels.Matcher) ([]string, error) {
|
||||||
|
@ -182,29 +187,6 @@ func (oh *HeadAndOOOIndexReader) LabelValues(ctx context.Context, name string, m
|
||||||
return labelValuesWithMatchers(ctx, oh, name, matchers...)
|
return labelValuesWithMatchers(ctx, oh, name, matchers...)
|
||||||
}
|
}
|
||||||
|
|
||||||
type chunkMetaAndChunkDiskMapperRef struct {
|
|
||||||
meta chunks.Meta
|
|
||||||
ref chunks.ChunkDiskMapperRef
|
|
||||||
}
|
|
||||||
|
|
||||||
func refLessByMinTimeAndMinRef(a, b chunkMetaAndChunkDiskMapperRef) int {
|
|
||||||
switch {
|
|
||||||
case a.meta.MinTime < b.meta.MinTime:
|
|
||||||
return -1
|
|
||||||
case a.meta.MinTime > b.meta.MinTime:
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
switch {
|
|
||||||
case a.meta.Ref < b.meta.Ref:
|
|
||||||
return -1
|
|
||||||
case a.meta.Ref > b.meta.Ref:
|
|
||||||
return 1
|
|
||||||
default:
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func lessByMinTimeAndMinRef(a, b chunks.Meta) int {
|
func lessByMinTimeAndMinRef(a, b chunks.Meta) int {
|
||||||
switch {
|
switch {
|
||||||
case a.MinTime < b.MinTime:
|
case a.MinTime < b.MinTime:
|
||||||
|
@ -243,36 +225,55 @@ func NewHeadAndOOOChunkReader(head *Head, mint, maxt int64, cr *headChunkReader,
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cr *HeadAndOOOChunkReader) ChunkOrIterable(meta chunks.Meta) (chunkenc.Chunk, chunkenc.Iterable, error) {
|
func (cr *HeadAndOOOChunkReader) ChunkOrIterable(meta chunks.Meta) (chunkenc.Chunk, chunkenc.Iterable, error) {
|
||||||
sid, _, _ := unpackHeadChunkRef(meta.Ref)
|
c, it, _, err := cr.chunkOrIterable(meta, false)
|
||||||
if !meta.MergeOOO {
|
return c, it, err
|
||||||
return cr.cr.ChunkOrIterable(meta)
|
|
||||||
}
|
|
||||||
|
|
||||||
s := cr.head.series.getByID(sid)
|
|
||||||
// This means that the series has been garbage collected.
|
|
||||||
if s == nil {
|
|
||||||
return nil, nil, storage.ErrNotFound
|
|
||||||
}
|
|
||||||
|
|
||||||
s.Lock()
|
|
||||||
if s.ooo == nil { // Must have s.ooo non-nil to call mergedChunks().
|
|
||||||
s.Unlock()
|
|
||||||
return cr.cr.ChunkOrIterable(meta)
|
|
||||||
}
|
|
||||||
mc, err := s.mergedChunks(meta, cr.head.chunkDiskMapper, cr.cr, cr.mint, cr.maxt, cr.maxMmapRef)
|
|
||||||
s.Unlock()
|
|
||||||
|
|
||||||
return nil, mc, err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ChunkOrIterableWithCopy implements ChunkReaderWithCopy. The special Copy
|
// ChunkOrIterableWithCopy implements ChunkReaderWithCopy. The special Copy
|
||||||
// behaviour is only implemented for the in-order head chunk.
|
// behaviour is only implemented for the in-order head chunk.
|
||||||
func (cr *HeadAndOOOChunkReader) ChunkOrIterableWithCopy(meta chunks.Meta) (chunkenc.Chunk, chunkenc.Iterable, int64, error) {
|
func (cr *HeadAndOOOChunkReader) ChunkOrIterableWithCopy(meta chunks.Meta) (chunkenc.Chunk, chunkenc.Iterable, int64, error) {
|
||||||
if !meta.MergeOOO {
|
return cr.chunkOrIterable(meta, true)
|
||||||
return cr.cr.ChunkOrIterableWithCopy(meta)
|
}
|
||||||
|
|
||||||
|
func (cr *HeadAndOOOChunkReader) chunkOrIterable(meta chunks.Meta, copyLastChunk bool) (chunkenc.Chunk, chunkenc.Iterable, int64, error) {
|
||||||
|
sid, cid, isOOO := unpackHeadChunkRef(meta.Ref)
|
||||||
|
s := cr.head.series.getByID(sid)
|
||||||
|
// This means that the series has been garbage collected.
|
||||||
|
if s == nil {
|
||||||
|
return nil, nil, 0, storage.ErrNotFound
|
||||||
}
|
}
|
||||||
chk, iter, err := cr.ChunkOrIterable(meta)
|
var isoState *isolationState
|
||||||
return chk, iter, 0, err
|
if cr.cr != nil {
|
||||||
|
isoState = cr.cr.isoState
|
||||||
|
}
|
||||||
|
|
||||||
|
s.Lock()
|
||||||
|
defer s.Unlock()
|
||||||
|
|
||||||
|
if meta.Chunk == nil {
|
||||||
|
c, maxt, err := cr.head.chunkFromSeries(s, cid, isOOO, meta.MinTime, meta.MaxTime, isoState, copyLastChunk)
|
||||||
|
return c, nil, maxt, err
|
||||||
|
}
|
||||||
|
mm, ok := meta.Chunk.(*multiMeta)
|
||||||
|
if !ok { // Complete chunk was supplied.
|
||||||
|
return meta.Chunk, nil, meta.MaxTime, nil
|
||||||
|
}
|
||||||
|
// We have a composite meta: construct a composite iterable.
|
||||||
|
mc := &mergedOOOChunks{}
|
||||||
|
for _, m := range mm.metas {
|
||||||
|
switch {
|
||||||
|
case m.Chunk != nil:
|
||||||
|
mc.chunkIterables = append(mc.chunkIterables, m.Chunk)
|
||||||
|
default:
|
||||||
|
_, cid, isOOO := unpackHeadChunkRef(m.Ref)
|
||||||
|
iterable, _, err := cr.head.chunkFromSeries(s, cid, isOOO, m.MinTime, m.MaxTime, isoState, copyLastChunk)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, 0, fmt.Errorf("invalid head chunk: %w", err)
|
||||||
|
}
|
||||||
|
mc.chunkIterables = append(mc.chunkIterables, iterable)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil, mc, meta.MaxTime, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cr *HeadAndOOOChunkReader) Close() error {
|
func (cr *HeadAndOOOChunkReader) Close() error {
|
||||||
|
|
|
@ -39,6 +39,11 @@ type chunkInterval struct {
|
||||||
maxt int64
|
maxt int64
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type expChunk struct {
|
||||||
|
c chunkInterval
|
||||||
|
m []chunkInterval
|
||||||
|
}
|
||||||
|
|
||||||
// permutateChunkIntervals returns all possible orders of the given chunkIntervals.
|
// permutateChunkIntervals returns all possible orders of the given chunkIntervals.
|
||||||
func permutateChunkIntervals(in []chunkInterval, out [][]chunkInterval, left, right int) [][]chunkInterval {
|
func permutateChunkIntervals(in []chunkInterval, out [][]chunkInterval, left, right int) [][]chunkInterval {
|
||||||
if left == right {
|
if left == right {
|
||||||
|
@ -65,7 +70,7 @@ func TestOOOHeadIndexReader_Series(t *testing.T) {
|
||||||
queryMinT int64
|
queryMinT int64
|
||||||
queryMaxT int64
|
queryMaxT int64
|
||||||
inputChunkIntervals []chunkInterval
|
inputChunkIntervals []chunkInterval
|
||||||
expChunks []chunkInterval
|
expChunks []expChunk
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
name: "Empty result and no error when head is empty",
|
name: "Empty result and no error when head is empty",
|
||||||
|
@ -107,8 +112,8 @@ func TestOOOHeadIndexReader_Series(t *testing.T) {
|
||||||
// ts 0 100 150 200 250 300 350 400 450 500 550 600 650 700
|
// ts 0 100 150 200 250 300 350 400 450 500 550 600 650 700
|
||||||
// Query Interval [-----------------------------------------------------------]
|
// Query Interval [-----------------------------------------------------------]
|
||||||
// Chunk 0: [---------------------------------------]
|
// Chunk 0: [---------------------------------------]
|
||||||
expChunks: []chunkInterval{
|
expChunks: []expChunk{
|
||||||
{0, 150, 350},
|
{c: chunkInterval{0, 150, 350}},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -121,8 +126,8 @@ func TestOOOHeadIndexReader_Series(t *testing.T) {
|
||||||
// ts 0 100 150 200 250 300 350 400 450 500 550 600 650 700
|
// ts 0 100 150 200 250 300 350 400 450 500 550 600 650 700
|
||||||
// Query Interval: [---------------------------------------]
|
// Query Interval: [---------------------------------------]
|
||||||
// Chunk 0: [-----------------------------------------------------------]
|
// Chunk 0: [-----------------------------------------------------------]
|
||||||
expChunks: []chunkInterval{
|
expChunks: []expChunk{
|
||||||
{0, 100, 400},
|
{c: chunkInterval{0, 100, 400}},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -142,9 +147,9 @@ func TestOOOHeadIndexReader_Series(t *testing.T) {
|
||||||
// Chunk 2: [-------------------]
|
// Chunk 2: [-------------------]
|
||||||
// Chunk 3: [-------------------]
|
// Chunk 3: [-------------------]
|
||||||
// Output Graphically [-----------------------------] [-----------------------------]
|
// Output Graphically [-----------------------------] [-----------------------------]
|
||||||
expChunks: []chunkInterval{
|
expChunks: []expChunk{
|
||||||
{0, 100, 250},
|
{c: chunkInterval{0, 100, 250}, m: []chunkInterval{{0, 100, 200}, {2, 150, 250}}},
|
||||||
{1, 500, 650},
|
{c: chunkInterval{1, 500, 650}, m: []chunkInterval{{1, 500, 600}, {3, 550, 650}}},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -164,8 +169,8 @@ func TestOOOHeadIndexReader_Series(t *testing.T) {
|
||||||
// Chunk 2: [-------------------]
|
// Chunk 2: [-------------------]
|
||||||
// Chunk 3: [------------------]
|
// Chunk 3: [------------------]
|
||||||
// Output Graphically [------------------------------------------------------------------------------]
|
// Output Graphically [------------------------------------------------------------------------------]
|
||||||
expChunks: []chunkInterval{
|
expChunks: []expChunk{
|
||||||
{0, 100, 500},
|
{c: chunkInterval{0, 100, 500}, m: []chunkInterval{{0, 100, 200}, {1, 200, 300}, {2, 300, 400}, {3, 400, 500}}},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -185,11 +190,11 @@ func TestOOOHeadIndexReader_Series(t *testing.T) {
|
||||||
// Chunk 2: [------------------]
|
// Chunk 2: [------------------]
|
||||||
// Chunk 3: [------------------]
|
// Chunk 3: [------------------]
|
||||||
// Output Graphically [------------------][------------------][------------------][------------------]
|
// Output Graphically [------------------][------------------][------------------][------------------]
|
||||||
expChunks: []chunkInterval{
|
expChunks: []expChunk{
|
||||||
{0, 100, 199},
|
{c: chunkInterval{0, 100, 199}},
|
||||||
{1, 200, 299},
|
{c: chunkInterval{1, 200, 299}},
|
||||||
{2, 300, 399},
|
{c: chunkInterval{2, 300, 399}},
|
||||||
{3, 400, 499},
|
{c: chunkInterval{3, 400, 499}},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -209,8 +214,8 @@ func TestOOOHeadIndexReader_Series(t *testing.T) {
|
||||||
// Chunk 2: [------------------]
|
// Chunk 2: [------------------]
|
||||||
// Chunk 3: [------------------]
|
// Chunk 3: [------------------]
|
||||||
// Output Graphically [-----------------------------------------------]
|
// Output Graphically [-----------------------------------------------]
|
||||||
expChunks: []chunkInterval{
|
expChunks: []expChunk{
|
||||||
{0, 100, 350},
|
{c: chunkInterval{0, 100, 350}, m: []chunkInterval{{0, 100, 200}, {1, 150, 300}, {2, 250, 350}}},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -228,8 +233,8 @@ func TestOOOHeadIndexReader_Series(t *testing.T) {
|
||||||
// Chunk 1: [-----------------------------]
|
// Chunk 1: [-----------------------------]
|
||||||
// Chunk 2: [------------------------------]
|
// Chunk 2: [------------------------------]
|
||||||
// Output Graphically [-----------------------------------------------------------------------------------------]
|
// Output Graphically [-----------------------------------------------------------------------------------------]
|
||||||
expChunks: []chunkInterval{
|
expChunks: []expChunk{
|
||||||
{1, 0, 500},
|
{c: chunkInterval{1, 0, 500}, m: []chunkInterval{{1, 0, 200}, {2, 150, 300}, {0, 250, 500}}},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -251,9 +256,9 @@ func TestOOOHeadIndexReader_Series(t *testing.T) {
|
||||||
// Chunk 3: [-------------------]
|
// Chunk 3: [-------------------]
|
||||||
// Chunk 4: [---------------------------------------]
|
// Chunk 4: [---------------------------------------]
|
||||||
// Output Graphically [---------------------------------------] [------------------------------------------------]
|
// Output Graphically [---------------------------------------] [------------------------------------------------]
|
||||||
expChunks: []chunkInterval{
|
expChunks: []expChunk{
|
||||||
{0, 100, 300},
|
{c: chunkInterval{0, 100, 300}, m: []chunkInterval{{0, 100, 300}, {2, 150, 250}}},
|
||||||
{4, 600, 850},
|
{c: chunkInterval{4, 600, 850}, m: []chunkInterval{{4, 600, 800}, {3, 650, 750}, {1, 770, 850}}},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -271,10 +276,10 @@ func TestOOOHeadIndexReader_Series(t *testing.T) {
|
||||||
// Chunk 1: [----------]
|
// Chunk 1: [----------]
|
||||||
// Chunk 2: [--------]
|
// Chunk 2: [--------]
|
||||||
// Output Graphically [-------] [--------] [----------]
|
// Output Graphically [-------] [--------] [----------]
|
||||||
expChunks: []chunkInterval{
|
expChunks: []expChunk{
|
||||||
{0, 100, 150},
|
{c: chunkInterval{0, 100, 150}},
|
||||||
{1, 300, 350},
|
{c: chunkInterval{2, 200, 250}},
|
||||||
{2, 200, 250},
|
{c: chunkInterval{1, 300, 350}},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -305,25 +310,38 @@ func TestOOOHeadIndexReader_Series(t *testing.T) {
|
||||||
s1.ooo = &memSeriesOOOFields{}
|
s1.ooo = &memSeriesOOOFields{}
|
||||||
|
|
||||||
// define our expected chunks, by looking at the expected ChunkIntervals and setting...
|
// define our expected chunks, by looking at the expected ChunkIntervals and setting...
|
||||||
|
// Ref to whatever Ref the chunk has, that we refer to by ID
|
||||||
|
findID := func(id int) chunks.ChunkRef {
|
||||||
|
for ref, c := range intervals {
|
||||||
|
if c.ID == id {
|
||||||
|
return chunks.ChunkRef(chunks.NewHeadChunkRef(chunks.HeadSeriesRef(s1ID), s1.oooHeadChunkID(ref)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
var expChunks []chunks.Meta
|
var expChunks []chunks.Meta
|
||||||
for _, e := range tc.expChunks {
|
for _, e := range tc.expChunks {
|
||||||
meta := chunks.Meta{
|
var chunk chunkenc.Chunk
|
||||||
Chunk: chunkenc.Chunk(nil),
|
if len(e.m) > 0 {
|
||||||
MinTime: e.mint,
|
mm := &multiMeta{}
|
||||||
MaxTime: e.maxt,
|
for _, x := range e.m {
|
||||||
MergeOOO: true, // Only OOO chunks are tested here, so we always request merge from OOO head.
|
meta := chunks.Meta{
|
||||||
}
|
MinTime: x.mint,
|
||||||
|
MaxTime: x.maxt,
|
||||||
// Ref to whatever Ref the chunk has, that we refer to by ID
|
Ref: findID(x.ID),
|
||||||
for ref, c := range intervals {
|
}
|
||||||
if c.ID == e.ID {
|
mm.metas = append(mm.metas, meta)
|
||||||
meta.Ref = chunks.ChunkRef(chunks.NewHeadChunkRef(chunks.HeadSeriesRef(s1ID), s1.oooHeadChunkID(ref)))
|
|
||||||
break
|
|
||||||
}
|
}
|
||||||
|
chunk = mm
|
||||||
|
}
|
||||||
|
meta := chunks.Meta{
|
||||||
|
Chunk: chunk,
|
||||||
|
MinTime: e.c.mint,
|
||||||
|
MaxTime: e.c.maxt,
|
||||||
|
Ref: findID(e.c.ID),
|
||||||
}
|
}
|
||||||
expChunks = append(expChunks, meta)
|
expChunks = append(expChunks, meta)
|
||||||
}
|
}
|
||||||
slices.SortFunc(expChunks, lessByMinTimeAndMinRef) // We always want the chunks to come back sorted by minTime asc.
|
|
||||||
|
|
||||||
if headChunk && len(intervals) > 0 {
|
if headChunk && len(intervals) > 0 {
|
||||||
// Put the last interval in the head chunk
|
// Put the last interval in the head chunk
|
||||||
|
@ -485,7 +503,7 @@ func testOOOHeadChunkReader_Chunk(t *testing.T, scenario sampleTypeScenario) {
|
||||||
cr := NewHeadAndOOOChunkReader(db.head, 0, 1000, nil, nil, 0)
|
cr := NewHeadAndOOOChunkReader(db.head, 0, 1000, nil, nil, 0)
|
||||||
defer cr.Close()
|
defer cr.Close()
|
||||||
c, iterable, err := cr.ChunkOrIterable(chunks.Meta{
|
c, iterable, err := cr.ChunkOrIterable(chunks.Meta{
|
||||||
Ref: 0x1800000, Chunk: chunkenc.Chunk(nil), MinTime: 100, MaxTime: 300, MergeOOO: true,
|
Ref: 0x1800000, Chunk: chunkenc.Chunk(nil), MinTime: 100, MaxTime: 300,
|
||||||
})
|
})
|
||||||
require.Nil(t, iterable)
|
require.Nil(t, iterable)
|
||||||
require.Equal(t, err, fmt.Errorf("not found"))
|
require.Equal(t, err, fmt.Errorf("not found"))
|
||||||
|
@ -498,6 +516,7 @@ func testOOOHeadChunkReader_Chunk(t *testing.T, scenario sampleTypeScenario) {
|
||||||
queryMaxT int64
|
queryMaxT int64
|
||||||
firstInOrderSampleAt int64
|
firstInOrderSampleAt int64
|
||||||
inputSamples []testValue
|
inputSamples []testValue
|
||||||
|
expSingleChunks bool
|
||||||
expChunkError bool
|
expChunkError bool
|
||||||
expChunksSamples []chunks.SampleSlice
|
expChunksSamples []chunks.SampleSlice
|
||||||
}{
|
}{
|
||||||
|
@ -510,7 +529,8 @@ func testOOOHeadChunkReader_Chunk(t *testing.T, scenario sampleTypeScenario) {
|
||||||
{Ts: minutes(30), V: 0},
|
{Ts: minutes(30), V: 0},
|
||||||
{Ts: minutes(40), V: 0},
|
{Ts: minutes(40), V: 0},
|
||||||
},
|
},
|
||||||
expChunkError: false,
|
expChunkError: false,
|
||||||
|
expSingleChunks: true,
|
||||||
// ts (in minutes) 0 10 20 30 40 50 60 70 80 90 100
|
// ts (in minutes) 0 10 20 30 40 50 60 70 80 90 100
|
||||||
// Query Interval [------------------------------------------------------------------------------------------]
|
// Query Interval [------------------------------------------------------------------------------------------]
|
||||||
// Chunk 0: Current Head [--------] (With 2 samples)
|
// Chunk 0: Current Head [--------] (With 2 samples)
|
||||||
|
@ -690,7 +710,8 @@ func testOOOHeadChunkReader_Chunk(t *testing.T, scenario sampleTypeScenario) {
|
||||||
{Ts: minutes(40), V: 3},
|
{Ts: minutes(40), V: 3},
|
||||||
{Ts: minutes(42), V: 3},
|
{Ts: minutes(42), V: 3},
|
||||||
},
|
},
|
||||||
expChunkError: false,
|
expChunkError: false,
|
||||||
|
expSingleChunks: true,
|
||||||
// ts (in minutes) 0 10 20 30 40 50 60 70 80 90 100
|
// ts (in minutes) 0 10 20 30 40 50 60 70 80 90 100
|
||||||
// Query Interval [------------------------------------------------------------------------------------------]
|
// Query Interval [------------------------------------------------------------------------------------------]
|
||||||
// Chunk 0 [-------]
|
// Chunk 0 [-------]
|
||||||
|
@ -845,9 +866,13 @@ func testOOOHeadChunkReader_Chunk(t *testing.T, scenario sampleTypeScenario) {
|
||||||
for i := 0; i < len(chks); i++ {
|
for i := 0; i < len(chks); i++ {
|
||||||
c, iterable, err := cr.ChunkOrIterable(chks[i])
|
c, iterable, err := cr.ChunkOrIterable(chks[i])
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Nil(t, c)
|
var it chunkenc.Iterator
|
||||||
|
if tc.expSingleChunks {
|
||||||
it := iterable.Iterator(nil)
|
it = c.Iterator(nil)
|
||||||
|
} else {
|
||||||
|
require.Nil(t, c)
|
||||||
|
it = iterable.Iterator(nil)
|
||||||
|
}
|
||||||
resultSamples, err := storage.ExpandSamples(it, nil)
|
resultSamples, err := storage.ExpandSamples(it, nil)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
requireEqualSamples(t, s1.String(), tc.expChunksSamples[i], resultSamples, true)
|
requireEqualSamples(t, s1.String(), tc.expChunksSamples[i], resultSamples, true)
|
||||||
|
@ -1030,94 +1055,6 @@ func testOOOHeadChunkReader_Chunk_ConsistentQueryResponseDespiteOfHeadExpanding(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TestSortByMinTimeAndMinRef tests that the sort function for chunk metas does sort
|
|
||||||
// by chunk meta MinTime and in case of same references by the lower reference.
|
|
||||||
func TestSortByMinTimeAndMinRef(t *testing.T) {
|
|
||||||
tests := []struct {
|
|
||||||
name string
|
|
||||||
input []chunkMetaAndChunkDiskMapperRef
|
|
||||||
exp []chunkMetaAndChunkDiskMapperRef
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
name: "chunks are ordered by min time",
|
|
||||||
input: []chunkMetaAndChunkDiskMapperRef{
|
|
||||||
{
|
|
||||||
meta: chunks.Meta{
|
|
||||||
Ref: 0,
|
|
||||||
MinTime: 0,
|
|
||||||
},
|
|
||||||
ref: chunks.ChunkDiskMapperRef(0),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
meta: chunks.Meta{
|
|
||||||
Ref: 1,
|
|
||||||
MinTime: 1,
|
|
||||||
},
|
|
||||||
ref: chunks.ChunkDiskMapperRef(1),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: []chunkMetaAndChunkDiskMapperRef{
|
|
||||||
{
|
|
||||||
meta: chunks.Meta{
|
|
||||||
Ref: 0,
|
|
||||||
MinTime: 0,
|
|
||||||
},
|
|
||||||
ref: chunks.ChunkDiskMapperRef(0),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
meta: chunks.Meta{
|
|
||||||
Ref: 1,
|
|
||||||
MinTime: 1,
|
|
||||||
},
|
|
||||||
ref: chunks.ChunkDiskMapperRef(1),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "if same mintime, lower reference goes first",
|
|
||||||
input: []chunkMetaAndChunkDiskMapperRef{
|
|
||||||
{
|
|
||||||
meta: chunks.Meta{
|
|
||||||
Ref: 10,
|
|
||||||
MinTime: 0,
|
|
||||||
},
|
|
||||||
ref: chunks.ChunkDiskMapperRef(0),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
meta: chunks.Meta{
|
|
||||||
Ref: 5,
|
|
||||||
MinTime: 0,
|
|
||||||
},
|
|
||||||
ref: chunks.ChunkDiskMapperRef(1),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
exp: []chunkMetaAndChunkDiskMapperRef{
|
|
||||||
{
|
|
||||||
meta: chunks.Meta{
|
|
||||||
Ref: 5,
|
|
||||||
MinTime: 0,
|
|
||||||
},
|
|
||||||
ref: chunks.ChunkDiskMapperRef(1),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
meta: chunks.Meta{
|
|
||||||
Ref: 10,
|
|
||||||
MinTime: 0,
|
|
||||||
},
|
|
||||||
ref: chunks.ChunkDiskMapperRef(0),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, tc := range tests {
|
|
||||||
t.Run(fmt.Sprintf("name=%s", tc.name), func(t *testing.T) {
|
|
||||||
slices.SortFunc(tc.input, refLessByMinTimeAndMinRef)
|
|
||||||
require.Equal(t, tc.exp, tc.input)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// TestSortMetaByMinTimeAndMinRef tests that the sort function for chunk metas does sort
|
// TestSortMetaByMinTimeAndMinRef tests that the sort function for chunk metas does sort
|
||||||
// by chunk meta MinTime and in case of same references by the lower reference.
|
// by chunk meta MinTime and in case of same references by the lower reference.
|
||||||
func TestSortMetaByMinTimeAndMinRef(t *testing.T) {
|
func TestSortMetaByMinTimeAndMinRef(t *testing.T) {
|
||||||
|
|
|
@ -603,7 +603,7 @@ func (w *Watcher) readSegment(r *LiveReader, segmentNum int, tail bool) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
case record.Metadata:
|
case record.Metadata:
|
||||||
if !w.sendMetadata || !tail {
|
if !w.sendMetadata {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
meta, err := dec.Metadata(rec, metadata[:0])
|
meta, err := dec.Metadata(rec, metadata[:0])
|
||||||
|
|
|
@ -13,13 +13,23 @@
|
||||||
|
|
||||||
package almost
|
package almost
|
||||||
|
|
||||||
import "math"
|
import (
|
||||||
|
"math"
|
||||||
|
|
||||||
|
"github.com/prometheus/prometheus/model/value"
|
||||||
|
)
|
||||||
|
|
||||||
var minNormal = math.Float64frombits(0x0010000000000000) // The smallest positive normal value of type float64.
|
var minNormal = math.Float64frombits(0x0010000000000000) // The smallest positive normal value of type float64.
|
||||||
|
|
||||||
// Equal returns true if a and b differ by less than their sum
|
// Equal returns true if a and b differ by less than their sum
|
||||||
// multiplied by epsilon.
|
// multiplied by epsilon.
|
||||||
func Equal(a, b, epsilon float64) bool {
|
func Equal(a, b, epsilon float64) bool {
|
||||||
|
// StaleNaN is a special value that is used as staleness maker, so
|
||||||
|
// the two values are equal when both are exactly equals to stale NaN.
|
||||||
|
if value.IsStaleNaN(a) || value.IsStaleNaN(b) {
|
||||||
|
return value.IsStaleNaN(a) && value.IsStaleNaN(b)
|
||||||
|
}
|
||||||
|
|
||||||
// NaN has no equality but for testing we still want to know whether both values
|
// NaN has no equality but for testing we still want to know whether both values
|
||||||
// are NaN.
|
// are NaN.
|
||||||
if math.IsNaN(a) && math.IsNaN(b) {
|
if math.IsNaN(a) && math.IsNaN(b) {
|
||||||
|
|
|
@ -1074,6 +1074,9 @@ func setupRemote(s storage.Storage) *httptest.Server {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
w.Header().Set("Content-Type", "application/x-protobuf")
|
||||||
|
w.Header().Set("Content-Encoding", "snappy")
|
||||||
|
|
||||||
if err := remote.EncodeReadResponse(&resp, w); err != nil {
|
if err := remote.EncodeReadResponse(&resp, w); err != nil {
|
||||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "@prometheus-io/codemirror-promql",
|
"name": "@prometheus-io/codemirror-promql",
|
||||||
"version": "0.54.0-rc.1",
|
"version": "0.54.1",
|
||||||
"description": "a CodeMirror mode for the PromQL language",
|
"description": "a CodeMirror mode for the PromQL language",
|
||||||
"types": "dist/esm/index.d.ts",
|
"types": "dist/esm/index.d.ts",
|
||||||
"module": "dist/esm/index.js",
|
"module": "dist/esm/index.js",
|
||||||
|
@ -29,7 +29,7 @@
|
||||||
},
|
},
|
||||||
"homepage": "https://github.com/prometheus/prometheus/blob/main/web/ui/module/codemirror-promql/README.md",
|
"homepage": "https://github.com/prometheus/prometheus/blob/main/web/ui/module/codemirror-promql/README.md",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@prometheus-io/lezer-promql": "0.54.0-rc.1",
|
"@prometheus-io/lezer-promql": "0.54.1",
|
||||||
"lru-cache": "^7.18.3"
|
"lru-cache": "^7.18.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "@prometheus-io/lezer-promql",
|
"name": "@prometheus-io/lezer-promql",
|
||||||
"version": "0.54.0-rc.1",
|
"version": "0.54.1",
|
||||||
"description": "lezer-based PromQL grammar",
|
"description": "lezer-based PromQL grammar",
|
||||||
"main": "dist/index.cjs",
|
"main": "dist/index.cjs",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|
14
web/ui/package-lock.json
generated
14
web/ui/package-lock.json
generated
|
@ -1,12 +1,12 @@
|
||||||
{
|
{
|
||||||
"name": "prometheus-io",
|
"name": "prometheus-io",
|
||||||
"version": "0.54.0-rc.1",
|
"version": "0.54.1",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "prometheus-io",
|
"name": "prometheus-io",
|
||||||
"version": "0.54.0-rc.1",
|
"version": "0.54.1",
|
||||||
"workspaces": [
|
"workspaces": [
|
||||||
"react-app",
|
"react-app",
|
||||||
"module/*"
|
"module/*"
|
||||||
|
@ -30,10 +30,10 @@
|
||||||
},
|
},
|
||||||
"module/codemirror-promql": {
|
"module/codemirror-promql": {
|
||||||
"name": "@prometheus-io/codemirror-promql",
|
"name": "@prometheus-io/codemirror-promql",
|
||||||
"version": "0.54.0-rc.1",
|
"version": "0.54.1",
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@prometheus-io/lezer-promql": "0.54.0-rc.1",
|
"@prometheus-io/lezer-promql": "0.54.1",
|
||||||
"lru-cache": "^7.18.3"
|
"lru-cache": "^7.18.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
@ -69,7 +69,7 @@
|
||||||
},
|
},
|
||||||
"module/lezer-promql": {
|
"module/lezer-promql": {
|
||||||
"name": "@prometheus-io/lezer-promql",
|
"name": "@prometheus-io/lezer-promql",
|
||||||
"version": "0.54.0-rc.1",
|
"version": "0.54.1",
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@lezer/generator": "^1.7.1",
|
"@lezer/generator": "^1.7.1",
|
||||||
|
@ -19352,7 +19352,7 @@
|
||||||
},
|
},
|
||||||
"react-app": {
|
"react-app": {
|
||||||
"name": "@prometheus-io/app",
|
"name": "@prometheus-io/app",
|
||||||
"version": "0.54.0-rc.1",
|
"version": "0.54.1",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@codemirror/autocomplete": "^6.17.0",
|
"@codemirror/autocomplete": "^6.17.0",
|
||||||
"@codemirror/commands": "^6.6.0",
|
"@codemirror/commands": "^6.6.0",
|
||||||
|
@ -19370,7 +19370,7 @@
|
||||||
"@lezer/lr": "^1.4.2",
|
"@lezer/lr": "^1.4.2",
|
||||||
"@nexucis/fuzzy": "^0.4.1",
|
"@nexucis/fuzzy": "^0.4.1",
|
||||||
"@nexucis/kvsearch": "^0.8.1",
|
"@nexucis/kvsearch": "^0.8.1",
|
||||||
"@prometheus-io/codemirror-promql": "0.54.0-rc.1",
|
"@prometheus-io/codemirror-promql": "0.54.1",
|
||||||
"bootstrap": "^4.6.2",
|
"bootstrap": "^4.6.2",
|
||||||
"css.escape": "^1.5.1",
|
"css.escape": "^1.5.1",
|
||||||
"downshift": "^9.0.6",
|
"downshift": "^9.0.6",
|
||||||
|
|
|
@ -28,5 +28,5 @@
|
||||||
"ts-jest": "^29.2.2",
|
"ts-jest": "^29.2.2",
|
||||||
"typescript": "^4.9.5"
|
"typescript": "^4.9.5"
|
||||||
},
|
},
|
||||||
"version": "0.54.0-rc.1"
|
"version": "0.54.1"
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "@prometheus-io/app",
|
"name": "@prometheus-io/app",
|
||||||
"version": "0.54.0-rc.1",
|
"version": "0.54.1",
|
||||||
"private": true,
|
"private": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@codemirror/autocomplete": "^6.17.0",
|
"@codemirror/autocomplete": "^6.17.0",
|
||||||
|
@ -19,7 +19,7 @@
|
||||||
"@lezer/lr": "^1.4.2",
|
"@lezer/lr": "^1.4.2",
|
||||||
"@nexucis/fuzzy": "^0.4.1",
|
"@nexucis/fuzzy": "^0.4.1",
|
||||||
"@nexucis/kvsearch": "^0.8.1",
|
"@nexucis/kvsearch": "^0.8.1",
|
||||||
"@prometheus-io/codemirror-promql": "0.54.0-rc.1",
|
"@prometheus-io/codemirror-promql": "0.54.1",
|
||||||
"bootstrap": "^4.6.2",
|
"bootstrap": "^4.6.2",
|
||||||
"css.escape": "^1.5.1",
|
"css.escape": "^1.5.1",
|
||||||
"downshift": "^9.0.6",
|
"downshift": "^9.0.6",
|
||||||
|
|
Loading…
Reference in a new issue