forked from cloudfoundry/bosh-agent
-
Notifications
You must be signed in to change notification settings - Fork 0
/
concrete_script.go
156 lines (125 loc) · 3.5 KB
/
concrete_script.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
package drain
import (
"strconv"
"strings"
"time"
"code.cloudfoundry.org/clock"
"github.com/cloudfoundry/bosh-agent/agent/script/cmd"
bosherr "github.com/cloudfoundry/bosh-utils/errors"
boshlog "github.com/cloudfoundry/bosh-utils/logger"
boshsys "github.com/cloudfoundry/bosh-utils/system"
)
type ConcreteScript struct {
fs boshsys.FileSystem
runner boshsys.CmdRunner
tag string
path string
params ScriptParams
timeService clock.Clock
logTag string
logger boshlog.Logger
cancelCh chan struct{}
}
func NewConcreteScript(
fs boshsys.FileSystem,
runner boshsys.CmdRunner,
tag string,
path string,
params ScriptParams,
timeService clock.Clock,
logger boshlog.Logger,
) ConcreteScript {
return ConcreteScript{
fs: fs,
runner: runner,
tag: tag,
path: path,
params: params,
timeService: timeService,
logTag: "DrainScript",
logger: logger,
cancelCh: make(chan struct{}, 1),
}
}
func (s ConcreteScript) Tag() string { return s.tag }
func (s ConcreteScript) Path() string { return s.path }
func (s ConcreteScript) Params() ScriptParams { return s.params }
func (s ConcreteScript) Exists() bool { return s.fs.FileExists(s.path) }
func (s ConcreteScript) Run() error {
params := s.params
for {
value, err := s.runOnce(params)
if err != nil {
return err
} else if value < 0 {
s.timeService.Sleep(time.Duration(-value) * time.Second)
params = params.ToStatusParams()
} else {
s.timeService.Sleep(time.Duration(value) * time.Second)
return nil
}
}
}
func (s ConcreteScript) Cancel() error {
select {
case s.cancelCh <- struct{}{}:
default:
}
return nil
}
func (s ConcreteScript) runOnce(params ScriptParams) (int, error) {
jobChange := params.JobChange()
hashChange := params.HashChange()
updatedPkgs := params.UpdatedPackages()
command := cmd.BuildCommand(s.path)
jobState, err := params.JobState()
if err != nil {
return 0, bosherr.WrapError(err, "Getting job state")
}
if jobState != "" {
command.Env["BOSH_JOB_STATE"] = jobState
}
jobNextState, err := params.JobNextState()
if err != nil {
return 0, bosherr.WrapError(err, "Getting job next state")
}
if jobNextState != "" {
command.Env["BOSH_JOB_NEXT_STATE"] = jobNextState
}
command.Args = append(command.Args, jobChange, hashChange)
command.Args = append(command.Args, updatedPkgs...)
process, err := s.runner.RunComplexCommandAsync(command)
if err != nil {
return 0, bosherr.WrapError(err, "Running drain script")
}
var result boshsys.Result
isCanceled := false
// Can only wait once on a process but cancelling can happen multiple times
for processExitedCh := process.Wait(); processExitedCh != nil; {
select {
case result = <-processExitedCh:
processExitedCh = nil
case <-s.cancelCh:
// Ignore possible TerminateNicely error since we cannot return it
err := process.TerminateNicely(10 * time.Second)
if err != nil {
s.logger.Error(s.logTag, "Failed to terminate %s", err.Error())
}
isCanceled = true
}
}
if isCanceled {
if result.Error != nil {
return 0, bosherr.WrapError(result.Error, "Script was cancelled by user request")
}
return 0, bosherr.Error("Script was cancelled by user request")
}
if result.Error != nil && result.ExitStatus == -1 {
return 0, bosherr.WrapError(result.Error, "Running drain script")
}
value, err := strconv.Atoi(strings.TrimSpace(result.Stdout))
if err != nil {
return 0, bosherr.WrapError(err, "Script did not return a signed integer")
}
return value, nil
}