You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

69 lines
1.7 KiB

package tdsync
import (
"context"
"github.com/go-faster/errors"
"go.uber.org/zap"
"github.com/gotd/td/clock"
)
// LogGroup is simple wrapper around CancellableGroup to log task state.
// Unlike WaitGroup and errgroup.Group this is not allowed to use zero value.
type LogGroup struct {
group CancellableGroup
log *zap.Logger
clock clock.Clock
}
// NewLogGroup creates new LogGroup.
func NewLogGroup(parent context.Context, log *zap.Logger) *LogGroup {
return &LogGroup{
group: *NewCancellableGroup(parent),
log: log,
clock: clock.System,
}
}
// SetClock sets Clock to use.
func (g *LogGroup) SetClock(c clock.Clock) {
g.clock = c
}
// Go calls the given function in a new goroutine.
//
// The first call to return a non-nil error cancels the group; its error will be
// returned by Wait.
func (g *LogGroup) Go(taskName string, f func(groupCtx context.Context) error) {
g.group.Go(func(ctx context.Context) error {
start := g.clock.Now()
l := g.log.With(zap.String("task", taskName)).WithOptions(zap.AddCallerSkip(1))
l.Debug("Task started")
if err := f(ctx); err != nil {
elapsed := g.clock.Now().Sub(start)
l.Debug("Task stopped", zap.Error(err), zap.Duration("elapsed", elapsed))
return errors.Wrapf(err, "task %s", taskName)
}
elapsed := g.clock.Now().Sub(start)
l.Debug("Task complete", zap.Duration("elapsed", elapsed))
return nil
})
}
// Cancel cancels all goroutines in group.
//
// Note: context cancellation error will be returned by Wait().
func (g *LogGroup) Cancel() {
g.group.Cancel()
}
// Wait blocks until all function calls from the Go method have returned, then
// returns the first non-nil error (if any) from them.
func (g *LogGroup) Wait() error {
return g.group.Wait()
}