Refactor to use embedded struct.

This commit is contained in:
Paul Banks 2018-06-14 13:52:48 +01:00 committed by Jack Pearkes
parent 2f8c1d2059
commit c6ef6a61c9
12 changed files with 551 additions and 366 deletions

View File

@ -2924,7 +2924,8 @@ func (a *Agent) ReloadConfig(newCfg *config.RuntimeConfig) error {
} }
// Update filtered metrics // Update filtered metrics
metrics.UpdateFilter(newCfg.TelemetryAllowedPrefixes, newCfg.TelemetryBlockedPrefixes) metrics.UpdateFilter(newCfg.Telemetry.AllowedPrefixes,
newCfg.Telemetry.BlockedPrefixes)
a.State.SetDiscardCheckOutput(newCfg.DiscardCheckOutput) a.State.SetDiscardCheckOutput(newCfg.DiscardCheckOutput)

View File

@ -9,6 +9,8 @@ import (
"strings" "strings"
"time" "time"
"github.com/mitchellh/mapstructure"
"github.com/hashicorp/go-memdb" "github.com/hashicorp/go-memdb"
"github.com/mitchellh/hashstructure" "github.com/mitchellh/hashstructure"
@ -104,7 +106,7 @@ func (s *HTTPServer) AgentMetrics(resp http.ResponseWriter, req *http.Request) (
return nil, acl.ErrPermissionDenied return nil, acl.ErrPermissionDenied
} }
if enablePrometheusOutput(req) { if enablePrometheusOutput(req) {
if s.agent.config.TelemetryPrometheusRetentionTime < 1 { if s.agent.config.Telemetry.PrometheusRetentionTime < 1 {
resp.WriteHeader(http.StatusUnsupportedMediaType) resp.WriteHeader(http.StatusUnsupportedMediaType)
fmt.Fprint(resp, "Prometheus is not enable since its retention time is not positive") fmt.Fprint(resp, "Prometheus is not enable since its retention time is not positive")
return nil, nil return nil, nil
@ -1061,26 +1063,35 @@ func (s *HTTPServer) AgentConnectProxyConfig(resp http.ResponseWriter, req *http
target.Port) target.Port)
} }
// Add telemetry config // Add telemetry config. Copy the global config so we can customize the
telemetry := s.agent.config.TelemetryConfig(false) // prefix.
if len(telemetry) > 0 { telemetryCfg := s.agent.config.Telemetry
// Rely on the fact that TelemetryConfig makes a new map each call to telemetryCfg.MetricsPrefix = telemetryCfg.MetricsPrefix + ".proxy." + target.ID
// override the prefix here without affecting other callers.
telemetry["MetricsPrefix"] = "consul.proxy." + target.ID
// Merge with any config passed by the user to allow service definition // First see if the user has specified telemetry
// to override. if userRaw, ok := config["telemetry"]; ok {
if userRaw, ok := config["telemetry"]; ok { // User specified domething, see if it is compatible with agent
if userT, ok := userRaw.(map[string]interface{}); ok { // telemetry config:
for k, v := range telemetry { var uCfg lib.TelemetryConfig
if _, ok := userT[k]; !ok { dec, err := mapstructure.NewDecoder(&mapstructure.DecoderConfig{
userT[k] = v Result: &uCfg,
} // Make sure that if the user passes something that isn't just a
} // simple override of a valid TelemetryConfig that we fail so that we
// don't clobber their custom config.
ErrorUnused: true,
})
if err == nil {
if err = dec.Decode(userRaw); err == nil {
// It did decode! Merge any unspecified fields from agent config.
uCfg.MergeDefaults(&telemetryCfg)
config["telemetry"] = uCfg
} }
} else {
config["telemetry"] = telemetry
} }
// Failed to decode, just keep user's config["telemetry"] verbatim
// with no agent merge.
} else {
// Add agent telemetry config.
config["telemetry"] = telemetryCfg
} }
reply := &api.ConnectProxyConfig{ reply := &api.ConnectProxyConfig{

View File

@ -21,6 +21,7 @@ import (
"github.com/hashicorp/consul/agent/connect" "github.com/hashicorp/consul/agent/connect"
"github.com/hashicorp/consul/agent/structs" "github.com/hashicorp/consul/agent/structs"
"github.com/hashicorp/consul/api" "github.com/hashicorp/consul/api"
"github.com/hashicorp/consul/lib"
"github.com/hashicorp/consul/logger" "github.com/hashicorp/consul/logger"
"github.com/hashicorp/consul/testutil/retry" "github.com/hashicorp/consul/testutil/retry"
"github.com/hashicorp/consul/types" "github.com/hashicorp/consul/types"
@ -3248,10 +3249,10 @@ func TestAgentConnectProxyConfig_aclServiceReadDeny(t *testing.T) {
require.True(acl.IsErrPermissionDenied(err)) require.True(acl.IsErrPermissionDenied(err))
} }
func makeTelemetryDefaults(targetID string) map[string]interface{} { func makeTelemetryDefaults(targetID string) lib.TelemetryConfig {
return map[string]interface{}{ return lib.TelemetryConfig{
"FilterDefault": true, FilterDefault: true,
"MetricsPrefix": "consul.proxy." + targetID, MetricsPrefix: "consul.proxy." + targetID,
} }
} }
@ -3403,10 +3404,10 @@ func TestAgentConnectProxyConfig_ConfigHandling(t *testing.T) {
"local_service_address": "127.0.0.1:8000", // port from service reg "local_service_address": "127.0.0.1:8000", // port from service reg
"connect_timeout_ms": 1000, "connect_timeout_ms": 1000,
"foo": "bar", "foo": "bar",
"telemetry": map[string]interface{}{ "telemetry": lib.TelemetryConfig{
"FilterDefault": true, FilterDefault: true,
"MetricsPrefix": "consul.proxy." + reg.ID, MetricsPrefix: "consul.proxy." + reg.ID,
"StatsiteAddr": "localhost:8989", StatsiteAddr: "localhost:8989",
}, },
}, },
}, },
@ -3445,7 +3446,8 @@ func TestAgentConnectProxyConfig_ConfigHandling(t *testing.T) {
"bind_port": 1024, "bind_port": 1024,
"local_service_address": "127.0.0.1:9191", "local_service_address": "127.0.0.1:9191",
"telemetry": map[string]interface{}{ "telemetry": map[string]interface{}{
"StatsiteAddr": "stats.it:10101", "statsite_address": "stats.it:10101",
"metrics_prefix": "foo", // important! checks that our prefix logic respects user customization
}, },
}, },
}, },
@ -3456,10 +3458,47 @@ func TestAgentConnectProxyConfig_ConfigHandling(t *testing.T) {
"bind_port": float64(1024), "bind_port": float64(1024),
"local_service_address": "127.0.0.1:9191", "local_service_address": "127.0.0.1:9191",
"connect_timeout_ms": float64(2000), "connect_timeout_ms": float64(2000),
"telemetry": lib.TelemetryConfig{
FilterDefault: true,
MetricsPrefix: "foo",
StatsiteAddr: "stats.it:10101",
},
},
},
{
name: "reg telemetry not compatible, preserved with no merge",
globalConfig: `
connect {
enabled = true
proxy {
allow_managed_api_registration = true
}
}
ports {
proxy_min_port = 10000
proxy_max_port = 10000
}
telemetry {
statsite_address = "localhost:8989"
}
`,
proxy: structs.ServiceDefinitionConnectProxy{
ExecMode: "script",
Command: []string{"foo.sh"},
Config: map[string]interface{}{
"telemetry": map[string]interface{}{
"foo": "bar",
},
},
},
wantMode: api.ProxyExecModeScript,
wantCommand: []string{"foo.sh"},
wantConfig: map[string]interface{}{
"bind_address": "127.0.0.1",
"bind_port": 10000, // "randomly" chosen from our range of 1
"local_service_address": "127.0.0.1:8000", // port from service reg
"telemetry": map[string]interface{}{ "telemetry": map[string]interface{}{
"FilterDefault": true, "foo": "bar",
"MetricsPrefix": "consul.proxy." + reg.ID,
"StatsiteAddr": "stats.it:10101",
}, },
}, },
}, },

