mirror of
				https://github.com/prometheus/prometheus.git
				synced 2025-10-26 05:51:01 +01:00 
			
		
		
		
	Histogram.Validate and FloatHistogram.Validate now return error on unsupported schemas. Scrape and remote-write handler reduces the schema to the maximum allowed if it is above the maximum, but below theoretical maximum of 52. For scrape the maximum is a configuration option, for remote-write it is 8. Note: OTLP endpont already does the reduction, without checking that it is below 52 as the spec does not specify a maximum. Signed-off-by: György Krajcsovits <gyorgy.krajcsovits@grafana.com>
		
			
				
	
	
		
			553 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			553 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2013 The Prometheus Authors
 | |
| // Licensed under the Apache License, Version 2.0 (the "License");
 | |
| // you may not use this file except in compliance with the License.
 | |
| // You may obtain a copy of the License at
 | |
| //
 | |
| // http://www.apache.org/licenses/LICENSE-2.0
 | |
| //
 | |
| // Unless required by applicable law or agreed to in writing, software
 | |
| // distributed under the License is distributed on an "AS IS" BASIS,
 | |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| // See the License for the specific language governing permissions and
 | |
| // limitations under the License.
 | |
| 
 | |
| package scrape
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"hash/fnv"
 | |
| 	"net/url"
 | |
| 	"strings"
 | |
| 	"sync"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/prometheus/common/model"
 | |
| 
 | |
| 	"github.com/prometheus/prometheus/config"
 | |
| 	"github.com/prometheus/prometheus/discovery/targetgroup"
 | |
| 	"github.com/prometheus/prometheus/model/histogram"
 | |
| 	"github.com/prometheus/prometheus/model/labels"
 | |
| 	"github.com/prometheus/prometheus/model/relabel"
 | |
| 	"github.com/prometheus/prometheus/model/value"
 | |
| 	"github.com/prometheus/prometheus/storage"
 | |
| )
 | |
| 
 | |
| // TargetHealth describes the health state of a target.
 | |
| type TargetHealth string
 | |
| 
 | |
| // The possible health states of a target based on the last performed scrape.
 | |
| const (
 | |
| 	HealthUnknown TargetHealth = "unknown"
 | |
| 	HealthGood    TargetHealth = "up"
 | |
| 	HealthBad     TargetHealth = "down"
 | |
| )
 | |
| 
 | |
| // Target refers to a singular HTTP or HTTPS endpoint.
 | |
| type Target struct {
 | |
| 	// Any labels that are added to this target and its metrics.
 | |
| 	labels labels.Labels
 | |
| 	// ScrapeConfig used to create this target.
 | |
| 	scrapeConfig *config.ScrapeConfig
 | |
| 	// Target and TargetGroup labels used to create this target.
 | |
| 	tLabels, tgLabels model.LabelSet
 | |
| 
 | |
| 	mtx                sync.RWMutex
 | |
| 	lastError          error
 | |
| 	lastScrape         time.Time
 | |
| 	lastScrapeDuration time.Duration
 | |
| 	health             TargetHealth
 | |
| 	metadata           MetricMetadataStore
 | |
| }
 | |
| 
 | |
| // NewTarget creates a reasonably configured target for querying.
 | |
