227 lines
4.3 KiB
Go
227 lines
4.3 KiB
Go
package expr
|
|
|
|
import (
|
|
"errors"
|
|
"time"
|
|
|
|
"github.com/PaesslerAG/gval"
|
|
"github.com/lestrrat-go/strftime"
|
|
"github.com/spf13/cast"
|
|
)
|
|
|
|
func TimeFunctions() []gval.Language {
|
|
return []gval.Language{
|
|
gval.Function("earliest", earliest),
|
|
gval.Function("latest", latest),
|
|
gval.Function("parseISOTime", parseISOTime),
|
|
gval.Function("modTime", modTime),
|
|
gval.Function("modDate", modDate),
|
|
gval.Function("modWeek", modWeek),
|
|
gval.Function("modMonth", modMonth),
|
|
gval.Function("modYear", modYear),
|
|
gval.Function("parseDuration", time.ParseDuration),
|
|
gval.Function("strftime", strfTime),
|
|
gval.Function("isLeapYear", isLeapYear),
|
|
gval.Function("now", now),
|
|
gval.Function("isWeekDay", isWeekDay),
|
|
gval.Function("sub", sub),
|
|
}
|
|
}
|
|
|
|
func now() time.Time {
|
|
return time.Now()
|
|
}
|
|
|
|
func isLeapYear(base interface{}) (bool, error) {
|
|
t, _, err := prepMod(base, 0)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return time.Date(t.Year(), time.December, 31, 0, 0, 0, 0, time.Local).YearDay() == 366, nil
|
|
}
|
|
|
|
func isLeapDay(base interface{}) (bool, error) {
|
|
t, _, err := prepMod(base, 0)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return t.Day() == 29 && t.Month() == 2, nil
|
|
}
|
|
|
|
func isWeekDay(base interface{}) (bool, error) {
|
|
t, _, err := prepMod(base, 0)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return time.Sunday < t.Weekday() && t.Weekday() < time.Saturday, nil
|
|
}
|
|
|
|
func earliest(f interface{}, aa ...interface{}) (*time.Time, error) {
|
|
t, _, err := prepMod(f, 0)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, a := range aa {
|
|
s, _, err := prepMod(a, 0)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if (*s).Before(*t) {
|
|
t = s
|
|
}
|
|
}
|
|
|
|
return t, nil
|
|
}
|
|
|
|
func latest(f interface{}, aa ...interface{}) (*time.Time, error) {
|
|
t, _, err := prepMod(f, 0)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, a := range aa {
|
|
s, _, err := prepMod(a, 0)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if s.After(*t) {
|
|
t = s
|
|
}
|
|
}
|
|
|
|
return t, nil
|
|
}
|
|
|
|
func parseISOTime(s string) (time.Time, error) {
|
|
return time.Parse(time.RFC3339, s)
|
|
}
|
|
|
|
func modTime(base interface{}, mod interface{}) (*time.Time, error) {
|
|
var (
|
|
err error
|
|
d time.Duration
|
|
t *time.Time
|
|
)
|
|
|
|
t, _, err = prepMod(base, 0)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
switch c := mod.(type) {
|
|
case time.Duration:
|
|
d = c
|
|
case string:
|
|
d, err = time.ParseDuration(c)
|
|
}
|
|
|
|
if err != nil {
|
|
return t, err
|
|
}
|
|
|
|
tmp := t.Add(d)
|
|
return &tmp, nil
|
|
}
|
|
|
|
func modDate(base interface{}, mod interface{}) (*time.Time, error) {
|
|
t, m, err := prepMod(base, mod)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
tmp := t.AddDate(0, 0, m)
|
|
return &tmp, nil
|
|
}
|
|
|
|
func modWeek(base interface{}, mod interface{}) (*time.Time, error) {
|
|
t, m, err := prepMod(base, mod)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
tmp := t.AddDate(0, 0, 7*m)
|
|
return &tmp, nil
|
|
}
|
|
|
|
func modMonth(base interface{}, mod interface{}) (*time.Time, error) {
|
|
t, m, err := prepMod(base, mod)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
tmp := t.AddDate(0, m, 0)
|
|
return &tmp, nil
|
|
}
|
|
|
|
func modYear(base interface{}, mod interface{}) (*time.Time, error) {
|
|
t, m, err := prepMod(base, mod)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
tmp := t.AddDate(m, 0, 0)
|
|
return &tmp, nil
|
|
}
|
|
|
|
func prepMod(base interface{}, mod interface{}) (*time.Time, int, error) {
|
|
var (
|
|
t *time.Time
|
|
)
|
|
|
|
switch auxt := base.(type) {
|
|
case time.Time:
|
|
t = &auxt
|
|
case *time.Time:
|
|
t = auxt
|
|
default:
|
|
return nil, 0, errors.New("unexpected input type")
|
|
}
|
|
|
|
m, err := cast.ToIntE(mod)
|
|
if err != nil {
|
|
return nil, 0, err
|
|
}
|
|
|
|
return t, m, nil
|
|
}
|
|
|
|
// Strftime formats time with POSIX standard format
|
|
// More details here:
|
|
// https://github.com/lestrrat-go/strftime#supported-conversion-specifications
|
|
func strfTime(base interface{}, f string) (string, error) {
|
|
t, _, err := prepMod(base, 0)
|
|
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
o, _ := strftime.Format(f, *t,
|
|
strftime.WithMilliseconds('b'),
|
|
strftime.WithUnixSeconds('L'))
|
|
|
|
return o, nil
|
|
}
|
|
|
|
// sub returns difference between two date into milliseconds
|
|
func sub(from interface{}, to interface{}) (out int64, err error) {
|
|
t1, _, err := prepMod(from, 0)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
t2, _, err := prepMod(to, 0)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
var duration time.Duration
|
|
|
|
if t1.After(*t2) {
|
|
duration = t1.Sub(*t2)
|
|
} else {
|
|
return -1, errors.New("expecting 2nd input to be less than 1st input")
|
|
}
|
|
|
|
return duration.Milliseconds(), nil
|
|
}
|