mirror of
https://github.com/goreleaser/goreleaser.git
synced 2025-01-06 03:13:48 +02:00
d338cf7285
* fix: skip aware semerrgroup Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com> * fix: skip aware semerrgroup Signed-off-by: Carlos Alexandro Becker <caarlos0@gmail.com>
113 lines
2.3 KiB
Go
113 lines
2.3 KiB
Go
// Package semerrgroup wraps a error group with a semaphore with configurable
|
|
// size, so you can control the number of tasks being executed simultaneously.
|
|
package semerrgroup
|
|
|
|
import (
|
|
"sync"
|
|
|
|
"github.com/goreleaser/goreleaser/internal/pipe"
|
|
"golang.org/x/sync/errgroup"
|
|
)
|
|
|
|
// Group is the Semphore ErrorGroup itself
|
|
type Group interface {
|
|
Go(func() error)
|
|
Wait() error
|
|
}
|
|
|
|
// New returns a new Group of a given size.
|
|
func New(size int) Group {
|
|
if size == 1 {
|
|
return &serialGroup{}
|
|
}
|
|
return ¶llelGroup{
|
|
ch: make(chan bool, size),
|
|
g: errgroup.Group{},
|
|
}
|
|
}
|
|
|
|
var _ Group = ¶llelGroup{}
|
|
|
|
type parallelGroup struct {
|
|
ch chan bool
|
|
g errgroup.Group
|
|
}
|
|
|
|
// Go execs one function respecting the group and semaphore.
|
|
func (s *parallelGroup) Go(fn func() error) {
|
|
s.g.Go(func() error {
|
|
s.ch <- true
|
|
defer func() {
|
|
<-s.ch
|
|
}()
|
|
return fn()
|
|
})
|
|
}
|
|
|
|
// Wait waits for the group to complete and return an error if any.
|
|
func (s *parallelGroup) Wait() error {
|
|
return s.g.Wait()
|
|
}
|
|
|
|
var _ Group = &serialGroup{}
|
|
|
|
type serialGroup struct {
|
|
err error
|
|
errOnce sync.Once
|
|
}
|
|
|
|
// Go execs runs `fn` and saves the result if no error has been encountered.
|
|
func (s *serialGroup) Go(fn func() error) {
|
|
if s.err != nil {
|
|
return
|
|
}
|
|
if err := fn(); err != nil {
|
|
s.errOnce.Do(func() {
|
|
s.err = err
|
|
})
|
|
}
|
|
}
|
|
|
|
// Wait waits for Go to complete and returns the first error encountered.
|
|
func (s *serialGroup) Wait() error {
|
|
return s.err
|
|
}
|
|
|
|
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 {
|
|
var err = fn()
|
|
// if the err is a skip, set it for later, but return nil for now so the
|
|
// the group proceeds.
|
|
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
|
|
}
|