clash/common/picker/picker.go

63 lines
1.5 KiB
Go
Raw Normal View History

2018-12-05 13:13:29 +00:00
package picker
2019-07-02 11:18:03 +00:00
import (
"context"
"sync"
"time"
2019-07-02 11:18:03 +00:00
)
// Picker provides synchronization, and Context cancelation
// for groups of goroutines working on subtasks of a common task.
// Inspired by errGroup
type Picker struct {
cancel func()
wg sync.WaitGroup
once sync.Once
result interface{}
}
// WithContext returns a new Picker and an associated Context derived from ctx.
// and cancel when first element return.
2019-07-02 11:18:03 +00:00
func WithContext(ctx context.Context) (*Picker, context.Context) {
ctx, cancel := context.WithCancel(ctx)
return &Picker{cancel: cancel}, ctx
}
// WithTimeout returns a new Picker and an associated Context derived from ctx with timeout,
// but it doesn't cancel when first element return.
func WithTimeout(ctx context.Context, timeout time.Duration) (*Picker, context.Context, context.CancelFunc) {
ctx, cancel := context.WithTimeout(ctx, timeout)
return &Picker{}, ctx, cancel
}
2019-07-02 11:18:03 +00:00
// Wait blocks until all function calls from the Go method have returned,
// then returns the first nil error result (if any) from them.
func (p *Picker) Wait() interface{} {
p.wg.Wait()
if p.cancel != nil {
p.cancel()
}
return p.result
}
// Go calls the given function in a new goroutine.
// The first call to return a nil error cancels the group; its result will be returned by Wait.
func (p *Picker) Go(f func() (interface{}, error)) {
p.wg.Add(1)
2018-12-05 13:13:29 +00:00
go func() {
2019-07-02 11:18:03 +00:00
defer p.wg.Done()
2018-12-05 13:13:29 +00:00
2019-07-02 11:18:03 +00:00
if ret, err := f(); err == nil {
p.once.Do(func() {
p.result = ret
if p.cancel != nil {
p.cancel()
}
})
2018-12-05 13:13:29 +00:00
}
}()
}