2016-02-04 22:33:11 +01:00
|
|
|
package worker
|
|
|
|
|
2017-06-04 11:16:55 +02:00
|
|
|
import "context"
|
|
|
|
|
2016-02-05 22:22:24 +01:00
|
|
|
// Job is one unit of work. It is given to a Func, and the returned result and
|
|
|
|
// error are stored in Result and Error.
|
|
|
|
type Job struct {
|
|
|
|
Data interface{}
|
2016-02-04 22:33:11 +01:00
|
|
|
Result interface{}
|
|
|
|
Error error
|
|
|
|
}
|
|
|
|
|
|
|
|
// Func does the actual work within a Pool.
|
2017-06-04 11:16:55 +02:00
|
|
|
type Func func(ctx context.Context, job Job) (result interface{}, err error)
|
2016-02-04 22:33:11 +01:00
|
|
|
|
|
|
|
// Pool implements a worker pool.
|
|
|
|
type Pool struct {
|
|
|
|
f Func
|
|
|
|
jobCh <-chan Job
|
2016-02-05 22:22:24 +01:00
|
|
|
resCh chan<- Job
|
2016-02-05 22:23:34 +01:00
|
|
|
|
|
|
|
numWorkers int
|
|
|
|
workersExit chan struct{}
|
|
|
|
allWorkersDone chan struct{}
|
2016-02-04 22:33:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// New returns a new worker pool with n goroutines, each running the function
|
|
|
|
// f. The workers are started immediately.
|
2017-06-04 11:16:55 +02:00
|
|
|
func New(ctx context.Context, n int, f Func, jobChan <-chan Job, resultChan chan<- Job) *Pool {
|
2016-02-04 22:33:11 +01:00
|
|
|
p := &Pool{
|
2016-02-05 22:23:34 +01:00
|
|
|
f: f,
|
|
|
|
workersExit: make(chan struct{}),
|
|
|
|
allWorkersDone: make(chan struct{}),
|
|
|
|
numWorkers: n,
|
|
|
|
jobCh: jobChan,
|
|
|
|
resCh: resultChan,
|
2016-02-04 22:33:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < n; i++ {
|
2017-06-05 23:56:59 +02:00
|
|
|
go p.runWorker(ctx, i)
|
2016-02-04 22:33:11 +01:00
|
|
|
}
|
|
|
|
|
2016-02-05 22:23:34 +01:00
|
|
|
go p.waitForExit()
|
|
|
|
|
2016-02-04 22:33:11 +01:00
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
2016-02-05 22:23:34 +01:00
|
|
|
// waitForExit receives from p.workersExit until all worker functions have
|
|
|
|
// exited, then closes the result channel.
|
|
|
|
func (p *Pool) waitForExit() {
|
|
|
|
n := p.numWorkers
|
|
|
|
for n > 0 {
|
|
|
|
<-p.workersExit
|
|
|
|
n--
|
|
|
|
}
|
|
|
|
close(p.allWorkersDone)
|
|
|
|
close(p.resCh)
|
|
|
|
}
|
|
|
|
|
2016-02-04 22:33:11 +01:00
|
|
|
// runWorker runs a worker function.
|
2017-06-05 23:56:59 +02:00
|
|
|
func (p *Pool) runWorker(ctx context.Context, numWorker int) {
|
2016-02-05 22:23:34 +01:00
|
|
|
defer func() {
|
|
|
|
p.workersExit <- struct{}{}
|
|
|
|
}()
|
2016-02-04 22:33:11 +01:00
|
|
|
|
|
|
|
var (
|
|
|
|
// enable the input channel when starting up a new goroutine
|
|
|
|
inCh = p.jobCh
|
|
|
|
// but do not enable the output channel until we have a result
|
2016-02-05 22:22:24 +01:00
|
|
|
outCh chan<- Job
|
2016-02-04 22:33:11 +01:00
|
|
|
|
|
|
|
job Job
|
|
|
|
ok bool
|
|
|
|
)
|
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
2017-06-05 23:56:59 +02:00
|
|
|
case <-ctx.Done():
|
2016-02-04 22:33:11 +01:00
|
|
|
return
|
|
|
|
|
|
|
|
case job, ok = <-inCh:
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-06-05 23:56:59 +02:00
|
|
|
job.Result, job.Error = p.f(ctx, job)
|
2016-02-04 22:33:11 +01:00
|
|
|
inCh = nil
|
|
|
|
outCh = p.resCh
|
|
|
|
|
2016-02-05 22:22:24 +01:00
|
|
|
case outCh <- job:
|
2016-02-04 22:33:11 +01:00
|
|
|
outCh = nil
|
|
|
|
inCh = p.jobCh
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait waits for all worker goroutines to terminate, afterwards the output
|
|
|
|
// channel is closed.
|
|
|
|
func (p *Pool) Wait() {
|
2016-02-05 22:23:34 +01:00
|
|
|
<-p.allWorkersDone
|
2016-02-04 22:33:11 +01:00
|
|
|
}
|