-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathexecutor.go
More file actions
92 lines (74 loc) · 1.59 KB
/
executor.go
File metadata and controls
92 lines (74 loc) · 1.59 KB
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
// Copyright 2025 FishGoddess. All rights reserved.
// Use of this source code is governed by a MIT style
// license that can be found in the LICENSE file.
package goes
import (
"context"
"errors"
"sync"
"sync/atomic"
)
const (
minWorkers = 1
maxWorkers = 10000
)
var (
ErrExecutorClosed = errors.New("goes: executor is closed")
)
// Executor starts some workers to do tasks concurrently.
type Executor struct {
conf *config
workers uint
tasks chan Task
done chan struct{}
closed atomic.Bool
group sync.WaitGroup
}
// NewExecutor creates a executor with workers.
func NewExecutor(workers uint, opts ...Option) *Executor {
conf := newConfig().apply(opts...)
if workers < minWorkers {
workers = minWorkers
}
if workers > maxWorkers {
workers = maxWorkers
}
executor := &Executor{
conf: conf,
workers: workers,
tasks: make(chan Task, conf.queueSize),
done: make(chan struct{}),
}
for range workers {
executor.group.Go(executor.worker)
}
return executor
}
func (e *Executor) worker() {
for task := range e.tasks {
task.Do(e.conf.recovery)
}
}
// Submit submits a task to executor and returns an error if failed.
func (e *Executor) Submit(ctx context.Context, task Task) error {
if e.closed.Load() {
return ErrExecutorClosed
}
select {
case e.tasks <- task:
return nil
case <-ctx.Done():
return ctx.Err()
case <-e.done:
return ErrExecutorClosed
}
}
// Close closes the executor and waits all tasks to be done.
func (e *Executor) Close() {
if !e.closed.CompareAndSwap(false, true) {
return
}
close(e.done)
close(e.tasks)
e.group.Wait()
}