1
0
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:
3timeslazy 2018-11-20 03:49:34 +03:00 committed by Tim Voronov
parent 1866cb1e31
commit 023040866d
5 changed files with 298 additions and 49 deletions

View File

@ -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)
}

View 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
}

View 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)
}
}

View File

@ -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
View 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)
}