mirror of
https://github.com/MontFerret/ferret.git
synced 2025-01-16 03:21:03 +02:00
added DateCompare function (#182)
* add pkg/stdlib/objects Length function * rename lenght.go -> length.go * fix tests according to other tests * add new tests to length tests * delete objects method Length * add objects method Has * add objects function Keys * small fixes in Keys and Has functions * change Has function * unit tests for Keys function * add unit tests for merge. also little change in lib.go * add doc to Keys function * Merge function prototype * add unit tests for KEEP function * added KEEP function * added doc for KEYS function * update lib.go * update lib.go * upd merge prototype * addded isEqualObjects function to objects tests * change object method Compare * added unit tests for Compare method * changed Compare method * fix Compare method * rename method Clone to Copy * added Cloneable interface * added Value to Cloneable interface * implemented Cloneable intefrace by array * added some more unit tests for values.Array * fix values.Array.Compare method * added one more unit test * implemented Cloneable interface by Object * unit tests for Object.Clone * move core.IsCloneable to value.go * change Clone function * move IsClonable to package values * updated MERGE unit tests * added MERGE function * added MERGE to lib * added one more test * changed MERGE function * rewrite a few comments according to Go Best Practices * rewrite comments * fix bug when result of the KEEP function was dependent on source object * some more changes in KEEP function * init VALUES function * push test with bug * add stress test * small changes in stress tests * changes in object.Comapare * change object.Compare * add more tests for object.Compare * added comments to object.Compare function * change object.Comapare * delete useless comment * one more change in object.Compare * init datetime * added test for datetime * added lib.go * add helpers functions * made values.DefaultTimeLayout public * added DATE function * added DATE_DAYOFWEEK function * added DATE_YEAR function * added DATE_MONTH function * added one more testCase for DATE_MONTH * added DATE_DAY function * added DateDay to lib * added DATE_HOUR, DATE_MINUTE and DATE_SECOND functions * added DATE_DAYOFYEAR, DATE_LEAPYEAR, DATE_MILLISECOND functions * fix names in tests * one more case into dayofyear_test * added DATE_QUARTER function * added DATE_DAYS_IN_MONTH function * added DATE_FORMAT function * added -v flag into go test * update DATE_FORMAT test cases * added one more test case * add helpers functions * made values.DefaultTimeLayout public * added DATE function * added DATE_DAYOFWEEK function * added DATE_YEAR function * added DATE_MONTH function * added one more testCase for DATE_MONTH * added DATE_DAY function * added DateDay to lib * added DATE_HOUR, DATE_MINUTE and DATE_SECOND functions * added DATE_DAYOFYEAR, DATE_LEAPYEAR, DATE_MILLISECOND functions * fix names in tests * one more case into dayofyear_test * added DATE_QUARTER function * added DATE_DAYS_IN_MONTH function * added DATE_FORMAT function * added -v flag into go test * Set codecov support for all branches * update DATE_FORMAT test cases * Updated codecov settings * Added panic recovery mechanism (#158) * Bump github.com/mafredri/cdp from 0.19.0 to 0.20.0 (#159) Bumps [github.com/mafredri/cdp](https://github.com/mafredri/cdp) from 0.19.0 to 0.20.0. - [Release notes](https://github.com/mafredri/cdp/releases) - [Commits](https://github.com/mafredri/cdp/compare/v0.19.0...v0.20.0) Signed-off-by: dependabot[bot] <support@dependabot.com> * Bump github.com/gofrs/uuid from 3.1.1 to 3.1.2 (#160) Bumps [github.com/gofrs/uuid](https://github.com/gofrs/uuid) from 3.1.1 to 3.1.2. - [Release notes](https://github.com/gofrs/uuid/releases) - [Commits](https://github.com/gofrs/uuid/compare/v3.1.1...v3.1.2) Signed-off-by: dependabot[bot] <support@dependabot.com> * added one more test case * sorter instead Compare now * rename utils.LOG -> utils.PRINT * rename utils.Logs -> utils.Print * added DATE_ADD, DATE_SUBTRACT functions * use keyed fields now * added DATE_DIFF function * delete unused var * delete useless type cast * fixed a bug when adding/subtrating did not take an amount of units * added DateCompare function * renames * fix small bug * fix
This commit is contained in:
parent
1866cb1e31
commit
023040866d
@ -2,12 +2,9 @@ package datetime
|
||||
|
||||
import (
|
||||
"context"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/MontFerret/ferret/pkg/runtime/core"
|
||||
"github.com/MontFerret/ferret/pkg/runtime/values"
|
||||
"github.com/pkg/errors"
|
||||
)
|
||||
|
||||
var (
|
||||
@ -40,12 +37,14 @@ func DateAdd(_ context.Context, args ...core.Value) (core.Value, error) {
|
||||
return values.None, err
|
||||
}
|
||||
|
||||
dt, err := addUnit(date, int(amount), unit.String())
|
||||
u, err := UnitFromString(unit.String())
|
||||
if err != nil {
|
||||
return values.None, err
|
||||
}
|
||||
|
||||
return dt, nil
|
||||
tm := AddUnit(date.Time, int(amount), u)
|
||||
|
||||
return values.NewDateTime(tm), nil
|
||||
}
|
||||
|
||||
// DateSubtract subtract amount given in unit to date.
|
||||
@ -68,12 +67,14 @@ func DateSubtract(_ context.Context, args ...core.Value) (core.Value, error) {
|
||||
return values.None, err
|
||||
}
|
||||
|
||||
dt, err := addUnit(date, -1*int(amount), unit.String())
|
||||
u, err := UnitFromString(unit.String())
|
||||
if err != nil {
|
||||
return values.None, err
|
||||
}
|
||||
|
||||
return dt, nil
|
||||
tm := AddUnit(date.Time, -1*int(amount), u)
|
||||
|
||||
return values.NewDateTime(tm), nil
|
||||
}
|
||||
|
||||
func getArgs(args []core.Value) (values.DateTime, values.Int, values.String, error) {
|
||||
@ -97,25 +98,3 @@ func getArgs(args []core.Value) (values.DateTime, values.Int, values.String, err
|
||||
|
||||
return date, amount, unit, nil
|
||||
}
|
||||
|
||||
func addUnit(dt values.DateTime, amount int, unit string) (values.DateTime, error) {
|
||||
switch strings.ToLower(unit) {
|
||||
case "y", "year", "years":
|
||||
return values.NewDateTime(dt.AddDate(amount*1, 0, 0)), nil
|
||||
case "m", "month", "months":
|
||||
return values.NewDateTime(dt.AddDate(0, amount*1, 0)), nil
|
||||
case "w", "week", "weeks":
|
||||
return values.NewDateTime(dt.AddDate(0, 0, amount*7)), nil
|
||||
case "d", "day", "days":
|
||||
return values.NewDateTime(dt.AddDate(0, 0, amount*1)), nil
|
||||
case "h", "hour", "hours":
|
||||
return values.NewDateTime(dt.Add(time.Duration(amount) * time.Hour)), nil
|
||||
case "i", "minute", "minutes":
|
||||
return values.NewDateTime(dt.Add(time.Duration(amount) * time.Minute)), nil
|
||||
case "s", "second", "seconds":
|
||||
return values.NewDateTime(dt.Add(time.Duration(amount) * time.Second)), nil
|
||||
case "f", "millisecond", "milliseconds":
|
||||
return values.NewDateTime(dt.Add(time.Duration(amount) * time.Millisecond)), nil
|
||||
}
|
||||
return values.DateTime{}, errors.Errorf("no such unit '%s'", unit)
|
||||
}
|
||||
|
66
pkg/stdlib/datetime/compare.go
Normal file
66
pkg/stdlib/datetime/compare.go
Normal file
@ -0,0 +1,66 @@
|
||||
package datetime
|
||||
|
||||
import (
|
||||
"github.com/pkg/errors"
|
||||
|
||||
"context"
|
||||
|
||||
"github.com/MontFerret/ferret/pkg/runtime/core"
|
||||
"github.com/MontFerret/ferret/pkg/runtime/values"
|
||||
)
|
||||
|
||||
// DateCompare check if two partial dates match.
|
||||
// @params date1, date2 (DateTime) - comparable dates.
|
||||
// @params unitRangeStart (String) - unit to start from.
|
||||
// @params unitRangeEnd (String, Optional) - unit to end with.
|
||||
// Error will be returned if unitRangeStart unit less that unitRangeEnd.
|
||||
// @return (Boolean) - true if the dates match, else false.
|
||||
func DateCompare(_ context.Context, args ...core.Value) (core.Value, error) {
|
||||
err := core.ValidateArgs(args, 3, 4)
|
||||
if err != nil {
|
||||
return values.None, err
|
||||
}
|
||||
|
||||
err = core.ValidateValueTypePairs(
|
||||
core.PairValueType{Value: args[0], Types: sliceDateTime},
|
||||
core.PairValueType{Value: args[1], Types: sliceDateTime},
|
||||
core.PairValueType{Value: args[2], Types: sliceStringType},
|
||||
)
|
||||
if err != nil {
|
||||
return values.None, err
|
||||
}
|
||||
|
||||
date1 := args[0].(values.DateTime)
|
||||
date2 := args[1].(values.DateTime)
|
||||
rangeStart := args[2].(values.String)
|
||||
rangeEnd := values.NewString("millisecond")
|
||||
|
||||
if len(args) == 4 {
|
||||
if err = core.ValidateType(args[3], core.StringType); err != nil {
|
||||
return values.None, err
|
||||
}
|
||||
rangeEnd = args[3].(values.String)
|
||||
}
|
||||
|
||||
unitStart, err := UnitFromString(rangeStart.String())
|
||||
if err != nil {
|
||||
return values.None, err
|
||||
}
|
||||
|
||||
unitEnd, err := UnitFromString(rangeEnd.String())
|
||||
if err != nil {
|
||||
return values.None, err
|
||||
}
|
||||
|
||||
if unitStart < unitEnd {
|
||||
return values.None, errors.Errorf("start unit less that end unit")
|
||||
}
|
||||
|
||||
for u := unitEnd; u <= unitStart; u++ {
|
||||
if IsDatesEqual(date1.Time, date2.Time, u) {
|
||||
return values.NewBoolean(true), nil
|
||||
}
|
||||
}
|
||||
|
||||
return values.NewBoolean(false), nil
|
||||
}
|
107
pkg/stdlib/datetime/compare_test.go
Normal file
107
pkg/stdlib/datetime/compare_test.go
Normal file
@ -0,0 +1,107 @@
|
||||
package datetime_test
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/MontFerret/ferret/pkg/runtime/core"
|
||||
"github.com/MontFerret/ferret/pkg/runtime/values"
|
||||
"github.com/MontFerret/ferret/pkg/stdlib/datetime"
|
||||
)
|
||||
|
||||
func TestDateCompare(t *testing.T) {
|
||||
expectedTrue := values.NewBoolean(true)
|
||||
expectedFalse := values.NewBoolean(false)
|
||||
|
||||
tcs := []*testCase{
|
||||
&testCase{
|
||||
Name: "When less than 3 arguments",
|
||||
Expected: values.None,
|
||||
Args: []core.Value{values.NewInt(0), values.NewInt(0)},
|
||||
ShouldErr: true,
|
||||
},
|
||||
&testCase{
|
||||
Name: "When more than 4 arguments",
|
||||
Expected: values.None,
|
||||
Args: []core.Value{
|
||||
values.NewInt(0), values.NewInt(0), values.NewInt(0),
|
||||
values.NewInt(0), values.NewInt(0),
|
||||
},
|
||||
ShouldErr: true,
|
||||
},
|
||||
&testCase{
|
||||
Name: "when wrong type of arguments",
|
||||
Expected: values.None,
|
||||
Args: []core.Value{
|
||||
values.NewCurrentDateTime(),
|
||||
values.NewCurrentDateTime(),
|
||||
values.NewInt(0),
|
||||
},
|
||||
ShouldErr: true,
|
||||
},
|
||||
&testCase{
|
||||
Name: "when wrong type of optional argument",
|
||||
Expected: values.None,
|
||||
Args: []core.Value{
|
||||
values.NewCurrentDateTime(),
|
||||
values.NewCurrentDateTime(),
|
||||
values.NewString("year"),
|
||||
values.NewInt(0),
|
||||
},
|
||||
ShouldErr: true,
|
||||
},
|
||||
&testCase{
|
||||
Name: "when start unit less that end unit",
|
||||
Expected: values.None,
|
||||
Args: []core.Value{
|
||||
values.NewCurrentDateTime(),
|
||||
values.NewCurrentDateTime(),
|
||||
values.NewString("day"),
|
||||
values.NewString("year"),
|
||||
},
|
||||
ShouldErr: true,
|
||||
},
|
||||
&testCase{
|
||||
Name: "when years are equal",
|
||||
Expected: expectedTrue,
|
||||
Args: []core.Value{
|
||||
values.NewCurrentDateTime(),
|
||||
values.NewCurrentDateTime(),
|
||||
values.NewString("year"),
|
||||
},
|
||||
},
|
||||
&testCase{
|
||||
Name: "when years are not equal",
|
||||
Expected: expectedFalse,
|
||||
Args: []core.Value{
|
||||
mustLayoutDt("2006-01-02", "1999-02-07"),
|
||||
mustLayoutDt("2006-01-02", "2000-02-07"),
|
||||
values.NewString("year"),
|
||||
values.NewString("year"),
|
||||
},
|
||||
},
|
||||
&testCase{
|
||||
Name: "when months are equal",
|
||||
Expected: expectedTrue,
|
||||
Args: []core.Value{
|
||||
mustLayoutDt("2006-01-02", "1999-02-07"),
|
||||
mustLayoutDt("2006-01-02", "2000-02-09"),
|
||||
values.NewString("year"),
|
||||
values.NewString("days"),
|
||||
},
|
||||
},
|
||||
&testCase{
|
||||
Name: "when days are equal",
|
||||
Expected: expectedTrue,
|
||||
Args: []core.Value{
|
||||
values.NewCurrentDateTime(),
|
||||
values.NewCurrentDateTime(),
|
||||
values.NewString("days"),
|
||||
values.NewString("days"),
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
for _, tc := range tcs {
|
||||
tc.Do(t, datetime.DateCompare)
|
||||
}
|
||||
}
|
@ -2,11 +2,9 @@ package datetime
|
||||
|
||||
import (
|
||||
"context"
|
||||
"strings"
|
||||
|
||||
"github.com/MontFerret/ferret/pkg/runtime/core"
|
||||
"github.com/MontFerret/ferret/pkg/runtime/values"
|
||||
"github.com/pkg/errors"
|
||||
)
|
||||
|
||||
// DateDiff returns the difference between two dates in given time unit.
|
||||
@ -71,23 +69,9 @@ func DateDiff(_ context.Context, args ...core.Value) (core.Value, error) {
|
||||
}
|
||||
|
||||
func nsecToUnit(nsec float64, unit string) (float64, error) {
|
||||
switch strings.ToLower(unit) {
|
||||
case "y", "year", "years":
|
||||
return nsec / 31536e12, nil
|
||||
case "m", "month", "months":
|
||||
return nsec / 26784e11, nil
|
||||
case "w", "week", "weeks":
|
||||
return nsec / 6048e11, nil
|
||||
case "d", "day", "days":
|
||||
return nsec / 864e11, nil
|
||||
case "h", "hour", "hours":
|
||||
return nsec / 36e11, nil
|
||||
case "i", "minute", "minutes":
|
||||
return nsec / 6e10, nil
|
||||
case "s", "second", "seconds":
|
||||
return nsec / 1e9, nil
|
||||
case "f", "millisecond", "milliseconds":
|
||||
return nsec / 1e6, nil
|
||||
u, err := UnitFromString(unit)
|
||||
if err != nil {
|
||||
return -1, err
|
||||
}
|
||||
return -1, errors.Errorf("no such unit '%s'", unit)
|
||||
return nsec / u.Nanosecond(), nil
|
||||
}
|
||||
|
113
pkg/stdlib/datetime/unit.go
Normal file
113
pkg/stdlib/datetime/unit.go
Normal file
@ -0,0 +1,113 @@
|
||||
package datetime
|
||||
|
||||
import (
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/pkg/errors"
|
||||
)
|
||||
|
||||
// Unit specifies an unit of time (Millsecond, Second...).
|
||||
type Unit int
|
||||
|
||||
const (
|
||||
Millisecond Unit = iota
|
||||
Second
|
||||
Minute
|
||||
Hour
|
||||
Day
|
||||
Week
|
||||
Month
|
||||
Year
|
||||
)
|
||||
|
||||
var nanoseconds = []float64{
|
||||
1e6,
|
||||
1e9,
|
||||
6e10,
|
||||
36e11,
|
||||
864e11,
|
||||
6048e11,
|
||||
26784e11,
|
||||
31536e12,
|
||||
}
|
||||
|
||||
// Nanosecond returns representation of an Unit
|
||||
// in nanosconds
|
||||
func (u Unit) Nanosecond() float64 {
|
||||
return nanoseconds[u]
|
||||
}
|
||||
|
||||
// IsDatesEqual check if two partial dates match.
|
||||
// This case the day means not the amount of days in Time,
|
||||
// but the day of the month.
|
||||
// The same rules applied to each unit.
|
||||
func IsDatesEqual(tm1, tm2 time.Time, u Unit) bool {
|
||||
switch u {
|
||||
case Millisecond:
|
||||
tm1Msec := tm1.Nanosecond() / 1e6
|
||||
tm2Msec := tm2.Nanosecond() / 1e6
|
||||
return tm1Msec == tm2Msec
|
||||
case Second:
|
||||
return tm1.Second() == tm2.Second()
|
||||
case Minute:
|
||||
return tm1.Minute() == tm2.Minute()
|
||||
case Hour:
|
||||
return tm1.Hour() == tm2.Hour()
|
||||
case Day:
|
||||
return tm1.Day() == tm2.Day()
|
||||
case Week:
|
||||
tm1Wk := tm1.Day() / 7
|
||||
tm2Wk := tm2.Day() / 7
|
||||
return tm1Wk == tm2Wk
|
||||
case Month:
|
||||
return tm1.Month() == tm2.Month()
|
||||
case Year:
|
||||
return tm1.Year() == tm2.Year()
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
// AddUnit add amount given in u to tm
|
||||
func AddUnit(tm time.Time, amount int, u Unit) (res time.Time) {
|
||||
if u < Day {
|
||||
return tm.Add(time.Duration(amount) * time.Duration(int64(u.Nanosecond())))
|
||||
}
|
||||
|
||||
switch u {
|
||||
case Day:
|
||||
res = tm.AddDate(0, 0, amount*1)
|
||||
case Week:
|
||||
res = tm.AddDate(0, 0, amount*7)
|
||||
case Month:
|
||||
res = tm.AddDate(0, amount*1, 0)
|
||||
case Year:
|
||||
res = tm.AddDate(amount*1, 0, 0)
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
// UnitFromString returns true and an Unit object if
|
||||
// Unit with that name exists. Returns false, otherwise.
|
||||
func UnitFromString(s string) (Unit, error) {
|
||||
switch strings.ToLower(s) {
|
||||
case "y", "year", "years":
|
||||
return Year, nil
|
||||
case "m", "month", "months":
|
||||
return Month, nil
|
||||
case "w", "week", "weeks":
|
||||
return Week, nil
|
||||
case "d", "day", "days":
|
||||
return Day, nil
|
||||
case "h", "hour", "hours":
|
||||
return Hour, nil
|
||||
case "i", "minute", "minutes":
|
||||
return Minute, nil
|
||||
case "s", "second", "seconds":
|
||||
return Second, nil
|
||||
case "f", "millisecond", "milliseconds":
|
||||
return Millisecond, nil
|
||||
}
|
||||
return -1, errors.Errorf("no such unit '%s'", s)
|
||||
}
|
Loading…
Reference in New Issue
Block a user