From 6efc6fccc1b14eb7c0a83522d77187bb95a5de3d Mon Sep 17 00:00:00 2001 From: Mischa Thompson Date: Tue, 31 Oct 2023 13:27:45 -0700 Subject: [PATCH 1/2] Add appendExternalLabels function and test --- .../agent_management_remote_config_test.go | 44 +++++++++++++++++++ pkg/config/agentmanagement_remote_config.go | 22 +++++++++- 2 files changed, 64 insertions(+), 2 deletions(-) diff --git a/pkg/config/agent_management_remote_config_test.go b/pkg/config/agent_management_remote_config_test.go index 64444f2f169a..d9bb7ed2b7a8 100644 --- a/pkg/config/agent_management_remote_config_test.go +++ b/pkg/config/agent_management_remote_config_test.go @@ -5,6 +5,7 @@ import ( "time" "github.com/grafana/agent/pkg/metrics/instance" + "github.com/prometheus/prometheus/model/labels" "github.com/stretchr/testify/require" ) @@ -179,4 +180,47 @@ integration_configs: require.Equal(t, true, c.Integrations.ConfigV1.ReplaceInstanceLabel) require.Equal(t, 5*time.Second, c.Integrations.ConfigV1.IntegrationRestartBackoff) }) + + t.Run("external labels provided", func(t *testing.T) { + rc := RemoteConfig{ + BaseConfig: BaseConfigContent(baseConfig), + Snippets: allSnippets, + AgentMetadata: AgentMetadata{ + ExternalLabels: map[string]string{ + "foo": "bar", + }, + }, + } + c, err := rc.BuildAgentConfig() + require.NoError(t, err) + require.Equal(t, 1, len(c.Logs.Configs)) + require.Equal(t, 1, len(c.Metrics.Configs)) + require.Contains(t, c.Metrics.Global.Prometheus.ExternalLabels, labels.Label{Name: "foo", Value: "bar"}) + }) + + t.Run("external labels don't override base config", func(t *testing.T) { + baseConfig := ` +server: + log_level: debug +metrics: + global: + external_labels: + foo: bar +` + rc := RemoteConfig{ + BaseConfig: BaseConfigContent(baseConfig), + Snippets: allSnippets, + AgentMetadata: AgentMetadata{ + ExternalLabels: map[string]string{ + "foo": "baz", + }, + }, + } + c, err := rc.BuildAgentConfig() + require.NoError(t, err) + require.Equal(t, 1, len(c.Logs.Configs)) + require.Equal(t, 1, len(c.Metrics.Configs)) + require.Contains(t, c.Metrics.Global.Prometheus.ExternalLabels, labels.Label{Name: "foo", Value: "bar"}) + require.NotContains(t, c.Metrics.Global.Prometheus.ExternalLabels, labels.Label{Name: "foo", Value: "baz"}) + }) } diff --git a/pkg/config/agentmanagement_remote_config.go b/pkg/config/agentmanagement_remote_config.go index d40b70551da1..78d3685a0a13 100644 --- a/pkg/config/agentmanagement_remote_config.go +++ b/pkg/config/agentmanagement_remote_config.go @@ -6,13 +6,15 @@ import ( "github.com/grafana/agent/pkg/metrics/instance" "github.com/grafana/loki/clients/pkg/promtail/scrapeconfig" pc "github.com/prometheus/prometheus/config" + "github.com/prometheus/prometheus/model/labels" "gopkg.in/yaml.v2" ) type ( RemoteConfig struct { - BaseConfig BaseConfigContent `json:"base_config" yaml:"base_config"` - Snippets []Snippet `json:"snippets" yaml:"snippets"` + BaseConfig BaseConfigContent `json:"base_config" yaml:"base_config"` + Snippets []Snippet `json:"snippets" yaml:"snippets"` + AgentMetadata AgentMetadata `json:"agent_metadata,omitempty" yaml:"agent_metadata,omitempty"` } // BaseConfigContent is the content of a base config @@ -24,6 +26,10 @@ type ( Config string `json:"config" yaml:"config"` } + AgentMetadata struct { + ExternalLabels map[string]string `json:"external_labels,omitempty" yaml:"external_labels,omitempty"` + } + // SnippetContent defines the internal structure of a snippet configuration. SnippetContent struct { // MetricsScrapeConfigs is a YAML containing list of metrics scrape configs. @@ -63,6 +69,7 @@ func (rc *RemoteConfig) BuildAgentConfig() (*Config, error) { if err != nil { return nil, err } + appendExternalLabels(&c, rc.AgentMetadata.ExternalLabels) return &c, nil } @@ -116,3 +123,14 @@ func appendSnippets(c *Config, snippets []Snippet) error { c.Integrations.ConfigV1.Integrations = append(c.Integrations.ConfigV1.Integrations, integrationConfigs.Integrations...) return nil } + +func appendExternalLabels(c *Config, externalLabels map[string]string) { + // Start off with the existing external labels, which will only be added to (not replaced) + newExternalLabels := c.Metrics.Global.Prometheus.ExternalLabels.Map() + for k, v := range externalLabels { + if _, ok := newExternalLabels[k]; !ok { + newExternalLabels[k] = v + } + } + c.Metrics.Global.Prometheus.ExternalLabels = labels.FromMap(newExternalLabels) +} From 94a6346425e1f72f2f50234b9dfcfec10a25d1ad Mon Sep 17 00:00:00 2001 From: Mischa Thompson Date: Tue, 31 Oct 2023 13:38:43 -0700 Subject: [PATCH 2/2] Add some base case tests and avoid doing anything if there are no external labels provided --- .../agent_management_remote_config_test.go | 30 +++++++++++++++++++ pkg/config/agentmanagement_remote_config.go | 4 +++ 2 files changed, 34 insertions(+) diff --git a/pkg/config/agent_management_remote_config_test.go b/pkg/config/agent_management_remote_config_test.go index d9bb7ed2b7a8..887173411994 100644 --- a/pkg/config/agent_management_remote_config_test.go +++ b/pkg/config/agent_management_remote_config_test.go @@ -181,6 +181,36 @@ integration_configs: require.Equal(t, 5*time.Second, c.Integrations.ConfigV1.IntegrationRestartBackoff) }) + t.Run("no external labels provided", func(t *testing.T) { + rc := RemoteConfig{ + BaseConfig: BaseConfigContent(baseConfig), + Snippets: allSnippets, + } + c, err := rc.BuildAgentConfig() + require.NoError(t, err) + require.Equal(t, 1, len(c.Logs.Configs)) + require.Empty(t, c.Metrics.Global.Prometheus.ExternalLabels) + }) + + t.Run("no external labels provided in remote config", func(t *testing.T) { + baseConfig := ` +server: + log_level: debug +metrics: + global: + external_labels: + foo: bar` + rc := RemoteConfig{ + BaseConfig: BaseConfigContent(baseConfig), + Snippets: allSnippets, + } + c, err := rc.BuildAgentConfig() + require.NoError(t, err) + require.Equal(t, 1, len(c.Logs.Configs)) + require.Equal(t, 1, len(c.Metrics.Global.Prometheus.ExternalLabels)) + require.Contains(t, c.Metrics.Global.Prometheus.ExternalLabels, labels.Label{Name: "foo", Value: "bar"}) + }) + t.Run("external labels provided", func(t *testing.T) { rc := RemoteConfig{ BaseConfig: BaseConfigContent(baseConfig), diff --git a/pkg/config/agentmanagement_remote_config.go b/pkg/config/agentmanagement_remote_config.go index 78d3685a0a13..f5deeed9a47c 100644 --- a/pkg/config/agentmanagement_remote_config.go +++ b/pkg/config/agentmanagement_remote_config.go @@ -125,6 +125,10 @@ func appendSnippets(c *Config, snippets []Snippet) error { } func appendExternalLabels(c *Config, externalLabels map[string]string) { + // Avoid doing anything if there are no external labels + if len(externalLabels) == 0 { + return + } // Start off with the existing external labels, which will only be added to (not replaced) newExternalLabels := c.Metrics.Global.Prometheus.ExternalLabels.Map() for k, v := range externalLabels {