Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add stop functionality for output config changes #34049

Closed
Closed
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 4 additions & 2 deletions x-pack/libbeat/management/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -11,8 +11,9 @@ import (

// Config for central management
type Config struct {
Enabled bool `config:"enabled" yaml:"enabled"`
Blacklist ConfigBlacklistSettings `config:"blacklist" yaml:"blacklist"`
Enabled bool `config:"enabled" yaml:"enabled"`
Blacklist ConfigBlacklistSettings `config:"blacklist" yaml:"blacklist"`
OutputRestart bool `config:"output_restart" yaml:"output_restart"`
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
OutputRestart bool `config:"output_restart" yaml:"output_restart"`
RestartOnOutputChange bool `config:"restart_on_output_change" yaml:"output_restart"`

nit: Let's use the exact same name that was used in the 8.5 spec files so it is clear we just changed the implementation when comparing the two: https://github.com/elastic/elastic-agent/blob/f3d39629cc740ce7cf868a228da9e2dec1a83dec/internal/spec/filebeat.yml#L12

}

// ConfigBlock stores a piece of config from central management
Expand All @@ -29,6 +30,7 @@ type ConfigBlocksWithType struct {
// ConfigBlocks holds a list of type + configs objects
type ConfigBlocks []ConfigBlocksWithType

// DefaultConfig returns the default config for the V2 manager
func DefaultConfig() *Config {
return &Config{
Blacklist: ConfigBlacklistSettings{
Expand Down
36 changes: 28 additions & 8 deletions x-pack/libbeat/management/managerV2.go
Original file line number Diff line number Diff line change
Expand Up @@ -41,15 +41,19 @@ type BeatV2Manager struct {
payload map[string]interface{}

// stop callback must be registered by libbeat, as with the V1 callback
stopFunc func()
stopMut sync.Mutex
beatStop sync.Once
stopFunc func()
stopOnOutputReload bool
stopMut sync.Mutex
beatStop sync.Once

// sync channel for shutting down the manager after we get a stop from
// either the agent or the beat
stopChan chan struct{}

isRunning bool
// is set on first instance of a config reload,
// allowing us to restart the beat if stopOnOutputReload is set
outputIsConfigured bool

// used for the debug callback to report as-running config
lastOutputCfg *reload.ConfigWithMeta
Expand Down Expand Up @@ -87,12 +91,16 @@ func NewV2AgentManager(config *conf.C, registry *reload.Registry, _ uuid.UUID) (
// NewV2AgentManagerWithClient actually creates the manager instance used by the rest of the beats.
func NewV2AgentManagerWithClient(config *Config, registry *reload.Registry, agentClient client.V2) (lbmanagement.Manager, error) {
log := logp.NewLogger(lbmanagement.DebugK)
if config.OutputRestart {
log.Infof("Output reload is enabled, the beat will restart as needed on change of output config")
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: No need to use Infof here, just Info as you're not formatting anything.

Suggested change
log.Infof("Output reload is enabled, the beat will restart as needed on change of output config")
log.Info("Output reload is enabled, the beat will restart as needed on change of output config")

}
m := &BeatV2Manager{
config: config,
logger: log.Named("V2-manager"),
registry: registry,
units: make(map[string]*client.Unit),
stopChan: make(chan struct{}, 1),
stopOnOutputReload: config.OutputRestart,
config: config,
logger: log.Named("V2-manager"),
registry: registry,
units: make(map[string]*client.Unit),
stopChan: make(chan struct{}, 1),
}

if config.Enabled {
Expand Down Expand Up @@ -338,6 +346,16 @@ func (cm *BeatV2Manager) handleOutputReload(unit *client.Unit) {
}
cm.logger.Debugf("Got Output unit config '%s'", rawConfig.GetId())

// if needed, stop the beat, let agent restart
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We are sure this only triggers when the actual output configuration changes? And it won't restart for changes in the unit that are not the actual output configuration, like the log level? https://github.com/elastic/elastic-agent-client/blob/15881a8e64ef95f32f688c933319cd926ff500b0/elastic-agent-client.proto#L280

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah, didn't think of that! Should probably add some kind of check to diff the actual config.

if cm.stopOnOutputReload && cm.outputIsConfigured {
cm.logger.Infof("beat will now stop for output reload")
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: same thing here

Suggested change
cm.logger.Infof("beat will now stop for output reload")
cm.logger.Info("beat will now stop for output reload")

// in the future we'll want some set "reloading" state,
// but for now set the state this way.
_ = unit.UpdateState(client.UnitStateStopping, "got output unit, beat will restart", nil)
cm.Stop()
return
}

reloadConfig, err := groupByOutputs(rawConfig)
if err != nil {
errString := fmt.Errorf("Failed to generate config for output: %w", err)
Expand All @@ -359,6 +377,8 @@ func (cm *BeatV2Manager) handleOutputReload(unit *client.Unit) {
_ = unit.UpdateState(client.UnitStateFailed, errString.Error(), nil)
return
}
// set to true, we'll reload the output if we need to re-configure
cm.outputIsConfigured = true
_ = unit.UpdateState(client.UnitStateHealthy, "reloaded output component", nil)
}

Expand Down