-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathexecutor_errgroup.go
76 lines (65 loc) · 1.5 KB
/
executor_errgroup.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
package sync
import (
"context"
"sync"
"sync/atomic"
"golang.org/x/sync/errgroup"
)
// errGroupExecutor is an Executor that executes units of work, blocking when Go is called once the maxConcurrency
// is reached, only continuing subsequent Go calls when the nuber of executing functions drops below maxConcurrency
type errGroupExecutor struct {
maxConcurrency int
canceled atomic.Bool
g errgroup.Group
wg sync.WaitGroup
childLock sync.RWMutex
childExecutor *errGroupExecutor
}
func newErrGroupExecutor(maxConcurrency int) *errGroupExecutor {
e := &errGroupExecutor{
maxConcurrency: maxConcurrency,
}
e.g.SetLimit(maxConcurrency)
return e
}
func (e *errGroupExecutor) Go(f func()) {
e.wg.Add(1)
fn := func() error {
defer e.wg.Done()
if e.canceled.Load() {
return nil
}
f()
return nil
}
e.g.Go(fn)
}
func (e *errGroupExecutor) Wait(ctx context.Context) {
e.canceled.Store(ctx.Err() != nil)
done := make(chan struct{})
go func() {
e.wg.Wait()
close(done)
}()
select {
case <-ctx.Done():
e.canceled.Store(true)
case <-done:
}
}
func (e *errGroupExecutor) ChildExecutor() Executor {
e.childLock.RLock()
child := e.childExecutor
e.childLock.RUnlock()
if child != nil {
return child
}
e.childLock.Lock()
defer e.childLock.Unlock()
if e.childExecutor == nil {
// create child executor with same bound
e.childExecutor = newErrGroupExecutor(e.maxConcurrency)
}
return e.childExecutor
}
var _ Executor = (*errGroupExecutor)(nil)