-
Notifications
You must be signed in to change notification settings - Fork 2
/
attack.go
70 lines (63 loc) · 1.8 KB
/
attack.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
package loaderbot
import (
"context"
"time"
)
// Attack must be implemented by a service client.
type Attack interface {
// Setup should establish the connection to the service
// It may want to access the Config of the Runner.
Setup(c RunnerConfig) error
// Do performs one request and is executed in a separate goroutine.
// The context is used to CancelFunc the request on timeout.
Do(ctx context.Context) DoResult
// Teardown can be used to close the connection to the service
Teardown() error
// Clone should return a fresh new Attack
// Make sure the new Attack has values for shared struct fields initialized at Setup.
Clone(r *Runner) Attack
}
// attack receives schedule signal and attacks target calling Do() method, returning AttackResult with timings
func attack(a Attack, r *Runner) {
for nextMsg := range r.next {
token := nextMsg
requestCtx, requestCtxCancel := context.WithTimeout(context.Background(), time.Duration(r.Cfg.AttackerTimeout)*time.Second)
tStart := time.Now()
done := make(chan DoResult, 1)
var doResult DoResult
go func() {
select {
case <-r.TimeoutCtx.Done():
requestCtxCancel()
return
case <-requestCtx.Done():
case done <- a.Do(requestCtx):
}
}()
// either get the result from the attacker or from the timeout
select {
case <-r.TimeoutCtx.Done():
requestCtxCancel()
return
case <-requestCtx.Done():
doResult = DoResult{
RequestLabel: r.Name,
Error: errAttackDoTimedOut,
}
case doResult = <-done:
}
tEnd := time.Now()
atkResult := AttackResult{
AttackToken: token,
Begin: tStart,
End: tEnd,
Elapsed: tEnd.Sub(tStart),
DoResult: doResult,
}
requestCtxCancel()
if err := a.Teardown(); err != nil {
r.L.Infof("teardown failed: %s", err)
}
r.results <- atkResult
}
}