Skip to content

Commit 7e86d24

Browse files
authored
Uninstall finds and kills any running elastic-agent watch process (#3384)
* kill watcher on uninstall * Empty commit. * Fix killWatcher. * Empty commit. * Another fix for killWatcher. * Empty commit. * Catch ErrProcessDone. * Empty commit. * Empty commit * Add changelog fragment. * Make it work on Windows. * Change killWatcher to be in a loop. * Add loop to killWatcher. * Revert "Skip TestStandaloneUpgradeFailsStatus to fix failing integration tests again (#3391)" This reverts commit bf467e3. * Revert "Fix integration tests by waiting for the watcher to finish during upgrade tests (#3370)" This reverts commit 94764be. * Fix test. * Revert "Revert "Skip TestStandaloneUpgradeFailsStatus to fix failing integration tests again (#3391)"" This reverts commit 3b0f040. * Add progress tracking for uninstall like install. * Log when no watchers our found. * Improve uninstall. * Fix data race.
1 parent b1d2e6b commit 7e86d24

File tree

9 files changed

+420
-159
lines changed

9 files changed

+420
-159
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
# Kind can be one of:
2+
# - breaking-change: a change to previously-documented behavior
3+
# - deprecation: functionality that is being removed in a later release
4+
# - bug-fix: fixes a problem in a previous version
5+
# - enhancement: extends functionality but does not break or fix existing behavior
6+
# - feature: new functionality
7+
# - known-issue: problems that we are aware of in a given version
8+
# - security: impacts on the security of a product or a user’s deployment.
9+
# - upgrade: important information for someone upgrading from a prior version
10+
# - other: does not fit into any of the other categories
11+
kind: bug-fix
12+
13+
# Change summary; a 80ish characters long description of the change.
14+
summary: Uninstall finds and kills any running watcher process
15+
16+
# Long description; in case the summary is not enough to describe the change
17+
# this field accommodate a description without length limits.
18+
# NOTE: This field will be rendered only for breaking-change and known-issue kinds at the moment.
19+
#description:
20+
21+
# Affected component; usually one of "elastic-agent", "fleet-server", "filebeat", "metricbeat", "auditbeat", "all", etc.
22+
component:
23+
24+
# PR URL; optional; the PR number that added the changeset.
25+
# If not present is automatically filled by the tooling finding the PR where this changelog fragment has been added.
26+
# NOTE: the tooling supports backports, so it's able to fill the original PR number instead of the backport PR number.
27+
# Please provide it if you are adding a fragment for a different PR.
28+
pr: https://github.com/elastic/elastic-agent/pull/3384
29+
30+
# Issue URL; optional; the GitHub issue related to this changeset (either closes or is part of).
31+
# If not present is automatically filled by the tooling with the issue linked to the PR number.
32+
issue: https://github.com/elastic/elastic-agent/issues/3371

internal/pkg/agent/cmd/install.go

+32-21
Original file line numberDiff line numberDiff line change
@@ -180,37 +180,53 @@ func installCmd(streams *cli.IOStreams, cmd *cobra.Command) error {
180180
}
181181

182182
pt := install.NewProgressTracker(streams.Out)
183-
pt.Start()
184-
defer pt.Stop()
183+
s := pt.Start()
184+
defer func() {
185+
if err != nil {
186+
s.Failed()
187+
} else {
188+
s.Succeeded()
189+
}
190+
}()
185191

186192
cfgFile := paths.ConfigFile()
187193
if status != install.PackageInstall {
188-
err = install.Install(cfgFile, topPath, pt)
194+
err = install.Install(cfgFile, topPath, s)
189195
if err != nil {
190196
return err
191197
}
192198

193199
defer func() {
194200
if err != nil {
195-
_ = install.Uninstall(cfgFile, topPath, "")
201+
uninstallStep := s.StepStart("Uninstalling")
202+
innerErr := install.Uninstall(cfgFile, topPath, "", uninstallStep)
203+
if innerErr != nil {
204+
uninstallStep.Failed()
205+
} else {
206+
uninstallStep.Succeeded()
207+
}
196208
}
197209
}()
198210

199211
if !delayEnroll {
200-
pt.StepStart("Starting service")
212+
startServiceStep := s.StepStart("Starting service")
201213
err = install.StartService(topPath)
202214
if err != nil {
203-
pt.StepFailed()
215+
startServiceStep.Failed()
204216
fmt.Fprintf(streams.Out, "Installation failed to start Elastic Agent service.\n")
205217
return err
206218
}
207-
pt.StepSucceeded()
219+
startServiceStep.Succeeded()
208220

209221
defer func() {
210222
if err != nil {
211-
fmt.Fprint(streams.Out, "Stopping service... ")
212-
_ = install.StopService(topPath)
213-
pt.StepSucceeded()
223+
stoppingServiceStep := s.StepStart("Stopping service")
224+
innerErr := install.StopService(topPath)
225+
if innerErr != nil {
226+
stoppingServiceStep.Failed()
227+
} else {
228+
stoppingServiceStep.Succeeded()
229+
}
214230
}
215231
}()
216232
}
@@ -224,25 +240,20 @@ func installCmd(streams *cli.IOStreams, cmd *cobra.Command) error {
224240
enrollCmd.Stdout = os.Stdout
225241
enrollCmd.Stderr = os.Stderr
226242

227-
fmt.Fprint(streams.Out, "Enrolling Elastic Agent with Fleet... ")
243+
enrollStep := s.StepStart("Enrolling Elastic Agent with Fleet")
228244
err = enrollCmd.Start()
229245
if err != nil {
230-
pt.StepFailed()
246+
enrollStep.Failed()
231247
return fmt.Errorf("failed to execute enroll command: %w", err)
232248
}
233249
err = enrollCmd.Wait()
234250
if err != nil {
235-
pt.StepFailed()
236-
if status != install.PackageInstall {
237-
var exitErr *exec.ExitError
238-
_ = install.Uninstall(cfgFile, topPath, "")
239-
if err != nil && errors.As(err, &exitErr) {
240-
return fmt.Errorf("enroll command failed with exit code: %d", exitErr.ExitCode())
241-
}
242-
}
251+
enrollStep.Failed()
252+
// uninstall doesn't need to be performed here the defer above will
253+
// catch the error and perform the uninstall
243254
return fmt.Errorf("enroll command failed for unknown reason: %w", err)
244255
}
245-
pt.StepSucceeded()
256+
enrollStep.Succeeded()
246257
}
247258

248259
if err := info.CreateInstallMarker(topPath); err != nil {

internal/pkg/agent/cmd/uninstall.go

+7-1
Original file line numberDiff line numberDiff line change
@@ -80,9 +80,15 @@ func uninstallCmd(streams *cli.IOStreams, cmd *cobra.Command) error {
8080
}
8181
}
8282

83-
err = install.Uninstall(paths.ConfigFile(), paths.Top(), uninstallToken)
83+
pt := install.NewProgressTracker(streams.Out)
84+
s := pt.Start()
85+
86+
err = install.Uninstall(paths.ConfigFile(), paths.Top(), uninstallToken, s)
8487
if err != nil {
88+
s.Failed()
8589
return err
90+
} else {
91+
s.Succeeded()
8692
}
8793
fmt.Fprintf(streams.Out, "Elastic Agent has been uninstalled.\n")
8894

internal/pkg/agent/install/install.go

+12-12
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ const (
2222
)
2323

2424
// Install installs Elastic Agent persistently on the system including creating and starting its service.
25-
func Install(cfgFile, topPath string, pt *ProgressTracker) error {
25+
func Install(cfgFile, topPath string, pt ProgressTrackerStep) error {
2626
dir, err := findDirectory()
2727
if err != nil {
2828
return errors.New(err, "failed to discover the source directory for installation", errors.TypeFilesystem)
@@ -33,16 +33,16 @@ func Install(cfgFile, topPath string, pt *ProgressTracker) error {
3333
// There is no uninstall token for "install" command.
3434
// Uninstall will fail on protected agent.
3535
// The protected Agent will need to be uninstalled first before it can be installed.
36-
pt.StepStart("Uninstalling current Elastic Agent")
37-
err = Uninstall(cfgFile, topPath, "")
36+
s := pt.StepStart("Uninstalling current Elastic Agent")
37+
err = Uninstall(cfgFile, topPath, "", s)
3838
if err != nil {
39-
pt.StepFailed()
39+
s.Failed()
4040
return errors.New(
4141
err,
4242
fmt.Sprintf("failed to uninstall Agent at (%s)", filepath.Dir(topPath)),
4343
errors.M("directory", filepath.Dir(topPath)))
4444
}
45-
pt.StepSucceeded()
45+
s.Succeeded()
4646

4747
// ensure parent directory exists
4848
err = os.MkdirAll(filepath.Dir(topPath), 0755)
@@ -54,21 +54,21 @@ func Install(cfgFile, topPath string, pt *ProgressTracker) error {
5454
}
5555

5656
// copy source into install path
57-
pt.StepStart("Copying files")
57+
s = pt.StepStart("Copying files")
5858
err = copy.Copy(dir, topPath, copy.Options{
5959
OnSymlink: func(_ string) copy.SymlinkAction {
6060
return copy.Shallow
6161
},
6262
Sync: true,
6363
})
6464
if err != nil {
65-
pt.StepFailed()
65+
s.Failed()
6666
return errors.New(
6767
err,
6868
fmt.Sprintf("failed to copy source directory (%s) to destination (%s)", dir, topPath),
6969
errors.M("source", dir), errors.M("destination", topPath))
7070
}
71-
pt.StepSucceeded()
71+
s.Succeeded()
7272

7373
// place shell wrapper, if present on platform
7474
if paths.ShellWrapperPath != "" {
@@ -132,21 +132,21 @@ func Install(cfgFile, topPath string, pt *ProgressTracker) error {
132132
}
133133

134134
// install service
135-
pt.StepStart("Installing service")
135+
s = pt.StepStart("Installing service")
136136
svc, err := newService(topPath)
137137
if err != nil {
138-
pt.StepFailed()
138+
s.Failed()
139139
return err
140140
}
141141
err = svc.Install()
142142
if err != nil {
143-
pt.StepFailed()
143+
s.Failed()
144144
return errors.New(
145145
err,
146146
fmt.Sprintf("failed to install service (%s)", paths.ServiceName),
147147
errors.M("service", paths.ServiceName))
148148
}
149-
pt.StepSucceeded()
149+
s.Succeeded()
150150

151151
return nil
152152
}

0 commit comments

Comments
 (0)