123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596 |
- package cron
- import (
- "errors"
- "fmt"
- "runtime"
- "sync"
- "time"
- )
- // JobWrapper decorates the given Job with some behavior.
- type JobWrapper func(Job) Job
- // Chain is a sequence of JobWrappers that decorates submitted jobs with
- // cross-cutting behaviors like logging or synchronization.
- type Chain struct {
- wrappers []JobWrapper
- }
- // NewChain returns a Chain consisting of the given JobWrappers.
- func NewChain(c ...JobWrapper) Chain {
- return Chain{c}
- }
- // Then decorates the given job with all JobWrappers in the chain.
- //
- // This:
- //
- // NewChain(m1, m2, m3).Then(job)
- //
- // is equivalent to:
- //
- // m1(m2(m3(job)))
- func (c Chain) Then(j Job) Job {
- for i := range c.wrappers {
- j = c.wrappers[len(c.wrappers)-i-1](j)
- }
- return j
- }
- // Recover panics in wrapped jobs and log them with the provided logger.
- func Recover(logger Logger) JobWrapper {
- return func(j Job) Job {
- return FuncJob(func() {
- defer func() {
- if r := recover(); r != nil {
- const size = 64 << 10
- buf := make([]byte, size)
- buf = buf[:runtime.Stack(buf, false)]
- err, ok := r.(error)
- if !ok {
- err = errors.New("panic: " + fmt.Sprint(r))
- }
- logger.Error(err, "panic", "stack", "...\n"+string(buf))
- }
- }()
- j.Run()
- })
- }
- }
- // DelayIfStillRunning serializes jobs, delaying subsequent runs until the
- // previous one is complete. Jobs running after a delay of more than a minute
- // have the delay logged at Info.
- func DelayIfStillRunning(logger Logger) JobWrapper {
- return func(j Job) Job {
- var mu sync.Mutex
- return FuncJob(func() {
- start := time.Now()
- mu.Lock()
- defer mu.Unlock()
- if dur := time.Since(start); dur > time.Minute {
- logger.Info("delay", "duration", dur)
- }
- j.Run()
- })
- }
- }
- // SkipIfStillRunning skips an invocation of the Job if a previous invocation is
- // still running. It logs skips to the given logger at Info level.
- func SkipIfStillRunning(logger Logger) JobWrapper {
- return func(j Job) Job {
- var ch = make(chan struct{}, 1)
- ch <- struct{}{}
- return FuncJob(func() {
- select {
- case v := <-ch:
- defer func() { ch <- v }()
- j.Run()
- default:
- logger.Info("skip")
- }
- })
- }
- }
|