View File

@ -17,6 +17,7 @@ import (
"github.com/hashicorp/consul/agent/consul" "github.com/hashicorp/consul/agent/consul"
"github.com/hashicorp/consul/agent/structs" "github.com/hashicorp/consul/agent/structs"
"github.com/hashicorp/consul/ipaddr" "github.com/hashicorp/consul/ipaddr"
"github.com/hashicorp/consul/lib"
"github.com/hashicorp/consul/tlsutil" "github.com/hashicorp/consul/tlsutil"
"github.com/hashicorp/consul/types" "github.com/hashicorp/consul/types"
multierror "github.com/hashicorp/go-multierror" multierror "github.com/hashicorp/go-multierror"
@ -625,29 +626,31 @@ func (b *Builder) Build() (rt RuntimeConfig, err error) {
HTTPResponseHeaders: c.HTTPConfig.ResponseHeaders, HTTPResponseHeaders: c.HTTPConfig.ResponseHeaders,
// Telemetry // Telemetry
TelemetryCirconusAPIApp: b.stringVal(c.Telemetry.CirconusAPIApp), Telemetry: lib.TelemetryConfig{
TelemetryCirconusAPIToken: b.stringVal(c.Telemetry.CirconusAPIToken), CirconusAPIApp: b.stringVal(c.Telemetry.CirconusAPIApp),
TelemetryCirconusAPIURL: b.stringVal(c.Telemetry.CirconusAPIURL), CirconusAPIToken: b.stringVal(c.Telemetry.CirconusAPIToken),
TelemetryCirconusBrokerID: b.stringVal(c.Telemetry.CirconusBrokerID), CirconusAPIURL: b.stringVal(c.Telemetry.CirconusAPIURL),
TelemetryCirconusBrokerSelectTag: b.stringVal(c.Telemetry.CirconusBrokerSelectTag), CirconusBrokerID: b.stringVal(c.Telemetry.CirconusBrokerID),
TelemetryCirconusCheckDisplayName: b.stringVal(c.Telemetry.CirconusCheckDisplayName), CirconusBrokerSelectTag: b.stringVal(c.Telemetry.CirconusBrokerSelectTag),
TelemetryCirconusCheckForceMetricActivation: b.stringVal(c.Telemetry.CirconusCheckForceMetricActivation), CirconusCheckDisplayName: b.stringVal(c.Telemetry.CirconusCheckDisplayName),
TelemetryCirconusCheckID: b.stringVal(c.Telemetry.CirconusCheckID), CirconusCheckForceMetricActivation: b.stringVal(c.Telemetry.CirconusCheckForceMetricActivation),
TelemetryCirconusCheckInstanceID: b.stringVal(c.Telemetry.CirconusCheckInstanceID), CirconusCheckID: b.stringVal(c.Telemetry.CirconusCheckID),
TelemetryCirconusCheckSearchTag: b.stringVal(c.Telemetry.CirconusCheckSearchTag), CirconusCheckInstanceID: b.stringVal(c.Telemetry.CirconusCheckInstanceID),
TelemetryCirconusCheckTags: b.stringVal(c.Telemetry.CirconusCheckTags), CirconusCheckSearchTag: b.stringVal(c.Telemetry.CirconusCheckSearchTag),
TelemetryCirconusSubmissionInterval: b.stringVal(c.Telemetry.CirconusSubmissionInterval), CirconusCheckTags: b.stringVal(c.Telemetry.CirconusCheckTags),
TelemetryCirconusSubmissionURL: b.stringVal(c.Telemetry.CirconusSubmissionURL), CirconusSubmissionInterval: b.stringVal(c.Telemetry.CirconusSubmissionInterval),
TelemetryDisableHostname: b.boolVal(c.Telemetry.DisableHostname), CirconusSubmissionURL: b.stringVal(c.Telemetry.CirconusSubmissionURL),
TelemetryDogstatsdAddr: b.stringVal(c.Telemetry.DogstatsdAddr), DisableHostname: b.boolVal(c.Telemetry.DisableHostname),
TelemetryDogstatsdTags: c.Telemetry.DogstatsdTags, DogstatsdAddr: b.stringVal(c.Telemetry.DogstatsdAddr),
TelemetryPrometheusRetentionTime: b.durationVal("prometheus_retention_time", c.Telemetry.PrometheusRetentionTime), DogstatsdTags: c.Telemetry.DogstatsdTags,
TelemetryFilterDefault: b.boolVal(c.Telemetry.FilterDefault), PrometheusRetentionTime: b.durationVal("prometheus_retention_time", c.Telemetry.PrometheusRetentionTime),
TelemetryAllowedPrefixes: telemetryAllowedPrefixes, FilterDefault: b.boolVal(c.Telemetry.FilterDefault),
TelemetryBlockedPrefixes: telemetryBlockedPrefixes, AllowedPrefixes: telemetryAllowedPrefixes,
TelemetryMetricsPrefix: b.stringVal(c.Telemetry.MetricsPrefix), BlockedPrefixes: telemetryBlockedPrefixes,
TelemetryStatsdAddr: b.stringVal(c.Telemetry.StatsdAddr), MetricsPrefix: b.stringVal(c.Telemetry.MetricsPrefix),
TelemetryStatsiteAddr: b.stringVal(c.Telemetry.StatsiteAddr), StatsdAddr: b.stringVal(c.Telemetry.StatsdAddr),
StatsiteAddr: b.stringVal(c.Telemetry.StatsiteAddr),
},
// Agent // Agent
AdvertiseAddrLAN: advertiseAddrLAN, AdvertiseAddrLAN: advertiseAddrLAN,

View File

@ -9,6 +9,7 @@ import (
"time" "time"
"github.com/hashicorp/consul/agent/structs" "github.com/hashicorp/consul/agent/structs"
"github.com/hashicorp/consul/lib"
"github.com/hashicorp/consul/tlsutil" "github.com/hashicorp/consul/tlsutil"
"github.com/hashicorp/consul/types" "github.com/hashicorp/consul/types"
"golang.org/x/time/rate" "golang.org/x/time/rate"
@ -299,177 +300,8 @@ type RuntimeConfig struct {
// hcl: http_config { response_headers = map[string]string } // hcl: http_config { response_headers = map[string]string }
HTTPResponseHeaders map[string]string HTTPResponseHeaders map[string]string
// TelemetryCirconus*: see https://github.com/circonus-labs/circonus-gometrics // Embed Telemetry Config
// for more details on the various configuration options. Telemetry lib.TelemetryConfig
// Valid configuration combinations:
// - CirconusAPIToken
// metric management enabled (search for existing check or create a new one)
// - CirconusSubmissionUrl
// metric management disabled (use check with specified submission_url,
// broker must be using a public SSL certificate)
// - CirconusAPIToken + CirconusCheckSubmissionURL
// metric management enabled (use check with specified submission_url)
// - CirconusAPIToken + CirconusCheckID
// metric management enabled (use check with specified id)
// TelemetryCirconusAPIApp is an app name associated with API token.
// Default: "consul"
//
// hcl: telemetry { circonus_api_app = string }
TelemetryCirconusAPIApp string
// TelemetryCirconusAPIToken is a valid API Token used to create/manage check. If provided,
// metric management is enabled.
// Default: none
//
// hcl: telemetry { circonus_api_token = string }
TelemetryCirconusAPIToken string
// TelemetryCirconusAPIURL is the base URL to use for contacting the Circonus API.
// Default: "https://api.circonus.com/v2"
//
// hcl: telemetry { circonus_api_url = string }
TelemetryCirconusAPIURL string
// TelemetryCirconusBrokerID is an explicit broker to use when creating a new check. The numeric portion
// of broker._cid. If metric management is enabled and neither a Submission URL nor Check ID
// is provided, an attempt will be made to search for an existing check using Instance ID and
// Search Tag. If one is not found, a new HTTPTRAP check will be created.
// Default: use Select Tag if provided, otherwise, a random Enterprise Broker associated
// with the specified API token or the default Circonus Broker.
// Default: none
//
// hcl: telemetry { circonus_broker_id = string }
TelemetryCirconusBrokerID string
// TelemetryCirconusBrokerSelectTag is a special tag which will be used to select a broker when
// a Broker ID is not provided. The best use of this is to as a hint for which broker
// should be used based on *where* this particular instance is running.
// (e.g. a specific geo location or datacenter, dc:sfo)
// Default: none
//
// hcl: telemetry { circonus_broker_select_tag = string }
TelemetryCirconusBrokerSelectTag string
// TelemetryCirconusCheckDisplayName is the name for the check which will be displayed in the Circonus UI.
// Default: value of CirconusCheckInstanceID
//
// hcl: telemetry { circonus_check_display_name = string }
TelemetryCirconusCheckDisplayName string
// TelemetryCirconusCheckForceMetricActivation will force enabling metrics, as they are encountered,
// if the metric already exists and is NOT active. If check management is enabled, the default
// behavior is to add new metrics as they are encountered. If the metric already exists in the
// check, it will *NOT* be activated. This setting overrides that behavior.
// Default: "false"
//
// hcl: telemetry { circonus_check_metrics_activation = (true|false)
TelemetryCirconusCheckForceMetricActivation string
// TelemetryCirconusCheckID is the check id (not check bundle id) from a previously created
// HTTPTRAP check. The numeric portion of the check._cid field.
// Default: none
//
// hcl: telemetry { circonus_check_id = string }
TelemetryCirconusCheckID string
// TelemetryCirconusCheckInstanceID serves to uniquely identify the metrics coming from this "instance".
// It can be used to maintain metric continuity with transient or ephemeral instances as
// they move around within an infrastructure.
// Default: hostname:app
//
// hcl: telemetry { circonus_check_instance_id = string }
TelemetryCirconusCheckInstanceID string
// TelemetryCirconusCheckSearchTag is a special tag which, when coupled with the instance id, helps to
// narrow down the search results when neither a Submission URL or Check ID is provided.
// Default: service:app (e.g. service:consul)
//
// hcl: telemetry { circonus_check_search_tag = string }
TelemetryCirconusCheckSearchTag string
// TelemetryCirconusCheckSearchTag is a special tag which, when coupled with the instance id, helps to
// narrow down the search results when neither a Submission URL or Check ID is provided.
// Default: service:app (e.g. service:consul)
//
// hcl: telemetry { circonus_check_tags = string }
TelemetryCirconusCheckTags string
// TelemetryCirconusSubmissionInterval is the interval at which metrics are submitted to Circonus.
// Default: 10s
//
// hcl: telemetry { circonus_submission_interval = "duration" }
TelemetryCirconusSubmissionInterval string
// TelemetryCirconusCheckSubmissionURL is the check.config.submission_url field from a
// previously created HTTPTRAP check.
// Default: none
//
// hcl: telemetry { circonus_submission_url = string }
TelemetryCirconusSubmissionURL string
// DisableHostname will disable hostname prefixing for all metrics.
//
// hcl: telemetry { disable_hostname = (true|false)
TelemetryDisableHostname bool
// TelemetryDogStatsdAddr is the address of a dogstatsd instance. If provided,
// metrics will be sent to that instance
//
// hcl: telemetry { dogstatsd_addr = string }
TelemetryDogstatsdAddr string
// TelemetryDogStatsdTags are the global tags that should be sent with each packet to dogstatsd
// It is a list of strings, where each string looks like "my_tag_name:my_tag_value"
//
// hcl: telemetry { dogstatsd_tags = []string }
TelemetryDogstatsdTags []string
// PrometheusRetentionTime is the retention time for prometheus metrics if greater than 0.
// A value of 0 disable Prometheus support. Regarding Prometheus, it is considered a good
// practice to put large values here (such as a few days), and at least the interval between
// prometheus requests.
//
// hcl: telemetry { prometheus_retention_time = "duration" }
TelemetryPrometheusRetentionTime time.Duration
// TelemetryFilterDefault is the default for whether to allow a metric that's not
// covered by the filter.
//
// hcl: telemetry { filter_default = (true|false) }
TelemetryFilterDefault bool
// TelemetryAllowedPrefixes is a list of filter rules to apply for allowing metrics
// by prefix. Use the 'prefix_filter' option and prefix rules with '+' to be
// included.
//
// hcl: telemetry { prefix_filter = []string{"+<expr>", "+<expr>", ...} }
TelemetryAllowedPrefixes []string
// TelemetryBlockedPrefixes is a list of filter rules to apply for blocking metrics
// by prefix. Use the 'prefix_filter' option and prefix rules with '-' to be
// excluded.
//
// hcl: telemetry { prefix_filter = []string{"-<expr>", "-<expr>", ...} }
TelemetryBlockedPrefixes []string
// TelemetryMetricsPrefix is the prefix used to write stats values to.
// Default: "consul."
//
// hcl: telemetry { metrics_prefix = string }
TelemetryMetricsPrefix string
// TelemetryStatsdAddr is the address of a statsd instance. If provided,
// metrics will be sent to that instance.
//
// hcl: telemetry { statsd_addr = string }
TelemetryStatsdAddr string
// TelemetryStatsiteAddr is the address of a statsite instance. If provided,
// metrics will be streamed to that instance.
//
// hcl: telemetry { statsite_addr = string }
TelemetryStatsiteAddr string
// Datacenter is the datacenter this node is in. Defaults to "dc1". // Datacenter is the datacenter this node is in. Defaults to "dc1".
// //

View File

@ -19,10 +19,11 @@ import (
"time" "time"
"github.com/hashicorp/consul/agent/structs" "github.com/hashicorp/consul/agent/structs"
"github.com/hashicorp/consul/lib"
"github.com/hashicorp/consul/testutil" "github.com/hashicorp/consul/testutil"
"github.com/hashicorp/consul/types" "github.com/hashicorp/consul/types"
"github.com/pascaldekloe/goe/verify" "github.com/pascaldekloe/goe/verify"
"github.com/sergi/go-diff/diffmatchpatch" "github.com/stretchr/testify/require"
) )
type configTest struct { type configTest struct {
@ -1852,8 +1853,8 @@ func TestConfigFlagsAndEdgecases(t *testing.T) {
`}, `},
patch: func(rt *RuntimeConfig) { patch: func(rt *RuntimeConfig) {
rt.DataDir = dataDir rt.DataDir = dataDir
rt.TelemetryAllowedPrefixes = []string{"foo"} rt.Telemetry.AllowedPrefixes = []string{"foo"}
rt.TelemetryBlockedPrefixes = []string{"bar"} rt.Telemetry.BlockedPrefixes = []string{"bar"}
}, },
warns: []string{`Filter rule must begin with either '+' or '-': "nix"`}, warns: []string{`Filter rule must begin with either '+' or '-': "nix"`},
}, },
@ -3822,41 +3823,43 @@ func TestFullConfig(t *testing.T) {
}, },
}, },
}, },
SerfAdvertiseAddrLAN: tcpAddr("17.99.29.16:8301"), SerfAdvertiseAddrLAN: tcpAddr("17.99.29.16:8301"),
SerfAdvertiseAddrWAN: tcpAddr("78.63.37.19:8302"), SerfAdvertiseAddrWAN: tcpAddr("78.63.37.19:8302"),
SerfBindAddrLAN: tcpAddr("99.43.63.15:8301"), SerfBindAddrLAN: tcpAddr("99.43.63.15:8301"),
SerfBindAddrWAN: tcpAddr("67.88.33.19:8302"), SerfBindAddrWAN: tcpAddr("67.88.33.19:8302"),
SessionTTLMin: 26627 * time.Second, SessionTTLMin: 26627 * time.Second,
SkipLeaveOnInt: true, SkipLeaveOnInt: true,
StartJoinAddrsLAN: []string{"LR3hGDoG", "MwVpZ4Up"}, StartJoinAddrsLAN: []string{"LR3hGDoG", "MwVpZ4Up"},
StartJoinAddrsWAN: []string{"EbFSc3nA", "kwXTh623"}, StartJoinAddrsWAN: []string{"EbFSc3nA", "kwXTh623"},
SyslogFacility: "hHv79Uia", SyslogFacility: "hHv79Uia",
TelemetryCirconusAPIApp: "p4QOTe9j", Telemetry: lib.TelemetryConfig{
TelemetryCirconusAPIToken: "E3j35V23", CirconusAPIApp: "p4QOTe9j",
TelemetryCirconusAPIURL: "mEMjHpGg", CirconusAPIToken: "E3j35V23",
TelemetryCirconusBrokerID: "BHlxUhed", CirconusAPIURL: "mEMjHpGg",
TelemetryCirconusBrokerSelectTag: "13xy1gHm", CirconusBrokerID: "BHlxUhed",
TelemetryCirconusCheckDisplayName: "DRSlQR6n", CirconusBrokerSelectTag: "13xy1gHm",
TelemetryCirconusCheckForceMetricActivation: "Ua5FGVYf", CirconusCheckDisplayName: "DRSlQR6n",
TelemetryCirconusCheckID: "kGorutad", CirconusCheckForceMetricActivation: "Ua5FGVYf",
TelemetryCirconusCheckInstanceID: "rwoOL6R4", CirconusCheckID: "kGorutad",
TelemetryCirconusCheckSearchTag: "ovT4hT4f", CirconusCheckInstanceID: "rwoOL6R4",
TelemetryCirconusCheckTags: "prvO4uBl", CirconusCheckSearchTag: "ovT4hT4f",
TelemetryCirconusSubmissionInterval: "DolzaflP", CirconusCheckTags: "prvO4uBl",
TelemetryCirconusSubmissionURL: "gTcbS93G", CirconusSubmissionInterval: "DolzaflP",
TelemetryDisableHostname: true, CirconusSubmissionURL: "gTcbS93G",
TelemetryDogstatsdAddr: "0wSndumK", DisableHostname: true,
TelemetryDogstatsdTags: []string{"3N81zSUB", "Xtj8AnXZ"}, DogstatsdAddr: "0wSndumK",
TelemetryFilterDefault: true, DogstatsdTags: []string{"3N81zSUB", "Xtj8AnXZ"},
TelemetryAllowedPrefixes: []string{"oJotS8XJ"}, FilterDefault: true,
TelemetryBlockedPrefixes: []string{"cazlEhGn"}, AllowedPrefixes: []string{"oJotS8XJ"},
TelemetryMetricsPrefix: "ftO6DySn", BlockedPrefixes: []string{"cazlEhGn"},
TelemetryPrometheusRetentionTime: 15 * time.Second, MetricsPrefix: "ftO6DySn",
TelemetryStatsdAddr: "drce87cy", PrometheusRetentionTime: 15 * time.Second,
TelemetryStatsiteAddr: "HpFwKB8R", StatsdAddr: "drce87cy",
TLSCipherSuites: []uint16{tls.TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305, tls.TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384}, StatsiteAddr: "HpFwKB8R",
TLSMinVersion: "pAOWafkR", },
TLSPreferServerCipherSuites: true, TLSCipherSuites: []uint16{tls.TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305, tls.TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384},
TLSMinVersion: "pAOWafkR",
TLSPreferServerCipherSuites: true,
TaggedAddresses: map[string]string{ TaggedAddresses: map[string]string{
"7MYgHrYH": "dALJAhLD", "7MYgHrYH": "dALJAhLD",
"h6DdBy6K": "ebrr9zZ8", "h6DdBy6K": "ebrr9zZ8",
@ -4399,29 +4402,30 @@ func TestSanitize(t *testing.T) {
"TLSMinVersion": "", "TLSMinVersion": "",
"TLSPreferServerCipherSuites": false, "TLSPreferServerCipherSuites": false,
"TaggedAddresses": {}, "TaggedAddresses": {},
"TelemetryAllowedPrefixes": [], "Telemetry":{
"TelemetryBlockedPrefixes": [], "AllowedPrefixes": [],
"TelemetryCirconusAPIApp": "", "BlockedPrefixes": [],
"TelemetryCirconusAPIToken": "hidden", "CirconusAPIApp": "",
"TelemetryCirconusAPIURL": "", "CirconusAPIToken": "hidden",
"TelemetryCirconusBrokerID": "", "CirconusAPIURL": "",
"TelemetryCirconusBrokerSelectTag": "", "CirconusBrokerID": "",
"TelemetryCirconusCheckDisplayName": "", "CirconusBrokerSelectTag": "",
"TelemetryCirconusCheckForceMetricActivation": "", "CirconusCheckDisplayName": "",
"TelemetryCirconusCheckID": "", "CirconusCheckForceMetricActivation": "",
"TelemetryCirconusCheckInstanceID": "", "CirconusCheckID": "",
"TelemetryCirconusCheckSearchTag": "", "CirconusCheckInstanceID": "",
"TelemetryCirconusCheckTags": "", "CirconusCheckSearchTag": "",
"TelemetryCirconusSubmissionInterval": "", "CirconusCheckTags": "",
"TelemetryCirconusSubmissionURL": "", "CirconusSubmissionInterval": "",
"TelemetryDisableHostname": false, "CirconusSubmissionURL": "",
"TelemetryDogstatsdAddr": "", "DisableHostname": false,
"TelemetryDogstatsdTags": [], "DogstatsdAddr": "",
"TelemetryFilterDefault": false, "DogstatsdTags": [],
"TelemetryMetricsPrefix": "", "FilterDefault": false,
"TelemetryPrometheusRetentionTime": "0s", "MetricsPrefix": "",
"TelemetryStatsdAddr": "", "PrometheusRetentionTime": "0s",
"TelemetryStatsiteAddr": "", "StatsdAddr": ""
},
"TranslateWANAddrs": false, "TranslateWANAddrs": false,
"UIDir": "", "UIDir": "",
"UnixSocketGroup": "", "UnixSocketGroup": "",
@ -4441,11 +4445,7 @@ func TestSanitize(t *testing.T) {
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
if got, want := string(b), rtJSON; got != want { require.JSONEq(t, rtJSON, string(b))
dmp := diffmatchpatch.New()
diffs := dmp.DiffMain(want, got, false)
t.Fatal(dmp.DiffPrettyText(diffs))
}
} }
func splitIPPort(hostport string) (net.IP, int) { func splitIPPort(hostport string) (net.IP, int) {

View File

@ -198,7 +198,7 @@ func (c *cmd) run(args []string) int {
c.logOutput = logOutput c.logOutput = logOutput
c.logger = log.New(logOutput, "", log.LstdFlags) c.logger = log.New(logOutput, "", log.LstdFlags)
memSink, err := lib.InitTelemetry(config.TelemetryConfig(false)) memSink, err := lib.InitTelemetry(config.Telemetry)
if err != nil { if err != nil {
c.UI.Error(err.Error()) c.UI.Error(err.Error())
return 1 return 1

View File

@ -9,6 +9,7 @@ import (
"github.com/hashicorp/consul/api" "github.com/hashicorp/consul/api"
"github.com/hashicorp/consul/connect" "github.com/hashicorp/consul/connect"
"github.com/hashicorp/consul/lib"
"github.com/hashicorp/consul/watch" "github.com/hashicorp/consul/watch"
"github.com/hashicorp/hcl" "github.com/hashicorp/hcl"
) )
@ -40,7 +41,7 @@ type Config struct {
// Telemetry stores configuration for go-metrics. It is typically populated // Telemetry stores configuration for go-metrics. It is typically populated
// from the agent's runtime config via the proxy config endpoint so that the // from the agent's runtime config via the proxy config endpoint so that the
// proxy will log metrics to the same location(s) as the agent. // proxy will log metrics to the same location(s) as the agent.
Telemetry map[string]interface{} Telemetry lib.TelemetryConfig
} }
// Service returns the *connect.Service structure represented by this config. // Service returns the *connect.Service structure represented by this config.
@ -265,8 +266,11 @@ func (w *AgentConfigWatcher) handler(blockVal watch.BlockingParamVal,
ProxiedServiceNamespace: "default", ProxiedServiceNamespace: "default",
} }
if t, ok := resp.Config["telemetry"].(map[string]interface{}); ok { if tRaw, ok := resp.Config["telemetry"]; ok {
cfg.Telemetry = t err := mapstructure.Decode(tRaw, &cfg.Telemetry)
if err != nil {
w.logger.Printf("[WARN] proxy telemetry config failed to parse: %s", err)
}
} }
// Unmarshal configs // Unmarshal configs

View File

@ -11,6 +11,7 @@ import (
"github.com/hashicorp/consul/agent" "github.com/hashicorp/consul/agent"
"github.com/hashicorp/consul/api" "github.com/hashicorp/consul/api"
"github.com/hashicorp/consul/connect" "github.com/hashicorp/consul/connect"
"github.com/hashicorp/consul/lib"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
) )
@ -118,7 +119,14 @@ func TestUpstreamResolverFromClient(t *testing.T) {
func TestAgentConfigWatcher(t *testing.T) { func TestAgentConfigWatcher(t *testing.T) {
t.Parallel() t.Parallel()
a := agent.NewTestAgent("agent_smith", "") a := agent.NewTestAgent("agent_smith", `
connect {
enabled = true
proxy {
allow_managed_api_registration = true
}
}
`)
defer a.Shutdown() defer a.Shutdown()
client := a.Client() client := a.Client()
@ -175,9 +183,9 @@ func TestAgentConfigWatcher(t *testing.T) {
ConnectTimeoutMs: 10000, // from applyDefaults ConnectTimeoutMs: 10000, // from applyDefaults
}, },
}, },
Telemetry: map[string]interface{}{ Telemetry: lib.TelemetryConfig{
"FilterDefault": true, FilterDefault: true,
"MetricsPrefix": "consul.proxy.web", MetricsPrefix: "consul.proxy.web",
}, },
} }

View File

@ -46,12 +46,9 @@ func (p *Proxy) Serve() error {
// Initial setup // Initial setup
// Setup telemetry if configured // Setup telemetry if configured
if len(newCfg.Telemetry) > 0 { _, err := lib.InitTelemetry(newCfg.Telemetry)
p.logger.Printf("[DEBUG] got Telemetry confg: %v", newCfg.Telemetry) if err != nil {
_, err := lib.InitTelemetry(newCfg.Telemetry) p.logger.Printf("[ERR] proxy telemetry config error: %s", err)
if err != nil {
p.logger.Printf("[ERR] proxy telemetry config error: %s", err)
}
} }
// Setup Service instance now we know target ID etc // Setup Service instance now we know target ID etc

View File

@ -1,6 +1,7 @@
package lib package lib
import ( import (
"reflect"
"time" "time"
metrics "github.com/armon/go-metrics" metrics "github.com/armon/go-metrics"
@ -9,24 +10,252 @@ import (
"github.com/armon/go-metrics/prometheus" "github.com/armon/go-metrics/prometheus"
) )
func statsiteSink(cfg map[string]interface{}, hostname string) (metrics.MetricSink, error) { // TelemetryConfig is embedded in config.RuntimeConfig and holds the
addr := cfgStringVal(cfg["StatsiteAddr"]) // configuration variables for go-metrics. It is a separate struct to allow it
// to be exported as JSON and passed to other process like managed connect
// proxies so they can inherit the agent's telemetry config.
//
// It is in lib package rather than agent/config because we need to use it in
// the shared InitTelemetry functions below, but we can't import agent/config
// due to a dependency cycle.
type TelemetryConfig struct {
// Circonus*: see https://github.com/circonus-labs/circonus-gometrics
// for more details on the various configuration options.
// Valid configuration combinations:
// - CirconusAPIToken
// metric management enabled (search for existing check or create a new one)
// - CirconusSubmissionUrl
// metric management disabled (use check with specified submission_url,
// broker must be using a public SSL certificate)
// - CirconusAPIToken + CirconusCheckSubmissionURL
// metric management enabled (use check with specified submission_url)
// - CirconusAPIToken + CirconusCheckID
// metric management enabled (use check with specified id)
// CirconusAPIApp is an app name associated with API token.
// Default: "consul"
//
// hcl: telemetry { circonus_api_app = string }
CirconusAPIApp string `json:"circonus_api_app,omitempty" mapstructure:"circonus_api_app"`
// CirconusAPIToken is a valid API Token used to create/manage check. If provided,
// metric management is enabled.
// Default: none
//
// hcl: telemetry { circonus_api_token = string }
CirconusAPIToken string `json:"circonus_api_token,omitempty" mapstructure:"circonus_api_token"`
// CirconusAPIURL is the base URL to use for contacting the Circonus API.
// Default: "https://api.circonus.com/v2"
//
// hcl: telemetry { circonus_api_url = string }
CirconusAPIURL string `json:"circonus_apiurl,omitempty" mapstructure:"circonus_apiurl"`
// CirconusBrokerID is an explicit broker to use when creating a new check. The numeric portion
// of broker._cid. If metric management is enabled and neither a Submission URL nor Check ID
// is provided, an attempt will be made to search for an existing check using Instance ID and
// Search Tag. If one is not found, a new HTTPTRAP check will be created.
// Default: use Select Tag if provided, otherwise, a random Enterprise Broker associated
// with the specified API token or the default Circonus Broker.
// Default: none
//
// hcl: telemetry { circonus_broker_id = string }
CirconusBrokerID string `json:"circonus_broker_id,omitempty" mapstructure:"circonus_broker_id"`
// CirconusBrokerSelectTag is a special tag which will be used to select a broker when
// a Broker ID is not provided. The best use of this is to as a hint for which broker
// should be used based on *where* this particular instance is running.
// (e.g. a specific geo location or datacenter, dc:sfo)
// Default: none
//
// hcl: telemetry { circonus_broker_select_tag = string }
CirconusBrokerSelectTag string `json:"circonus_broker_select_tag,omitempty" mapstructure:"circonus_broker_select_tag"`
// CirconusCheckDisplayName is the name for the check which will be displayed in the Circonus UI.
// Default: value of CirconusCheckInstanceID
//
// hcl: telemetry { circonus_check_display_name = string }
CirconusCheckDisplayName string `json:"circonus_check_display_name,omitempty" mapstructure:"circonus_check_display_name"`
// CirconusCheckForceMetricActivation will force enabling metrics, as they are encountered,
// if the metric already exists and is NOT active. If check management is enabled, the default
// behavior is to add new metrics as they are encountered. If the metric already exists in the
// check, it will *NOT* be activated. This setting overrides that behavior.
// Default: "false"
//
// hcl: telemetry { circonus_check_metrics_activation = (true|false)
CirconusCheckForceMetricActivation string `json:"circonus_check_force_metric_activation,omitempty" mapstructure:"circonus_check_force_metric_activation"`
// CirconusCheckID is the check id (not check bundle id) from a previously created
// HTTPTRAP check. The numeric portion of the check._cid field.
// Default: none
//
// hcl: telemetry { circonus_check_id = string }
CirconusCheckID string `json:"circonus_check_id,omitempty" mapstructure:"circonus_check_id"`
// CirconusCheckInstanceID serves to uniquely identify the metrics coming from this "instance".
// It can be used to maintain metric continuity with transient or ephemeral instances as
// they move around within an infrastructure.
// Default: hostname:app
//
// hcl: telemetry { circonus_check_instance_id = string }
CirconusCheckInstanceID string `json:"circonus_check_instance_id,omitempty" mapstructure:"circonus_check_instance_id"`
// CirconusCheckSearchTag is a special tag which, when coupled with the instance id, helps to
// narrow down the search results when neither a Submission URL or Check ID is provided.
// Default: service:app (e.g. service:consul)
//
// hcl: telemetry { circonus_check_search_tag = string }
CirconusCheckSearchTag string `json:"circonus_check_search_tag,omitempty" mapstructure:"circonus_check_search_tag"`
// CirconusCheckSearchTag is a special tag which, when coupled with the instance id, helps to
// narrow down the search results when neither a Submission URL or Check ID is provided.
// Default: service:app (e.g. service:consul)
//
// hcl: telemetry { circonus_check_tags = string }
CirconusCheckTags string `json:"circonus_check_tags,omitempty" mapstructure:"circonus_check_tags"`
// CirconusSubmissionInterval is the interval at which metrics are submitted to Circonus.
// Default: 10s
//
// hcl: telemetry { circonus_submission_interval = "duration" }
CirconusSubmissionInterval string `json:"circonus_submission_interval,omitempty" mapstructure:"circonus_submission_interval"`
// CirconusCheckSubmissionURL is the check.config.submission_url field from a
// previously created HTTPTRAP check.
// Default: none
//
// hcl: telemetry { circonus_submission_url = string }
CirconusSubmissionURL string `json:"circonus_submission_url,omitempty" mapstructure:"circonus_submission_url"`
// DisableHostname will disable hostname prefixing for all metrics.
//
// hcl: telemetry { disable_hostname = (true|false)
DisableHostname bool `json:"disable_hostname,omitempty" mapstructure:"disable_hostname"`
// DogStatsdAddr is the address of a dogstatsd instance. If provided,
// metrics will be sent to that instance
//
// hcl: telemetry { dogstatsd_addr = string }
DogstatsdAddr string `json:"dogstatsd_addr,omitempty" mapstructure:"dogstatsd_addr"`
// DogStatsdTags are the global tags that should be sent with each packet to dogstatsd
// It is a list of strings, where each string looks like "my_tag_name:my_tag_value"
//
// hcl: telemetry { dogstatsd_tags = []string }
DogstatsdTags []string `json:"dogstatsd_tags,omitempty" mapstructure:"dogstatsd_tags"`
// PrometheusRetentionTime is the retention time for prometheus metrics if greater than 0.
// A value of 0 disable Prometheus support. Regarding Prometheus, it is considered a good
// practice to put large values here (such as a few days), and at least the interval between
// prometheus requests.
//
// hcl: telemetry { prometheus_retention_time = "duration" }
PrometheusRetentionTime time.Duration `json:"prometheus_retention_time,omitempty" mapstructure:"prometheus_retention_time"`
// FilterDefault is the default for whether to allow a metric that's not
// covered by the filter.
//
// hcl: telemetry { filter_default = (true|false) }
FilterDefault bool `json:"filter_default,omitempty" mapstructure:"filter_default"`
// AllowedPrefixes is a list of filter rules to apply for allowing metrics
// by prefix. Use the 'prefix_filter' option and prefix rules with '+' to be
// included.
//
// hcl: telemetry { prefix_filter = []string{"+<expr>", "+<expr>", ...} }
AllowedPrefixes []string `json:"allowed_prefixes,omitempty" mapstructure:"allowed_prefixes"`
// BlockedPrefixes is a list of filter rules to apply for blocking metrics
// by prefix. Use the 'prefix_filter' option and prefix rules with '-' to be
// excluded.
//
// hcl: telemetry { prefix_filter = []string{"-<expr>", "-<expr>", ...} }
BlockedPrefixes []string `json:"blocked_prefixes,omitempty" mapstructure:"blocked_prefixes"`
// MetricsPrefix is the prefix used to write stats values to.
// Default: "consul."
//
// hcl: telemetry { metrics_prefix = string }
MetricsPrefix string `json:"metrics_prefix,omitempty" mapstructure:"metrics_prefix"`
// StatsdAddr is the address of a statsd instance. If provided,
// metrics will be sent to that instance.
//
// hcl: telemetry { statsd_address = string }
StatsdAddr string `json:"statsd_address,omitempty" mapstructure:"statsd_address"`
// StatsiteAddr is the address of a statsite instance. If provided,
// metrics will be streamed to that instance.
//
// hcl: telemetry { statsite_address = string }
StatsiteAddr string `json:"statsite_address,omitempty" mapstructure:"statsite_address"`
}
// MergeDefaults copies any non-zero field from defaults into the current
// config.
func (c *TelemetryConfig) MergeDefaults(defaults *TelemetryConfig) {
if defaults == nil {
return
}
cfgPtrVal := reflect.ValueOf(c)
cfgVal := cfgPtrVal.Elem()
otherVal := reflect.ValueOf(*defaults)
for i := 0; i < cfgVal.NumField(); i++ {
f := cfgVal.Field(i)
if !f.IsValid() || !f.CanSet() {
continue
}
// See if the current value is a zero-value, if _not_ skip it
//
// No built in way to check for zero-values for all types so only
// implementing this for the types we actually have for now. Test failure
// should catch the case where we add new types later.
switch f.Kind() {
case reflect.Slice:
if !f.IsNil() {
continue
}
case reflect.Int, reflect.Int64: // time.Duration == int64
if f.Int() != 0 {
continue
}
case reflect.String:
if f.String() != "" {
continue
}
case reflect.Bool:
if f.Bool() != false {
continue
}
default:
// Needs implementing, should be caught by tests.
continue
}
// It's zero, copy it from defaults
f.Set(otherVal.Field(i))
}
}
func statsiteSink(cfg TelemetryConfig, hostname string) (metrics.MetricSink, error) {
addr := cfg.StatsiteAddr
if addr == "" { if addr == "" {
return nil, nil return nil, nil
} }
return metrics.NewStatsiteSink(addr) return metrics.NewStatsiteSink(addr)
} }
func statsdSink(cfg map[string]interface{}, hostname string) (metrics.MetricSink, error) { func statsdSink(cfg TelemetryConfig, hostname string) (metrics.MetricSink, error) {
addr := cfgStringVal(cfg["StatsdAddr"]) addr := cfg.StatsdAddr
if addr == "" { if addr == "" {
return nil, nil return nil, nil
} }
return metrics.NewStatsdSink(addr) return metrics.NewStatsdSink(addr)
} }
func dogstatdSink(cfg map[string]interface{}, hostname string) (metrics.MetricSink, error) { func dogstatdSink(cfg TelemetryConfig, hostname string) (metrics.MetricSink, error) {
addr := cfgStringVal(cfg["DogstatsdAddr"]) addr := cfg.DogstatsdAddr
if addr == "" { if addr == "" {
return nil, nil return nil, nil
} }
@ -34,16 +263,16 @@ func dogstatdSink(cfg map[string]interface{}, hostname string) (metrics.MetricSi
if err != nil { if err != nil {
return nil, err return nil, err
} }
sink.SetTags(cfgStrSliceVal(cfg["DogstatsdTags"])) sink.SetTags(cfg.DogstatsdTags)
return sink, nil return sink, nil
} }
func prometheusSink(cfg map[string]interface{}, hostname string) (metrics.MetricSink, error) { func prometheusSink(cfg TelemetryConfig, hostname string) (metrics.MetricSink, error) {
if cfgDurationVal(cfg["PrometheusRetentionTime"]).Nanoseconds() < 1 { if cfg.PrometheusRetentionTime.Nanoseconds() < 1 {
return nil, nil return nil, nil
} }
prometheusOpts := prometheus.PrometheusOpts{ prometheusOpts := prometheus.PrometheusOpts{
Expiration: cfgDurationVal(cfg["PrometheusRetentionTime"]), Expiration: cfg.PrometheusRetentionTime,
} }
sink, err := prometheus.NewPrometheusSinkFrom(prometheusOpts) sink, err := prometheus.NewPrometheusSinkFrom(prometheusOpts)
if err != nil { if err != nil {
@ -52,27 +281,27 @@ func prometheusSink(cfg map[string]interface{}, hostname string) (metrics.Metric
return sink, nil return sink, nil
} }
func circonusSink(cfg map[string]interface{}, hostname string) (metrics.MetricSink, error) { func circonusSink(cfg TelemetryConfig, hostname string) (metrics.MetricSink, error) {
token := cfgStringVal(cfg["CirconusAPIToken"]) token := cfg.CirconusAPIToken
url := cfgStringVal(cfg["CirconusSubmissionURL"]) url := cfg.CirconusSubmissionURL
if token == "" && url == "" { if token == "" && url == "" {
return nil, nil return nil, nil
} }
conf := &circonus.Config{} conf := &circonus.Config{}
conf.Interval = cfgStringVal(cfg["CirconusSubmissionInterval"]) conf.Interval = cfg.CirconusSubmissionInterval
conf.CheckManager.API.TokenKey = token conf.CheckManager.API.TokenKey = token
conf.CheckManager.API.TokenApp = cfgStringVal(cfg["CirconusAPIApp"]) conf.CheckManager.API.TokenApp = cfg.CirconusAPIApp
conf.CheckManager.API.URL = cfgStringVal(cfg["CirconusAPIURL"]) conf.CheckManager.API.URL = cfg.CirconusAPIURL
conf.CheckManager.Check.SubmissionURL = url conf.CheckManager.Check.SubmissionURL = url
conf.CheckManager.Check.ID = cfgStringVal(cfg["CirconusCheckID"]) conf.CheckManager.Check.ID = cfg.CirconusCheckID
conf.CheckManager.Check.ForceMetricActivation = cfgStringVal(cfg["CirconusCheckForceMetricActivation"]) conf.CheckManager.Check.ForceMetricActivation = cfg.CirconusCheckForceMetricActivation
conf.CheckManager.Check.InstanceID = cfgStringVal(cfg["CirconusCheckInstanceID"]) conf.CheckManager.Check.InstanceID = cfg.CirconusCheckInstanceID
conf.CheckManager.Check.SearchTag = cfgStringVal(cfg["CirconusCheckSearchTag"]) conf.CheckManager.Check.SearchTag = cfg.CirconusCheckSearchTag
conf.CheckManager.Check.DisplayName = cfgStringVal(cfg["CirconusCheckDisplayName"]) conf.CheckManager.Check.DisplayName = cfg.CirconusCheckDisplayName
conf.CheckManager.Check.Tags = cfgStringVal(cfg["CirconusCheckTags"]) conf.CheckManager.Check.Tags = cfg.CirconusCheckTags
conf.CheckManager.Broker.ID = cfgStringVal(cfg["CirconusBrokerID"]) conf.CheckManager.Broker.ID = cfg.CirconusBrokerID
conf.CheckManager.Broker.SelectTag = cfgStringVal(cfg["CirconusBrokerSelectTag"]) conf.CheckManager.Broker.SelectTag = cfg.CirconusBrokerSelectTag
if conf.CheckManager.Check.DisplayName == "" { if conf.CheckManager.Check.DisplayName == "" {
conf.CheckManager.Check.DisplayName = "Consul" conf.CheckManager.Check.DisplayName = "Consul"
@ -94,51 +323,22 @@ func circonusSink(cfg map[string]interface{}, hostname string) (metrics.MetricSi
return sink, nil return sink, nil
} }
func cfgStringVal(i interface{}) string {
v, ok := i.(string)
if ok {
return v
}
return ""
}
func cfgBoolVal(i interface{}) bool {
v, ok := i.(bool)
if ok {
return v
}
return false
}
func cfgDurationVal(i interface{}) time.Duration {
v, ok := i.(time.Duration)
if ok {
return v
}
return time.Duration(0)
}
func cfgStrSliceVal(i interface{}) []string {
v, ok := i.([]string)
if ok {
return v
}
return nil
}
// InitTelemetry configures go-metrics based on map of telemetry config // InitTelemetry configures go-metrics based on map of telemetry config
// values as returned by RuntimecfgStringVal(cfg["Config"])(). // values as returned by Runtimecfg.Config().
func InitTelemetry(cfg map[string]interface{}) (*metrics.InmemSink, error) { func InitTelemetry(cfg TelemetryConfig) (*metrics.InmemSink, error) {
// Setup telemetry // Setup telemetry
// Aggregate on 10 second intervals for 1 minute. Expose the // Aggregate on 10 second intervals for 1 minute. Expose the
// metrics over stderr when there is a SIGUSR1 received. // metrics over stderr when there is a SIGUSR1 received.
memSink := metrics.NewInmemSink(10*time.Second, time.Minute) memSink := metrics.NewInmemSink(10*time.Second, time.Minute)
metrics.DefaultInmemSignal(memSink) metrics.DefaultInmemSignal(memSink)
metricsConf := metrics.DefaultConfig(cfgStringVal(cfg["MetricsPrefix"])) metricsConf := metrics.DefaultConfig(cfg.MetricsPrefix)
metricsConf.EnableHostname = !cfgBoolVal(cfg["DisableHostname"]) metricsConf.EnableHostname = !cfg.DisableHostname
metricsConf.FilterDefault = cfgBoolVal(cfg["FilterDefault"]) metricsConf.FilterDefault = cfg.FilterDefault
metricsConf.AllowedPrefixes = cfgStrSliceVal(cfg["AllowedPrefixes"]) metricsConf.AllowedPrefixes = cfg.AllowedPrefixes
metricsConf.BlockedPrefixes = cfgStrSliceVal(cfg["BlockedPrefixes"]) metricsConf.BlockedPrefixes = cfg.BlockedPrefixes
var sinks metrics.FanoutSink var sinks metrics.FanoutSink
addSink := func(name string, fn func(map[string]interface{}, string) (metrics.MetricSink, error)) error { addSink := func(name string, fn func(TelemetryConfig, string) (metrics.MetricSink, error)) error {
s, err := fn(cfg, metricsConf.HostName) s, err := fn(cfg, metricsConf.HostName)
if err != nil { if err != nil {
return err return err

90
lib/telemetry_test.go Normal file
View File

@ -0,0 +1,90 @@
package lib
import (
"reflect"
"testing"
"time"
"github.com/stretchr/testify/require"
)
func makeFullTelemetryConfig(t *testing.T) TelemetryConfig {
var (
strSliceVal = []string{"foo"}
strVal = "foo"
intVal = int64(1 * time.Second)
)
cfg := TelemetryConfig{}
cfgP := reflect.ValueOf(&cfg)
cfgV := cfgP.Elem()
for i := 0; i < cfgV.NumField(); i++ {
f := cfgV.Field(i)
if !f.IsValid() || !f.CanSet() {
continue
}
// Set non-zero values for all fields. We only implement kinds that exist
// now for brevity but will fail the test if a new field type is added since
// this is likely not implemented in MergeDefaults either.
switch f.Kind() {
case reflect.Slice:
if f.Type() != reflect.TypeOf(strSliceVal) {
t.Fatalf("unknown slice type in TelemetryConfig." +
" You need to update MergeDefaults and this test code.")
}
f.Set(reflect.ValueOf(strSliceVal))
case reflect.Int, reflect.Int64: // time.Duration == int64
f.SetInt(intVal)
case reflect.String:
f.SetString(strVal)
case reflect.Bool:
f.SetBool(true)
default:
t.Fatalf("unknown field type in TelemetryConfig" +
" You need to update MergeDefaults and this test code.")
}
}
return cfg
}
func TestTelemetryConfig_MergeDefaults(t *testing.T) {
tests := []struct {
name string
cfg TelemetryConfig
defaults TelemetryConfig
want TelemetryConfig
}{
{
name: "basic merge",
cfg: TelemetryConfig{
StatsiteAddr: "stats.it:4321",
},
defaults: TelemetryConfig{
StatsdAddr: "localhost:5678",
StatsiteAddr: "localhost:1234",
},
want: TelemetryConfig{
StatsdAddr: "localhost:5678",
StatsiteAddr: "stats.it:4321",
},
},
{
// This test uses reflect to build a TelemetryConfig with every value set
// to ensure that we exercise every possible field type. This means that
// if new fields are added that are not supported types in the code, this
// test should either ensure they work or fail to build the test case and
// fail the test.
name: "exhaustive",
cfg: TelemetryConfig{},
defaults: makeFullTelemetryConfig(t),
want: makeFullTelemetryConfig(t),
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
c := tt.cfg
c.MergeDefaults(&tt.defaults)
require.Equal(t, tt.want, c)
})
}
}