-
Notifications
You must be signed in to change notification settings - Fork 1
/
asyncwait.go
70 lines (60 loc) · 1.32 KB
/
asyncwait.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 asyncwait
import (
"context"
"time"
)
// AsyncWait async wait representation
type AsyncWait interface {
// Wait method for wait result
Wait(func() bool) bool
}
var _ AsyncWait = (*asyncWait)(nil)
type asyncWait struct {
pollInterval time.Duration
timeout time.Duration
doneCh chan struct{}
}
// NewAsyncWait constructor for AsyncWait
func NewAsyncWait(timeout, pollInterval time.Duration) AsyncWait {
return &asyncWait{
pollInterval: pollInterval,
timeout: timeout,
doneCh: make(chan struct{}),
}
}
// Wait while timeout, make polls every pollInterval for the predicate until it returns true
func (aw asyncWait) Wait(predicate func() bool) bool {
ctx, cancel := context.WithTimeout(context.Background(), aw.timeout)
defer cancel()
ticker := time.NewTicker(aw.pollInterval)
defer ticker.Stop()
runPredicate := func() {
if predicate() {
select {
case aw.doneCh <- struct{}{}:
case <-ctx.Done():
}
}
}
// Try to execute predicate immediately
go runPredicate()
// If the first call was not successful, start polling
for {
select {
case <-aw.doneCh:
return true
case <-ctx.Done():
return false
case <-ticker.C:
go func() {
if predicate() {
select {
case aw.doneCh <- struct{}{}:
case <-ctx.Done():
return
}
}
}()
}
}
}