wisski-cloud-distillery/pkg/timex/timex.go
2022-11-16 13:07:12 +01:00

54 lines
1.2 KiB
Go

// Package timex provides Interval and Wait
package timex
import (
"context"
"time"
)
// TickContext is like [time.Tick], but closes the returned channel once the context closes.
// As such it can be recovered by the garbage collector; see [time.TickContext].
//
// Unlike [time.Tick], immediatly send the current time on the given channel.
func TickContext(c context.Context, d time.Duration) <-chan time.Time {
if d < 0 {
return nil
}
timer := make(chan time.Time, 1)
timer <- time.Now()
go func() {
t := time.NewTicker(d)
defer t.Stop()
defer close(timer)
for {
select {
case tick := <-t.C:
timer <- tick
case <-c.Done():
return
}
}
}()
return timer
}
// TickUntilFunc invokes f every d until either context is closed, or f returns true.
// f is invoked once immediatly when the timer starts.
//
// TickUntilFunc blocks until f is no longer invoked.
//
// Returns the error of the context (if any).
func TickUntilFunc(f func(t time.Time) bool, c context.Context, d time.Duration) error {
context, cancel := context.WithCancel(c)
defer cancel()
for t := range TickContext(context, d) {
if f(t) {
break
}
}
return c.Err()
}