mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2025-01-18 20:40:43 +00:00
162 lines
3.7 KiB
Go
162 lines
3.7 KiB
Go
|
package cron
|
||
|
|
||
|
import (
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
// SpecSchedule specifies a duty cycle (to the second granularity), based on a
|
||
|
// traditional crontab specification. It is computed initially and stored as bit sets.
|
||
|
type SpecSchedule struct {
|
||
|
Second, Minute, Hour, Dom, Month, Dow uint64
|
||
|
}
|
||
|
|
||
|
// bounds provides a range of acceptable values (plus a map of name to value).
|
||
|
type bounds struct {
|
||
|
min, max uint
|
||
|
names map[string]uint
|
||
|
}
|
||
|
|
||
|
// The bounds for each field.
|
||
|
var (
|
||
|
seconds = bounds{0, 59, nil}
|
||
|
minutes = bounds{0, 59, nil}
|
||
|
hours = bounds{0, 23, nil}
|
||
|
dom = bounds{1, 31, nil}
|
||
|
months = bounds{1, 12, map[string]uint{
|
||
|
"jan": 1,
|
||
|
"feb": 2,
|
||
|
"mar": 3,
|
||
|
"apr": 4,
|
||
|
"may": 5,
|
||
|
"jun": 6,
|
||
|
"jul": 7,
|
||
|
"aug": 8,
|
||
|
"sep": 9,
|
||
|
"oct": 10,
|
||
|
"nov": 11,
|
||
|
"dec": 12,
|
||
|
}}
|
||
|
dow = bounds{0, 6, map[string]uint{
|
||
|
"sun": 0,
|
||
|
"mon": 1,
|
||
|
"tue": 2,
|
||
|
"wed": 3,
|
||
|
"thu": 4,
|
||
|
"fri": 5,
|
||
|
"sat": 6,
|
||
|
}}
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
// Set the top bit if a star was included in the expression.
|
||
|
starBit = 1 << 63
|
||
|
)
|
||
|
|
||
|
// Next returns the next time this schedule is activated, greater than the given
|
||
|
// time. If no time can be found to satisfy the schedule, return the zero time.
|
||
|
func (s *SpecSchedule) Next(t time.Time) time.Time {
|
||
|
// General approach:
|
||
|
// For Month, Day, Hour, Minute, Second:
|
||
|
// Check if the time value matches. If yes, continue to the next field.
|
||
|
// If the field doesn't match the schedule, then increment the field until it matches.
|
||
|
// While incrementing the field, a wrap-around brings it back to the beginning
|
||
|
// of the field list (since it is necessary to re-verify previous field
|
||
|
// values)
|
||
|
|
||
|
// Start at the earliest possible time (the upcoming second).
|
||
|
t = t.Add(1*time.Second - time.Duration(t.Nanosecond())*time.Nanosecond)
|
||
|
|
||
|
// This flag indicates whether a field has been incremented.
|
||
|
added := false
|
||
|
|
||
|
// If no time is found within five years, return zero.
|
||
|
yearLimit := t.Year() + 5
|
||
|
|
||
|
WRAP:
|
||
|
if t.Year() > yearLimit {
|
||
|
return time.Time{}
|
||
|
}
|
||
|
|
||
|
// Find the first applicable month.
|
||
|
// If it's this month, then do nothing.
|
||
|
for 1<<uint(t.Month())&s.Month == 0 {
|
||
|
// If we have to add a month, reset the other parts to 0.
|
||
|
if !added {
|
||
|
added = true
|
||
|
// Otherwise, set the date at the beginning (since the current time is irrelevant).
|
||
|
t = time.Date(t.Year(), t.Month(), 1, 0, 0, 0, 0, t.Location())
|
||
|
}
|
||
|
t = t.AddDate(0, 1, 0)
|
||
|
|
||
|
// Wrapped around.
|
||
|
if t.Month() == time.January {
|
||
|
goto WRAP
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Now get a day in that month.
|
||
|
for !dayMatches(s, t) {
|
||
|
if !added {
|
||
|
added = true
|
||
|
t = time.Date(t.Year(), t.Month(), t.Day(), 0, 0, 0, 0, t.Location())
|
||
|
}
|
||
|
t = t.AddDate(0, 0, 1)
|
||
|
|
||
|
if t.Day() == 1 {
|
||
|
goto WRAP
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for 1<<uint(t.Hour())&s.Hour == 0 {
|
||
|
if !added {
|
||
|
added = true
|
||
|
t = time.Date(t.Year(), t.Month(), t.Day(), t.Hour(), 0, 0, 0, t.Location())
|
||
|
}
|
||
|
t = t.Add(1 * time.Hour)
|
||
|
|
||
|
if t.Hour() == 0 {
|
||
|
goto WRAP
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for 1<<uint(t.Minute())&s.Minute == 0 {
|
||
|
if !added {
|
||
|
added = true
|
||
|
t = time.Date(t.Year(), t.Month(), t.Day(), t.Hour(), t.Minute(), 0, 0, t.Location())
|
||
|
}
|
||
|
t = t.Add(1 * time.Minute)
|
||
|
|
||
|
if t.Minute() == 0 {
|
||
|
goto WRAP
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for 1<<uint(t.Second())&s.Second == 0 {
|
||
|
if !added {
|
||
|
added = true
|
||
|
t = time.Date(t.Year(), t.Month(), t.Day(), t.Hour(), t.Minute(), t.Second(), 0, t.Location())
|
||
|
}
|
||
|
t = t.Add(1 * time.Second)
|
||
|
|
||
|
if t.Second() == 0 {
|
||
|
goto WRAP
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return t
|
||
|
}
|
||
|
|
||
|
// dayMatches returns true if the schedule's day-of-week and day-of-month
|
||
|
// restrictions are satisfied by the given time.
|
||
|
func dayMatches(s *SpecSchedule, t time.Time) bool {
|
||
|
var (
|
||
|
domMatch bool = 1<<uint(t.Day())&s.Dom > 0
|
||
|
dowMatch bool = 1<<uint(t.Weekday())&s.Dow > 0
|
||
|
)
|
||
|
|
||
|
if s.Dom&starBit > 0 || s.Dow&starBit > 0 {
|
||
|
return domMatch && dowMatch
|
||
|
}
|
||
|
return domMatch || dowMatch
|
||
|
}
|