diff --git a/CHANGELOG.next.asciidoc b/CHANGELOG.next.asciidoc index 4a71215602f..ad34eda6965 100644 --- a/CHANGELOG.next.asciidoc +++ b/CHANGELOG.next.asciidoc @@ -122,6 +122,7 @@ https://github.com/elastic/beats/compare/v7.0.0-alpha2...master[Check the HEAD d TLS or Beats that accept connections over TLS and validate client certificates. {pull}14146[14146] - Support usage of custom builders without hints and mappers {pull}13839[13839] - Fix kubernetes `metaGenerator.ResourceMetadata` when parent reference controller is nil {issue}14320[14320] {pull}14329[14329] +- Allow users to configure only `cluster_uuid` setting under `monitoring` namespace. {pull}14338[14338] *Auditbeat* diff --git a/libbeat/cmd/instance/beat.go b/libbeat/cmd/instance/beat.go index ba799d9ab7d..d2bfd65f0dd 100644 --- a/libbeat/cmd/instance/beat.go +++ b/libbeat/cmd/instance/beat.go @@ -411,30 +411,12 @@ func (b *Beat) launch(settings Settings, bt beat.Creator) error { return err } - monitoringCfg, reporterSettings, err := monitoring.SelectConfig(b.Config.MonitoringBeatConfig) + r, err := b.setupMonitoring(settings) if err != nil { return err } - - if monitoringCfg.Enabled() { - settings := report.Settings{ - DefaultUsername: settings.Monitoring.DefaultUsername, - Format: reporterSettings.Format, - ClusterUUID: reporterSettings.ClusterUUID, - } - reporter, err := report.New(b.Info, settings, monitoringCfg, b.Config.Output) - if err != nil { - return err - } - defer reporter.Stop() - - // Expose monitoring.cluster_uuid in state API - if reporterSettings.ClusterUUID != "" { - stateRegistry := monitoring.GetNamespace("state").GetRegistry() - monitoringRegistry := stateRegistry.NewRegistry("monitoring") - clusterUUIDRegVar := monitoring.NewString(monitoringRegistry, "cluster_uuid") - clusterUUIDRegVar.Set(reporterSettings.ClusterUUID) - } + if r != nil { + defer r.Stop() } if b.Config.MetricLogging == nil || b.Config.MetricLogging.Enabled() { @@ -896,6 +878,41 @@ func (b *Beat) clusterUUIDFetchingCallback() (elasticsearch.ConnectCallback, err return callback, nil } +func (b *Beat) setupMonitoring(settings Settings) (report.Reporter, error) { + monitoringCfg, reporterSettings, err := monitoring.SelectConfig(b.Config.MonitoringBeatConfig) + if err != nil { + return nil, err + } + + monitoringClusterUUID, err := monitoring.GetClusterUUID(monitoringCfg) + if err != nil { + return nil, err + } + + // Expose monitoring.cluster_uuid in state API + if monitoringClusterUUID != "" { + stateRegistry := monitoring.GetNamespace("state").GetRegistry() + monitoringRegistry := stateRegistry.NewRegistry("monitoring") + clusterUUIDRegVar := monitoring.NewString(monitoringRegistry, "cluster_uuid") + clusterUUIDRegVar.Set(monitoringClusterUUID) + } + + if monitoring.IsEnabled(monitoringCfg) { + settings := report.Settings{ + DefaultUsername: settings.Monitoring.DefaultUsername, + Format: reporterSettings.Format, + ClusterUUID: monitoringClusterUUID, + } + reporter, err := report.New(b.Info, settings, monitoringCfg, b.Config.Output) + if err != nil { + return nil, err + } + return reporter, nil + } + + return nil, nil +} + // handleError handles the given error by logging it and then returning the // error. If the err is nil or is a GracefulExit error then the method will // return nil without logging anything. diff --git a/libbeat/monitoring/monitoring.go b/libbeat/monitoring/monitoring.go index e7f13fb69ea..ffd21cb118b 100644 --- a/libbeat/monitoring/monitoring.go +++ b/libbeat/monitoring/monitoring.go @@ -97,14 +97,40 @@ func SelectConfig(beatCfg BeatConfig) (*common.Config, *report.Settings, error) return monitoringCfg, &report.Settings{Format: report.FormatXPackMonitoringBulk}, nil case beatCfg.Monitoring.Enabled(): monitoringCfg := beatCfg.Monitoring - var info struct { - ClusterUUID string `config:"cluster_uuid"` - } - if err := monitoringCfg.Unpack(&info); err != nil { - return nil, nil, err - } - return monitoringCfg, &report.Settings{Format: report.FormatBulk, ClusterUUID: info.ClusterUUID}, nil + return monitoringCfg, &report.Settings{Format: report.FormatBulk}, nil default: return nil, nil, nil } } + +// GetClusterUUID returns the value of the monitoring.cluster_uuid setting, if it is set. +func GetClusterUUID(monitoringCfg *common.Config) (string, error) { + if monitoringCfg == nil { + return "", nil + } + + var config struct { + ClusterUUID string `config:"cluster_uuid"` + } + if err := monitoringCfg.Unpack(&config); err != nil { + return "", err + } + + return config.ClusterUUID, nil +} + +// IsEnabled returns whether the monitoring reporter is enabled or not. +func IsEnabled(monitoringCfg *common.Config) bool { + if monitoringCfg == nil { + return false + } + + // If the only setting in the monitoring config is cluster_uuid, it is + // not enabled + fields := monitoringCfg.GetFields() + if len(fields) == 1 && fields[0] == "cluster_uuid" { + return false + } + + return monitoringCfg.Enabled() +} diff --git a/libbeat/tests/system/config/mockbeat.yml.j2 b/libbeat/tests/system/config/mockbeat.yml.j2 index 6023545f660..cd3fe5f7dad 100644 --- a/libbeat/tests/system/config/mockbeat.yml.j2 +++ b/libbeat/tests/system/config/mockbeat.yml.j2 @@ -116,10 +116,20 @@ xpack.monitoring.elasticsearch.state.period: 3s # to speed up tests {% if monitoring -%} #================================ X-Pack Monitoring (direct) ===================================== -monitoring.elasticsearch.hosts: {{monitoring.elasticsearch.hosts}} -monitoring.elasticsearch.metrics.period: 2s # to speed up tests -monitoring.elasticsearch.state.period: 3s # to speed up tests +monitoring: + {% if monitoring.elasticsearch -%} + elasticsearch.hosts: {{monitoring.elasticsearch.hosts}} + elasticsearch.metrics.period: 2s # to speed up tests + elasticsearch.state.period: 3s # to speed up tests + {% endif -%} + + {% if monitoring.cluster_uuid -%} + cluster_uuid: {{monitoring.cluster_uuid}} + {% endif -%} {% endif -%} # vim: set ft=jinja: +{% if http_enabled -%} +http.enabled: {{http_enabled}} +{% endif -%} diff --git a/libbeat/tests/system/test_monitoring.py b/libbeat/tests/system/test_monitoring.py index 833cb5cf02e..73767e24c7c 100644 --- a/libbeat/tests/system/test_monitoring.py +++ b/libbeat/tests/system/test_monitoring.py @@ -4,6 +4,9 @@ import re from nose.plugins.attrib import attr import unittest +import requests +import random +import string INTEGRATION_TESTS = os.environ.get('INTEGRATION_TESTS', False) @@ -150,6 +153,29 @@ def test_compare(self): self.assert_same_structure(indirect_beats_state_doc['beats_state'], direct_beats_state_doc['beats_state']) self.assert_same_structure(indirect_beats_stats_doc['beats_stats'], direct_beats_stats_doc['beats_stats']) + @unittest.skipUnless(INTEGRATION_TESTS, "integration test") + @attr('integration') + def test_cluster_uuid_setting(self): + """ + Test that monitoring.cluster_uuid setting may be set without any other monitoring.* settings + """ + test_cluster_uuid = self.random_string(10) + self.render_config_template( + "mockbeat", + monitoring={ + "cluster_uuid": test_cluster_uuid + }, + http_enabled="true" + ) + + proc = self.start_beat(config="mockbeat.yml") + self.wait_until(lambda: self.log_contains("mockbeat start running.")) + + state = self.get_beat_state() + proc.check_kill_and_wait() + + self.assertEqual(test_cluster_uuid, state["monitoring"]["cluster_uuid"]) + def search_monitoring_doc(self, monitoring_type): results = self.es_monitoring.search( index='.monitoring-beats-*', @@ -241,3 +267,11 @@ def get_elasticsearch_monitoring_url(self): host=os.getenv("ES_MONITORING_HOST", "localhost"), port=os.getenv("ES_MONITORING_PORT", "9210") ) + + def get_beat_state(self): + url = "http://localhost:5066/state" + return requests.get(url).json() + + def random_string(self, size): + char_pool = string.ascii_letters + string.digits + return ''.join(random.choice(char_pool) for i in range(size))