HCP Add node id/name to config (#17750) (#17796)

Co-authored-by: chappie <6537530+chapmanc@users.noreply.github.com>
pull/17728/head^2
hc-github-team-consul-core 2023-06-16 18:25:33 -04:00 committed by GitHub
parent aa4b01adc4
commit 404bc0f091
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
12 changed files with 136 additions and 54 deletions

View File

@ -984,7 +984,7 @@ func (b *builder) build() (rt RuntimeConfig, err error) {
AutoEncryptIPSAN: autoEncryptIPSAN, AutoEncryptIPSAN: autoEncryptIPSAN,
AutoEncryptAllowTLS: autoEncryptAllowTLS, AutoEncryptAllowTLS: autoEncryptAllowTLS,
AutoConfig: autoConfig, AutoConfig: autoConfig,
Cloud: b.cloudConfigVal(c.Cloud), Cloud: b.cloudConfigVal(c),
ConnectEnabled: connectEnabled, ConnectEnabled: connectEnabled,
ConnectCAProvider: connectCAProvider, ConnectCAProvider: connectCAProvider,
ConnectCAConfig: connectCAConfig, ConnectCAConfig: connectCAConfig,
@ -2541,21 +2541,26 @@ func validateAutoConfigAuthorizer(rt RuntimeConfig) error {
return nil return nil
} }
func (b *builder) cloudConfigVal(v *CloudConfigRaw) hcpconfig.CloudConfig { func (b *builder) cloudConfigVal(v Config) hcpconfig.CloudConfig {
val := hcpconfig.CloudConfig{ val := hcpconfig.CloudConfig{
ResourceID: os.Getenv("HCP_RESOURCE_ID"), ResourceID: os.Getenv("HCP_RESOURCE_ID"),
} }
if v == nil { // Node id might get overriden in setup.go:142
nodeID := stringVal(v.NodeID)
val.NodeID = types.NodeID(nodeID)
val.NodeName = b.nodeName(v.NodeName)
if v.Cloud == nil {
return val return val
} }
val.ClientID = stringVal(v.ClientID) val.ClientID = stringVal(v.Cloud.ClientID)
val.ClientSecret = stringVal(v.ClientSecret) val.ClientSecret = stringVal(v.Cloud.ClientSecret)
val.AuthURL = stringVal(v.AuthURL) val.AuthURL = stringVal(v.Cloud.AuthURL)
val.Hostname = stringVal(v.Hostname) val.Hostname = stringVal(v.Cloud.Hostname)
val.ScadaAddress = stringVal(v.ScadaAddress) val.ScadaAddress = stringVal(v.Cloud.ScadaAddress)
if resourceID := stringVal(v.ResourceID); resourceID != "" { if resourceID := stringVal(v.Cloud.ResourceID); resourceID != "" {
val.ResourceID = resourceID val.ResourceID = resourceID
} }
return val return val

View File

@ -619,6 +619,7 @@ func TestLoad_IntegrationWithFlags(t *testing.T) {
rt.NodeName = "a" rt.NodeName = "a"
rt.TLS.NodeName = "a" rt.TLS.NodeName = "a"
rt.DataDir = dataDir rt.DataDir = dataDir
rt.Cloud.NodeName = "a"
}, },
}) })
run(t, testCase{ run(t, testCase{
@ -630,6 +631,7 @@ func TestLoad_IntegrationWithFlags(t *testing.T) {
expected: func(rt *RuntimeConfig) { expected: func(rt *RuntimeConfig) {
rt.NodeID = "a" rt.NodeID = "a"
rt.DataDir = dataDir rt.DataDir = dataDir
rt.Cloud.NodeID = "a"
}, },
}) })
run(t, testCase{ run(t, testCase{
@ -2319,6 +2321,8 @@ func TestLoad_IntegrationWithFlags(t *testing.T) {
rt.Cloud = hcpconfig.CloudConfig{ rt.Cloud = hcpconfig.CloudConfig{
// ID is only populated from env if not populated from other sources. // ID is only populated from env if not populated from other sources.
ResourceID: "env-id", ResourceID: "env-id",
NodeName: "thehostname",
NodeID: "",
} }
// server things // server things
@ -2359,6 +2363,7 @@ func TestLoad_IntegrationWithFlags(t *testing.T) {
rt.Cloud = hcpconfig.CloudConfig{ rt.Cloud = hcpconfig.CloudConfig{
// ID is only populated from env if not populated from other sources. // ID is only populated from env if not populated from other sources.
ResourceID: "file-id", ResourceID: "file-id",
NodeName: "thehostname",
} }
// server things // server things
@ -6317,6 +6322,8 @@ func TestLoad_FullConfig(t *testing.T) {
Hostname: "DH4bh7aC", Hostname: "DH4bh7aC",
AuthURL: "332nCdR2", AuthURL: "332nCdR2",
ScadaAddress: "aoeusth232", ScadaAddress: "aoeusth232",
NodeID: types.NodeID("AsUIlw99"),
NodeName: "otlLxGaI",
}, },
DNSAddrs: []net.Addr{tcpAddr("93.95.95.81:7001"), udpAddr("93.95.95.81:7001")}, DNSAddrs: []net.Addr{tcpAddr("93.95.95.81:7001"), udpAddr("93.95.95.81:7001")},
DNSARecordLimit: 29907, DNSARecordLimit: 29907,

View File

@ -134,7 +134,9 @@
"ManagementToken": "hidden", "ManagementToken": "hidden",
"ResourceID": "cluster1", "ResourceID": "cluster1",
"ScadaAddress": "", "ScadaAddress": "",
"TLSConfig": null "TLSConfig": null,
"NodeID": "",
"NodeName": ""
}, },
"ConfigEntryBootstrap": [], "ConfigEntryBootstrap": [],
"ConnectCAConfig": {}, "ConnectCAConfig": {},

View File

@ -313,9 +313,14 @@ func (t *TelemetryConfig) Enabled() (string, bool) {
} }
// DefaultLabels returns a set of <key, value> string pairs that must be added as attributes to all exported telemetry data. // DefaultLabels returns a set of <key, value> string pairs that must be added as attributes to all exported telemetry data.
func (t *TelemetryConfig) DefaultLabels(nodeID string) map[string]string { func (t *TelemetryConfig) DefaultLabels(cfg config.CloudConfig) map[string]string {
labels := map[string]string{ labels := make(map[string]string)
"node_id": nodeID, // used to delineate Consul nodes in graphs nodeID := string(cfg.NodeID)
if nodeID != "" {
labels["node_id"] = nodeID
}
if cfg.NodeName != "" {
labels["node_name"] = cfg.NodeName
} }
for k, v := range t.Labels { for k, v := range t.Labels {

View File

@ -4,6 +4,8 @@ import (
"context" "context"
"testing" "testing"
"github.com/hashicorp/consul/agent/hcp/config"
"github.com/hashicorp/consul/types"
"github.com/hashicorp/hcp-sdk-go/clients/cloud-consul-telemetry-gateway/preview/2023-04-14/client/consul_telemetry_service" "github.com/hashicorp/hcp-sdk-go/clients/cloud-consul-telemetry-gateway/preview/2023-04-14/client/consul_telemetry_service"
"github.com/hashicorp/hcp-sdk-go/clients/cloud-consul-telemetry-gateway/preview/2023-04-14/models" "github.com/hashicorp/hcp-sdk-go/clients/cloud-consul-telemetry-gateway/preview/2023-04-14/models"
"github.com/stretchr/testify/mock" "github.com/stretchr/testify/mock"
@ -147,3 +149,53 @@ func TestConvertTelemetryConfig(t *testing.T) {
}) })
} }
} }
func Test_DefaultLabels(t *testing.T) {
for name, tc := range map[string]struct {
cfg config.CloudConfig
expectedLabels map[string]string
}{
"Success": {
cfg: config.CloudConfig{
NodeID: types.NodeID("nodeyid"),
NodeName: "nodey",
},
expectedLabels: map[string]string{
"node_id": "nodeyid",
"node_name": "nodey",
},
},
"NoNodeID": {
cfg: config.CloudConfig{
NodeID: types.NodeID(""),
NodeName: "nodey",
},
expectedLabels: map[string]string{
"node_name": "nodey",
},
},
"NoNodeName": {
cfg: config.CloudConfig{
NodeID: types.NodeID("nodeyid"),
NodeName: "",
},
expectedLabels: map[string]string{
"node_id": "nodeyid",
},
},
"Empty": {
cfg: config.CloudConfig{
NodeID: "",
NodeName: "",
},
expectedLabels: map[string]string{},
},
} {
t.Run(name, func(t *testing.T) {
tCfg := &TelemetryConfig{}
labels := tCfg.DefaultLabels(tc.cfg)
require.Equal(t, labels, tc.expectedLabels)
})
}
}

View File

@ -58,7 +58,7 @@ type otlpClient struct {
// NewMetricsClient returns a configured MetricsClient. // NewMetricsClient returns a configured MetricsClient.
// The current implementation uses otlpClient to provide retry functionality. // The current implementation uses otlpClient to provide retry functionality.
func NewMetricsClient(cfg CloudConfig, ctx context.Context) (MetricsClient, error) { func NewMetricsClient(ctx context.Context, cfg CloudConfig) (MetricsClient, error) {
if cfg == nil { if cfg == nil {
return nil, fmt.Errorf("failed to init telemetry client: provide valid cloudCfg (Cloud Configuration for TLS)") return nil, fmt.Errorf("failed to init telemetry client: provide valid cloudCfg (Cloud Configuration for TLS)")
} }

View File

@ -52,7 +52,7 @@ func TestNewMetricsClient(t *testing.T) {
}, },
} { } {
t.Run(name, func(t *testing.T) { t.Run(name, func(t *testing.T) {
client, err := NewMetricsClient(test.cfg, test.ctx) client, err := NewMetricsClient(test.ctx, test.cfg)
if test.wantErr != "" { if test.wantErr != "" {
require.Error(t, err) require.Error(t, err)
require.Contains(t, err.Error(), test.wantErr) require.Contains(t, err.Error(), test.wantErr)
@ -118,7 +118,7 @@ func TestExportMetrics(t *testing.T) {
})) }))
defer srv.Close() defer srv.Close()
client, err := NewMetricsClient(MockCloudCfg{}, context.Background()) client, err := NewMetricsClient(context.Background(), MockCloudCfg{})
require.NoError(t, err) require.NoError(t, err)
ctx := context.Background() ctx := context.Background()

View File

@ -0,0 +1,5 @@
package client
type MockMetricsClient struct {
MetricsClient
}

View File

@ -6,6 +6,7 @@ package config
import ( import (
"crypto/tls" "crypto/tls"
"github.com/hashicorp/consul/types"
hcpcfg "github.com/hashicorp/hcp-sdk-go/config" hcpcfg "github.com/hashicorp/hcp-sdk-go/config"
"github.com/hashicorp/hcp-sdk-go/resource" "github.com/hashicorp/hcp-sdk-go/resource"
) )
@ -25,6 +26,9 @@ type CloudConfig struct {
// TlsConfig for testing. // TlsConfig for testing.
TLSConfig *tls.Config TLSConfig *tls.Config
NodeID types.NodeID
NodeName string
} }
func (c *CloudConfig) WithTLSConfig(cfg *tls.Config) { func (c *CloudConfig) WithTLSConfig(cfg *tls.Config) {

View File

@ -14,7 +14,6 @@ import (
"github.com/hashicorp/consul/agent/hcp/config" "github.com/hashicorp/consul/agent/hcp/config"
"github.com/hashicorp/consul/agent/hcp/scada" "github.com/hashicorp/consul/agent/hcp/scada"
"github.com/hashicorp/consul/agent/hcp/telemetry" "github.com/hashicorp/consul/agent/hcp/telemetry"
"github.com/hashicorp/consul/types"
"github.com/hashicorp/go-hclog" "github.com/hashicorp/go-hclog"
) )
@ -25,10 +24,13 @@ type Deps struct {
Sink metrics.MetricSink Sink metrics.MetricSink
} }
func NewDeps(cfg config.CloudConfig, logger hclog.Logger, nodeID types.NodeID) (Deps, error) { func NewDeps(cfg config.CloudConfig, logger hclog.Logger) (Deps, error) {
ctx := context.Background()
ctx = hclog.WithContext(ctx, logger)
client, err := hcpclient.NewClient(cfg) client, err := hcpclient.NewClient(cfg)
if err != nil { if err != nil {
return Deps{}, fmt.Errorf("failed to init client: %w:", err) return Deps{}, fmt.Errorf("failed to init client: %w", err)
} }
provider, err := scada.New(cfg, logger.Named("scada")) provider, err := scada.New(cfg, logger.Named("scada"))
@ -36,7 +38,13 @@ func NewDeps(cfg config.CloudConfig, logger hclog.Logger, nodeID types.NodeID) (
return Deps{}, fmt.Errorf("failed to init scada: %w", err) return Deps{}, fmt.Errorf("failed to init scada: %w", err)
} }
sink := sink(client, &cfg, logger.Named("sink"), nodeID) metricsClient, err := hcpclient.NewMetricsClient(ctx, &cfg)
if err != nil {
logger.Error("failed to init metrics client", "error", err)
return Deps{}, fmt.Errorf("failed to init metrics client: %w", err)
}
sink := sink(ctx, client, metricsClient, cfg)
return Deps{ return Deps{
Client: client, Client: client,
@ -48,10 +56,13 @@ func NewDeps(cfg config.CloudConfig, logger hclog.Logger, nodeID types.NodeID) (
// sink provides initializes an OTELSink which forwards Consul metrics to HCP. // sink provides initializes an OTELSink which forwards Consul metrics to HCP.
// The sink is only initialized if the server is registered with the management plane (CCM). // The sink is only initialized if the server is registered with the management plane (CCM).
// This step should not block server initialization, so errors are logged, but not returned. // This step should not block server initialization, so errors are logged, but not returned.
func sink(hcpClient hcpclient.Client, cfg hcpclient.CloudConfig, logger hclog.Logger, nodeID types.NodeID) metrics.MetricSink { func sink(
ctx := context.Background() ctx context.Context,
ctx = hclog.WithContext(ctx, logger) hcpClient hcpclient.Client,
metricsClient hcpclient.MetricsClient,
cfg config.CloudConfig,
) metrics.MetricSink {
logger := hclog.FromContext(ctx).Named("sink")
reqCtx, cancel := context.WithTimeout(ctx, 5*time.Second) reqCtx, cancel := context.WithTimeout(ctx, 5*time.Second)
defer cancel() defer cancel()
@ -72,16 +83,10 @@ func sink(hcpClient hcpclient.Client, cfg hcpclient.CloudConfig, logger hclog.Lo
return nil return nil
} }
metricsClient, err := hcpclient.NewMetricsClient(cfg, ctx)
if err != nil {
logger.Error("failed to init metrics client", "error", err)
return nil
}
sinkOpts := &telemetry.OTELSinkOpts{ sinkOpts := &telemetry.OTELSinkOpts{
Ctx: ctx, Ctx: ctx,
Reader: telemetry.NewOTELReader(metricsClient, u, telemetry.DefaultExportInterval), Reader: telemetry.NewOTELReader(metricsClient, u, telemetry.DefaultExportInterval),
Labels: telemetryCfg.DefaultLabels(string(nodeID)), Labels: telemetryCfg.DefaultLabels(cfg),
Filters: telemetryCfg.MetricsConfig.Filters, Filters: telemetryCfg.MetricsConfig.Filters,
} }

View File

@ -1,10 +1,11 @@
package hcp package hcp
import ( import (
"context"
"fmt" "fmt"
"testing" "testing"
"github.com/hashicorp/go-hclog" "github.com/hashicorp/consul/agent/hcp/config"
"github.com/stretchr/testify/mock" "github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
@ -16,7 +17,7 @@ func TestSink(t *testing.T) {
t.Parallel() t.Parallel()
for name, test := range map[string]struct { for name, test := range map[string]struct {
expect func(*client.MockClient) expect func(*client.MockClient)
mockCloudCfg client.CloudConfig cloudCfg config.CloudConfig
expectedSink bool expectedSink bool
}{ }{
"success": { "success": {
@ -28,7 +29,10 @@ func TestSink(t *testing.T) {
}, },
}, nil) }, nil)
}, },
mockCloudCfg: client.MockCloudCfg{}, cloudCfg: config.CloudConfig{
NodeID: types.NodeID("nodeyid"),
NodeName: "nodey",
},
expectedSink: true, expectedSink: true,
}, },
"noSinkWhenServerNotRegisteredWithCCM": { "noSinkWhenServerNotRegisteredWithCCM": {
@ -40,26 +44,13 @@ func TestSink(t *testing.T) {
}, },
}, nil) }, nil)
}, },
mockCloudCfg: client.MockCloudCfg{}, cloudCfg: config.CloudConfig{},
}, },
"noSinkWhenCCMVerificationFails": { "noSinkWhenCCMVerificationFails": {
expect: func(mockClient *client.MockClient) { expect: func(mockClient *client.MockClient) {
mockClient.EXPECT().FetchTelemetryConfig(mock.Anything).Return(nil, fmt.Errorf("fetch failed")) mockClient.EXPECT().FetchTelemetryConfig(mock.Anything).Return(nil, fmt.Errorf("fetch failed"))
}, },
mockCloudCfg: client.MockCloudCfg{}, cloudCfg: config.CloudConfig{},
},
"noSinkWhenMetricsClientInitFails": {
mockCloudCfg: client.MockCloudCfg{
ConfigErr: fmt.Errorf("test bad hcp config"),
},
expect: func(mockClient *client.MockClient) {
mockClient.EXPECT().FetchTelemetryConfig(mock.Anything).Return(&client.TelemetryConfig{
Endpoint: "https://test.com",
MetricsConfig: &client.MetricsConfig{
Endpoint: "",
},
}, nil)
},
}, },
"failsWithFetchTelemetryFailure": { "failsWithFetchTelemetryFailure": {
expect: func(mockClient *client.MockClient) { expect: func(mockClient *client.MockClient) {
@ -93,14 +84,17 @@ func TestSink(t *testing.T) {
t.Run(name, func(t *testing.T) { t.Run(name, func(t *testing.T) {
t.Parallel() t.Parallel()
c := client.NewMockClient(t) c := client.NewMockClient(t)
l := hclog.NewNullLogger() mc := client.MockMetricsClient{}
test.expect(c) test.expect(c)
sinkOpts := sink(c, test.mockCloudCfg, l, types.NodeID("server1234")) ctx := context.Background()
s := sink(ctx, c, mc, test.cloudCfg)
if !test.expectedSink { if !test.expectedSink {
require.Nil(t, sinkOpts) require.Nil(t, s)
return return
} }
require.NotNil(t, sinkOpts) require.NotNil(t, s)
}) })
} }
} }

View File

@ -138,7 +138,10 @@ func NewBaseDeps(configLoader ConfigLoader, logOut io.Writer, providedLogger hcl
var extraSinks []metrics.MetricSink var extraSinks []metrics.MetricSink
if cfg.IsCloudEnabled() { if cfg.IsCloudEnabled() {
d.HCP, err = hcp.NewDeps(cfg.Cloud, d.Logger.Named("hcp"), cfg.NodeID) // This values is set late within newNodeIDFromConfig above
cfg.Cloud.NodeID = cfg.NodeID
d.HCP, err = hcp.NewDeps(cfg.Cloud, d.Logger.Named("hcp"))
if err != nil { if err != nil {
return d, err return d, err
} }