| func NewTarget(labels labels.Labels, scrapeConfig *config.ScrapeConfig, tLabels, tgLabels model.LabelSet) *Target {
 | |
| 	return &Target{
 | |
| 		labels:       labels,
 | |
| 		tLabels:      tLabels,
 | |
| 		tgLabels:     tgLabels,
 | |
| 		scrapeConfig: scrapeConfig,
 | |
| 		health:       HealthUnknown,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (t *Target) String() string {
 | |
| 	return t.URL().String()
 | |
| }
 | |
| 
 | |
| // MetricMetadataStore represents a storage for metadata.
 | |
| type MetricMetadataStore interface {
 | |
| 	ListMetadata() []MetricMetadata
 | |
| 	GetMetadata(mfName string) (MetricMetadata, bool)
 | |
| 	SizeMetadata() int
 | |
| 	LengthMetadata() int
 | |
| }
 | |
| 
 | |
| // MetricMetadata is a piece of metadata for a metric family.
 | |
| type MetricMetadata struct {
 | |
| 	MetricFamily string
 | |
| 	Type         model.MetricType
 | |
| 	Help         string
 | |
| 	Unit         string
 | |
| }
 | |
| 
 | |
| func (t *Target) ListMetadata() []MetricMetadata {
 | |
| 	t.mtx.RLock()
 | |
| 	defer t.mtx.RUnlock()
 | |
| 
 | |
| 	if t.metadata == nil {
 | |
| 		return nil
 | |
| 	}
 | |
| 	return t.metadata.ListMetadata()
 | |
| }
 | |
| 
 | |
| func (t *Target) SizeMetadata() int {
 | |
| 	t.mtx.RLock()
 | |
| 	defer t.mtx.RUnlock()
 | |
| 
 | |
| 	if t.metadata == nil {
 | |
| 		return 0
 | |
| 	}
 | |
| 
 | |
| 	return t.metadata.SizeMetadata()
 | |
| }
 | |
| 
 | |
| func (t *Target) LengthMetadata() int {
 | |
| 	t.mtx.RLock()
 | |
| 	defer t.mtx.RUnlock()
 | |
| 
 | |
| 	if t.metadata == nil {
 | |
| 		return 0
 | |
| 	}
 | |
| 
 | |
| 	return t.metadata.LengthMetadata()
 | |
| }
 | |
| 
 | |
| // GetMetadata returns type and help metadata for the given metric.
 | |
| func (t *Target) GetMetadata(mfName string) (MetricMetadata, bool) {
 | |
| 	t.mtx.RLock()
 | |
| 	defer t.mtx.RUnlock()
 | |
| 
 | |
| 	if t.metadata == nil {
 | |
| 		return MetricMetadata{}, false
 | |
| 	}
 | |
| 	return t.metadata.GetMetadata(mfName)
 | |
| }
 | |
| 
 | |
| func (t *Target) SetMetadataStore(s MetricMetadataStore) {
 | |
| 	t.mtx.Lock()
 | |
| 	defer t.mtx.Unlock()
 | |
| 	t.metadata = s
 | |
| }
 | |
| 
 | |
| // hash returns an identifying hash for the target.
 | |
| func (t *Target) hash() uint64 {
 | |
| 	h := fnv.New64a()
 | |
| 
 | |
| 	fmt.Fprintf(h, "%016d", t.labels.Hash())
 | |
| 	h.Write([]byte(t.URL().String()))
 | |
| 
 | |
| 	return h.Sum64()
 | |
| }
 | |
| 
 | |
| // offset returns the time until the next scrape cycle for the target.
 | |
| // It includes the global server offsetSeed for scrapes from multiple Prometheus to try to be at different times.
 | |
| func (t *Target) offset(interval time.Duration, offsetSeed uint64) time.Duration {
 | |
| 	now := time.Now().UnixNano()
 | |
| 
 | |
| 	// Base is a pinned to absolute time, no matter how often offset is called.
 | |
| 	var (
 | |
| 		base   = int64(interval) - now%int64(interval)
 | |
| 		offset = (t.hash() ^ offsetSeed) % uint64(interval)
 | |
| 		next   = base + int64(offset)
 | |
| 	)
 | |
| 
 | |
| 	if next > int64(interval) {
 | |
| 		next -= int64(interval)
 | |
| 	}
 | |
| 	return time.Duration(next)
 | |
| }
 | |
| 
 | |
| // Labels returns a copy of the set of all public labels of the target.
 | |
| func (t *Target) Labels(b *labels.Builder) labels.Labels {
 | |
| 	b.Reset(labels.EmptyLabels())
 | |
| 	t.labels.Range(func(l labels.Label) {
 | |
| 		if !strings.HasPrefix(l.Name, model.ReservedLabelPrefix) {
 | |
| 			b.Set(l.Name, l.Value)
 | |
| 		}
 | |
| 	})
 | |
| 	return b.Labels()
 | |
| }
 | |
| 
 | |
| // LabelsRange calls f on each public label of the target.
 | |
| func (t *Target) LabelsRange(f func(l labels.Label)) {
 | |
| 	t.labels.Range(func(l labels.Label) {
 | |
| 		if !strings.HasPrefix(l.Name, model.ReservedLabelPrefix) {
 | |
| 			f(l)
 | |
| 		}
 | |
| 	})
 | |
| }
 | |
| 
 | |
| // DiscoveredLabels returns a copy of the target's labels before any processing.
 | |
| func (t *Target) DiscoveredLabels(lb *labels.Builder) labels.Labels {
 | |
| 	t.mtx.Lock()
 | |
| 	cfg, tLabels, tgLabels := t.scrapeConfig, t.tLabels, t.tgLabels
 | |
| 	t.mtx.Unlock()
 | |
| 	PopulateDiscoveredLabels(lb, cfg, tLabels, tgLabels)
 | |
| 	return lb.Labels()
 | |
| }
 | |
| 
 | |
| // SetScrapeConfig sets new ScrapeConfig.
 | |
| func (t *Target) SetScrapeConfig(scrapeConfig *config.ScrapeConfig, tLabels, tgLabels model.LabelSet) {
 | |
| 	t.mtx.Lock()
 | |
| 	defer t.mtx.Unlock()
 | |
| 	t.scrapeConfig = scrapeConfig
 | |
| 	t.tLabels = tLabels
 | |
| 	t.tgLabels = tgLabels
 | |
| }
 | |
| 
 | |
| // URL returns a copy of the target's URL.
 | |
| func (t *Target) URL() *url.URL {
 | |
| 	t.mtx.Lock()
 | |
| 	configParams := t.scrapeConfig.Params
 | |
| 	t.mtx.Unlock()
 | |
| 	params := url.Values{}
 | |
| 
 | |
| 	for k, v := range configParams {
 | |
| 		params[k] = make([]string, len(v))
 | |
| 		copy(params[k], v)
 | |
| 	}
 | |
| 	t.labels.Range(func(l labels.Label) {
 | |
| 		if !strings.HasPrefix(l.Name, model.ParamLabelPrefix) {
 | |
| 			return
 | |
| 		}
 | |
| 		ks := l.Name[len(model.ParamLabelPrefix):]
 | |
| 
 | |
| 		if len(params[ks]) > 0 {
 | |
| 			params[ks][0] = l.Value
 | |
| 		} else {
 | |
| 			params[ks] = []string{l.Value}
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	return &url.URL{
 | |
| 		Scheme:   t.labels.Get(model.SchemeLabel),
 | |
| 		Host:     t.labels.Get(model.AddressLabel),
 | |
| 		Path:     t.labels.Get(model.MetricsPathLabel),
 | |
| 		RawQuery: params.Encode(),
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Report sets target data about the last scrape.
 | |
| func (t *Target) Report(start time.Time, dur time.Duration, err error) {
 | |
| 	t.mtx.Lock()
 | |
| 	defer t.mtx.Unlock()
 | |
| 
 | |
| 	if err == nil {
 | |
| 		t.health = HealthGood
 | |
| 	} else {
 | |
| 		t.health = HealthBad
 | |
| 	}
 | |
| 
 | |
| 	t.lastError = err
 | |
| 	t.lastScrape = start
 | |
| 	t.lastScrapeDuration = dur
 | |
| }
 | |
| 
 | |
| // LastError returns the error encountered during the last scrape.
 | |
| func (t *Target) LastError() error {
 | |
| 	t.mtx.RLock()
 | |
| 	defer t.mtx.RUnlock()
 | |
| 
 | |
| 	return t.lastError
 | |
| }
 | |
| 
 | |
| // LastScrape returns the time of the last scrape.
 | |
| func (t *Target) LastScrape() time.Time {
 | |
| 	t.mtx.RLock()
 | |
| 	defer t.mtx.RUnlock()
 | |
| 
 | |
| 	return t.lastScrape
 | |
| }
 | |
| 
 | |
| // LastScrapeDuration returns how long the last scrape of the target took.
 | |
| func (t *Target) LastScrapeDuration() time.Duration {
 | |
| 	t.mtx.RLock()
 | |
| 	defer t.mtx.RUnlock()
 | |
| 
 | |
| 	return t.lastScrapeDuration
 | |
| }
 | |
| 
 | |
| // Health returns the last known health state of the target.
 | |
| func (t *Target) Health() TargetHealth {
 | |
| 	t.mtx.RLock()
 | |
| 	defer t.mtx.RUnlock()
 | |
| 
 | |
| 	return t.health
 | |
| }
 | |
| 
 | |
| // intervalAndTimeout returns the interval and timeout derived from
 | |
| // the targets labels.
 | |
| func (t *Target) intervalAndTimeout(defaultInterval, defaultDuration time.Duration) (time.Duration, time.Duration, error) {
 | |
| 	t.mtx.RLock()
 | |
| 	defer t.mtx.RUnlock()
 | |
| 
 | |
| 	intervalLabel := t.labels.Get(model.ScrapeIntervalLabel)
 | |
| 	interval, err := model.ParseDuration(intervalLabel)
 | |
| 	if err != nil {
 | |
| 		return defaultInterval, defaultDuration, fmt.Errorf("error parsing interval label %q: %w", intervalLabel, err)
 | |
| 	}
 | |
| 	timeoutLabel := t.labels.Get(model.ScrapeTimeoutLabel)
 | |
| 	timeout, err := model.ParseDuration(timeoutLabel)
 | |
| 	if err != nil {
 | |
| 		return defaultInterval, defaultDuration, fmt.Errorf("error parsing timeout label %q: %w", timeoutLabel, err)
 | |
| 	}
 | |
| 
 | |
| 	return time.Duration(interval), time.Duration(timeout), nil
 | |
| }
 | |
| 
 | |
| // GetValue gets a label value from the entire label set.
 | |
| func (t *Target) GetValue(name string) string {
 | |
| 	return t.labels.Get(name)
 | |
| }
 | |
| 
 | |
| // Targets is a sortable list of targets.
 | |
| type Targets []*Target
 | |
| 
 | |
| func (ts Targets) Len() int           { return len(ts) }
 | |
| func (ts Targets) Less(i, j int) bool { return ts[i].URL().String() < ts[j].URL().String() }
 | |
| func (ts Targets) Swap(i, j int)      { ts[i], ts[j] = ts[j], ts[i] }
 | |
| 
 | |
| var (
 | |
| 	errSampleLimit = errors.New("sample limit exceeded")
 | |
| 	errBucketLimit = errors.New("histogram bucket limit exceeded")
 | |
| )
 | |
| 
 | |
| // limitAppender limits the number of total appended samples in a batch.
 | |
| type limitAppender struct {
 | |
| 	storage.Appender
 | |
| 
 | |
| 	limit int
 | |
| 	i     int
 | |
| }
 | |
| 
 | |
| func (app *limitAppender) Append(ref storage.SeriesRef, lset labels.Labels, t int64, v float64) (storage.SeriesRef, error) {
 | |
| 	// Bypass sample_limit checks only if we have a staleness marker for a known series (ref value is non-zero).
 | |
| 	// This ensures that if a series is already in TSDB then we always write the marker.
 | |
| 	if ref == 0 || !value.IsStaleNaN(v) {
 | |
| 		app.i++
 | |
| 		if app.i > app.limit {
 | |
| 			return 0, errSampleLimit
 | |
| 		}
 | |
| 	}
 | |
| 	ref, err := app.Appender.Append(ref, lset, t, v)
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 	return ref, nil
 | |
| }
 | |
| 
 | |
| type timeLimitAppender struct {
 | |
| 	storage.Appender
 | |
| 
 | |
| 	maxTime int64
 | |
| }
 | |
| 
 | |
| func (app *timeLimitAppender) Append(ref storage.SeriesRef, lset labels.Labels, t int64, v float64) (storage.SeriesRef, error) {
 | |
| 	if t > app.maxTime {
 | |
| 		return 0, storage.ErrOutOfBounds
 | |
| 	}
 | |
| 
 | |
| 	ref, err := app.Appender.Append(ref, lset, t, v)
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 	return ref, nil
 | |
| }
 | |
| 
 | |
| // bucketLimitAppender limits the number of total appended samples in a batch.
 | |
| type bucketLimitAppender struct {
 | |
| 	storage.Appender
 | |
| 
 | |
| 	limit int
 | |
| }
 | |
| 
 | |
| func (app *bucketLimitAppender) AppendHistogram(ref storage.SeriesRef, lset labels.Labels, t int64, h *histogram.Histogram, fh *histogram.FloatHistogram) (storage.SeriesRef, error) {
 | |
| 	if h != nil {
 | |
| 		// Return with an early error if the histogram has too many buckets and the
 | |
| 		// schema is not exponential, in which case we can't reduce the resolution.
 | |
| 		if len(h.PositiveBuckets)+len(h.NegativeBuckets) > app.limit && !histogram.IsExponentialSchema(h.Schema) {
 | |
| 			return 0, errBucketLimit
 | |
| 		}
 | |
| 		for len(h.PositiveBuckets)+len(h.NegativeBuckets) > app.limit {
 | |
| 			if h.Schema <= histogram.ExponentialSchemaMin {
 | |
| 				return 0, errBucketLimit
 | |
| 			}
 | |
| 			h = h.ReduceResolution(h.Schema - 1)
 | |
| 		}
 | |
| 	}
 | |
| 	if fh != nil {
 | |
| 		// Return with an early error if the histogram has too many buckets and the
 | |
| 		// schema is not exponential, in which case we can't reduce the resolution.
 | |
| 		if len(fh.PositiveBuckets)+len(fh.NegativeBuckets) > app.limit && !histogram.IsExponentialSchema(fh.Schema) {
 | |
| 			return 0, errBucketLimit
 | |
| 		}
 | |
| 		for len(fh.PositiveBuckets)+len(fh.NegativeBuckets) > app.limit {
 | |
| 			if fh.Schema <= histogram.ExponentialSchemaMin {
 | |
| 				return 0, errBucketLimit
 | |
| 			}
 | |
| 			fh = fh.ReduceResolution(fh.Schema - 1)
 | |
| 		}
 | |
| 	}
 | |
| 	ref, err := app.Appender.AppendHistogram(ref, lset, t, h, fh)
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 	return ref, nil
 | |
| }
 | |
| 
 | |
| type maxSchemaAppender struct {
 | |
| 	storage.Appender
 | |
| 
 | |
| 	maxSchema int32
 | |
| }
 | |
| 
 | |
| func (app *maxSchemaAppender) AppendHistogram(ref storage.SeriesRef, lset labels.Labels, t int64, h *histogram.Histogram, fh *histogram.FloatHistogram) (storage.SeriesRef, error) {
 | |
| 	if h != nil {
 | |
| 		if histogram.IsExponentialSchemaReserved(h.Schema) && h.Schema > app.maxSchema {
 | |
| 			h = h.ReduceResolution(app.maxSchema)
 | |
| 		}
 | |
| 	}
 | |
| 	if fh != nil {
 | |
| 		if histogram.IsExponentialSchemaReserved(fh.Schema) && fh.Schema > app.maxSchema {
 | |
| 			fh = fh.ReduceResolution(app.maxSchema)
 | |
| 		}
 | |
| 	}
 | |
| 	ref, err := app.Appender.AppendHistogram(ref, lset, t, h, fh)
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 	return ref, nil
 | |
| }
 | |
| 
 | |
| // PopulateDiscoveredLabels sets base labels on lb from target and group labels and scrape configuration, before relabeling.
 | |
| func PopulateDiscoveredLabels(lb *labels.Builder, cfg *config.ScrapeConfig, tLabels, tgLabels model.LabelSet) {
 | |
| 	lb.Reset(labels.EmptyLabels())
 | |
| 
 | |
| 	for ln, lv := range tLabels {
 | |
| 		lb.Set(string(ln), string(lv))
 | |
| 	}
 | |
| 	for ln, lv := range tgLabels {
 | |
| 		if _, ok := tLabels[ln]; !ok {
 | |
| 			lb.Set(string(ln), string(lv))
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Copy labels into the labelset for the target if they are not set already.
 | |
| 	scrapeLabels := []labels.Label{
 | |
| 		{Name: model.JobLabel, Value: cfg.JobName},
 | |
| 		{Name: model.ScrapeIntervalLabel, Value: cfg.ScrapeInterval.String()},
 | |
| 		{Name: model.ScrapeTimeoutLabel, Value: cfg.ScrapeTimeout.String()},
 | |
| 		{Name: model.MetricsPathLabel, Value: cfg.MetricsPath},
 | |
| 		{Name: model.SchemeLabel, Value: cfg.Scheme},
 | |
| 	}
 | |
| 
 | |
| 	for _, l := range scrapeLabels {
 | |
| 		if lb.Get(l.Name) == "" {
 | |
| 			lb.Set(l.Name, l.Value)
 | |
| 		}
 | |
| 	}
 | |
| 	// Encode scrape query parameters as labels.
 | |
| 	for k, v := range cfg.Params {
 | |
| 		if name := model.ParamLabelPrefix + k; len(v) > 0 && lb.Get(name) == "" {
 | |
| 			lb.Set(name, v[0])
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // PopulateLabels builds labels from target and group labels and scrape configuration,
 | |
| // performs defined relabeling, checks validity, and adds Prometheus standard labels such as 'instance'.
 | |
| // A return of empty labels and nil error means the target was dropped by relabeling.
 | |
| func PopulateLabels(lb *labels.Builder, cfg *config.ScrapeConfig, tLabels, tgLabels model.LabelSet) (res labels.Labels, err error) {
 | |
| 	PopulateDiscoveredLabels(lb, cfg, tLabels, tgLabels)
 | |
| 	keep := relabel.ProcessBuilder(lb, cfg.RelabelConfigs...)
 | |
| 
 | |
| 	// Check if the target was dropped.
 | |
| 	if !keep {
 | |
| 		return labels.EmptyLabels(), nil
 | |
| 	}
 | |
| 	if v := lb.Get(model.AddressLabel); v == "" {
 | |
| 		return labels.EmptyLabels(), errors.New("no address")
 | |
| 	}
 | |
| 
 | |
| 	addr := lb.Get(model.AddressLabel)
 | |
| 
 | |
| 	if err := config.CheckTargetAddress(model.LabelValue(addr)); err != nil {
 | |
| 		return labels.EmptyLabels(), err
 | |
| 	}
 | |
| 
 | |
| 	interval := lb.Get(model.ScrapeIntervalLabel)
 | |
| 	intervalDuration, err := model.ParseDuration(interval)
 | |
| 	if err != nil {
 | |
| 		return labels.EmptyLabels(), fmt.Errorf("error parsing scrape interval: %w", err)
 | |
| 	}
 | |
| 	if time.Duration(intervalDuration) == 0 {
 | |
| 		return labels.EmptyLabels(), errors.New("scrape interval cannot be 0")
 | |
| 	}
 | |
| 
 | |
| 	timeout := lb.Get(model.ScrapeTimeoutLabel)
 | |
| 	timeoutDuration, err := model.ParseDuration(timeout)
 | |
| 	if err != nil {
 | |
| 		return labels.EmptyLabels(), fmt.Errorf("error parsing scrape timeout: %w", err)
 | |
| 	}
 | |
| 	if time.Duration(timeoutDuration) == 0 {
 | |
| 		return labels.EmptyLabels(), errors.New("scrape timeout cannot be 0")
 | |
| 	}
 | |
| 
 | |
| 	if timeoutDuration > intervalDuration {
 | |
| 		return labels.EmptyLabels(), fmt.Errorf("scrape timeout cannot be greater than scrape interval (%q > %q)", timeout, interval)
 | |
| 	}
 | |
| 
 | |
| 	// Meta labels are deleted after relabelling. Other internal labels propagate to
 | |
| 	// the target which decides whether they will be part of their label set.
 | |
| 	lb.Range(func(l labels.Label) {
 | |
| 		if strings.HasPrefix(l.Name, model.MetaLabelPrefix) {
 | |
| 			lb.Del(l.Name)
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	// Default the instance label to the target address.
 | |
| 	if v := lb.Get(model.InstanceLabel); v == "" {
 | |
| 		lb.Set(model.InstanceLabel, addr)
 | |
| 	}
 | |
| 
 | |
| 	res = lb.Labels()
 | |
| 	err = res.Validate(func(l labels.Label) error {
 | |
| 		// Check label values are valid, drop the target if not.
 | |
| 		if !model.LabelValue(l.Value).IsValid() {
 | |
| 			return fmt.Errorf("invalid label value for %q: %q", l.Name, l.Value)
 | |
| 		}
 | |
| 		return nil
 | |
| 	})
 | |
| 	if err != nil {
 | |
| 		return labels.EmptyLabels(), err
 | |
| 	}
 | |
| 	return res, nil
 | |
| }
 | |
| 
 | |
| // TargetsFromGroup builds targets based on the given TargetGroup and config.
 | |
| func TargetsFromGroup(tg *targetgroup.Group, cfg *config.ScrapeConfig, targets []*Target, lb *labels.Builder) ([]*Target, []error) {
 | |
| 	targets = targets[:0]
 | |
| 	failures := []error{}
 | |
| 
 | |
| 	for i, tLabels := range tg.Targets {
 | |
| 		lset, err := PopulateLabels(lb, cfg, tLabels, tg.Labels)
 | |
| 		if err != nil {
 | |
| 			failures = append(failures, fmt.Errorf("instance %d in group %s: %w", i, tg, err))
 | |
| 		} else {
 | |
| 			targets = append(targets, NewTarget(lset, cfg, tLabels, tg.Labels))
 | |
| 		}
 | |
| 	}
 | |
| 	return targets, failures
 | |
| }
 |