2018-07-09 22:04:25 -07:00
|
|
|
// Package semerrgroup wraps a error group with a semaphore with configurable
|
|
|
|
// size, so you can control the number of tasks being executed simultaneously.
|
2018-07-09 21:38:00 -07:00
|
|
|
package semerrgroup
|
|
|
|
|
2019-10-06 14:58:33 -03:00
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/goreleaser/goreleaser/internal/pipe"
|
|
|
|
"golang.org/x/sync/errgroup"
|
|
|
|
)
|
2018-07-09 21:38:00 -07:00
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// Group is the Semphore ErrorGroup itself.
|
2019-08-05 04:36:14 -07:00
|
|
|
type Group interface {
|
|
|
|
Go(func() error)
|
|
|
|
Wait() error
|
2018-07-09 21:38:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// New returns a new Group of a given size.
|
2019-08-05 04:36:14 -07:00
|
|
|
func New(size int) Group {
|
2022-06-11 16:54:55 -03:00
|
|
|
var g errgroup.Group
|
|
|
|
g.SetLimit(size)
|
|
|
|
return &g
|
2019-08-05 04:36:14 -07:00
|
|
|
}
|
2019-10-06 14:58:33 -03:00
|
|
|
|
|
|
|
var _ Group = &skipAwareGroup{}
|
|
|
|
|
|
|
|
// NewSkipAware returns a new Group of a given size and aware of pipe skips.
|
|
|
|
func NewSkipAware(g Group) Group {
|
|
|
|
return &skipAwareGroup{g: g}
|
|
|
|
}
|
|
|
|
|
|
|
|
type skipAwareGroup struct {
|
|
|
|
g Group
|
|
|
|
skipErr error
|
|
|
|
skipOnce sync.Once
|
|
|
|
}
|
|
|
|
|
|
|
|
// Go execs runs `fn` and saves the result if no error has been encountered.
|
|
|
|
func (s *skipAwareGroup) Go(fn func() error) {
|
|
|
|
s.g.Go(func() error {
|
2021-04-25 14:20:49 -03:00
|
|
|
err := fn()
|
2019-10-06 14:58:33 -03:00
|
|
|
// if the err is a skip, set it for later, but return nil for now so the
|
2022-08-30 22:42:29 +08:00
|
|
|
// group proceeds.
|
2019-10-06 14:58:33 -03:00
|
|
|
if pipe.IsSkip(err) {
|
|
|
|
s.skipOnce.Do(func() {
|
|
|
|
s.skipErr = err
|
|
|
|
})
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait waits for Go to complete and returns the first error encountered.
|
|
|
|
func (s *skipAwareGroup) Wait() error {
|
|
|
|
// if we got a "real error", return it, otherwise return skipErr or nil.
|
|
|
|
if err := s.g.Wait(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return s.skipErr
|
|
|
|
}
|