gte/internal/task/recur.go

163 lines
2.5 KiB
Go

package task
import (
"fmt"
"strings"
"time"
)
type Recurrer interface {
RecursOn(date Date) bool
String() string
}
func NewRecurrer(recurStr string) Recurrer {
terms := strings.Split(recurStr, ", ")
if len(terms) < 2 {
return nil
}
start := NewDateFromString(terms[0])
if start.IsZero() {
return nil
}
terms = terms[1:]
if recur, ok := ParseDaily(start, terms); ok {
return recur
}
if recur, ok := ParseWeekly(start, terms); ok {
return recur
}
if recur, ok := ParseBiweekly(start, terms); ok {
return recur
}
return nil
}
type Daily struct {
Start Date
}
func ParseDaily(start Date, terms []string) (Recurrer, bool) {
if len(terms) < 1 {
return nil, false
}
if terms[0] != "daily" {
return nil, false
}
return Daily{
Start: start,
}, true
}
func (d Daily) RecursOn(date Date) bool {
return date.Equal(d.Start) || date.After(d.Start)
}
func (d Daily) String() string {
return fmt.Sprintf("%s, daily", d.Start.String())
}
type Weekly struct {
Start Date
Weekday time.Weekday
}
// yyyy-mm-dd, weekly, wednesday
func ParseWeekly(start Date, terms []string) (Recurrer, bool) {
if len(terms) < 2 {
return nil, false
}
if terms[0] != "weekly" {
return nil, false
}
wd, ok := ParseWeekday(terms[1])
if !ok {
return nil, false
}
return Weekly{
Start: start,
Weekday: wd,
}, true
}
func (w Weekly) RecursOn(date Date) bool {
if w.Start.After(date) {
return false
}
return w.Weekday == date.Weekday()
}
func (w Weekly) String() string {
return fmt.Sprintf("%s, weekly, %s", w.Start.String(), strings.ToLower(w.Weekday.String()))
}
type Biweekly struct {
Start Date
Weekday time.Weekday
}
// yyyy-mm-dd, biweekly, wednesday
func ParseBiweekly(start Date, terms []string) (Recurrer, bool) {
if len(terms) < 2 {
return nil, false
}
if terms[0] != "biweekly" {
return nil, false
}
wd, ok := ParseWeekday(terms[1])
if !ok {
return nil, false
}
return Biweekly{
Start: start,
Weekday: wd,
}, true
}
func (b Biweekly) RecursOn(date Date) bool {
if b.Start.After(date) {
return false
}
if b.Weekday != date.Weekday() {
return false
}
// find first
tDate := b.Start
for {
if tDate.Weekday() == b.Weekday {
break
}
tDate = tDate.AddDays(1)
}
// add weeks
for {
switch {
case tDate.Equal(date):
return true
case tDate.After(date):
return false
}
tDate = tDate.AddDays(14)
}
}
func (b Biweekly) String() string {
return fmt.Sprintf("%s, biweekly, %s", b.Start.String(), strings.ToLower(b.Weekday.String()))
}