minor improvement of the x/jsonx.SimpleDate time parser

more features designed for the past 2-3 months to come, this is just a hotfix
This commit is contained in:
Gerasimos (Makis) Maropoulos 2024-04-08 21:39:45 +03:00
parent 384ca4db70
commit c2238c71b8
No known key found for this signature in database
GPG Key ID: D6032D1840F48BEC
2 changed files with 66 additions and 8 deletions

View File

@ -10,12 +10,17 @@ import (
"github.com/kataras/iris/v12/x/timex"
)
const (
// SimpleDateLayout represents the "year-month-day" Go time format.
const SimpleDateLayout = "2006-01-02"
SimpleDateLayout = "2006-01-02"
simpleDateLayoutPostgres = "2006-1-2"
)
// SimpleDate holds a json "year-month-day" time.
type SimpleDate time.Time
var _ Exampler = (*SimpleDate)(nil)
// SimpleDateFromTime accepts a "t" Time and returns
// a SimpleDate. If format fails, it returns the zero value of time.Time.
func SimpleDateFromTime(t time.Time) SimpleDate {
@ -24,6 +29,10 @@ func SimpleDateFromTime(t time.Time) SimpleDate {
}
// ParseSimpleDate reads from "s" and returns the SimpleDate time.
//
// The function supports the following formats:
// - "2024-01-01"
// - "2024-1-1"
func ParseSimpleDate(s string) (SimpleDate, error) {
if s == "" || s == "null" {
return SimpleDate{}, nil
@ -36,10 +45,16 @@ func ParseSimpleDate(s string) (SimpleDate, error) {
tt, err = time.Parse(SimpleDateLayout, s)
if err != nil {
return SimpleDate{}, err
// After v5.0.0-alpha.3 of pgx this is coming as "1993-1-1" instead of the stored
// value "1993-01-01".
var err2 error
tt, err2 = time.Parse(simpleDateLayoutPostgres, s)
if err2 != nil {
return SimpleDate{}, fmt.Errorf("%s: %w", err2.Error(), err)
}
}
return SimpleDate(tt.UTC()), nil
return SimpleDate(tt), nil
}
// UnmarshalJSON binds the json "data" to "t" with the `SimpleDateLayout`.
@ -49,11 +64,10 @@ func (t *SimpleDate) UnmarshalJSON(data []byte) error {
}
data = trimQuotes(data)
if len(data) == 0 {
return nil // as an excepption here, allow empty "" on simple dates, as the server would render it on a response.
}
dataStr := string(data)
if len(dataStr) == 0 {
return nil // do not allow empty "" on simple dates.
}
tt, err := time.Parse(SimpleDateLayout, dataStr)
if err != nil {
@ -74,6 +88,14 @@ func (t SimpleDate) MarshalJSON() ([]byte, error) {
return emptyQuoteBytes, nil
}
// Examples returns a list of example values.
func (t SimpleDate) ListExamples() any {
return []string{
"2024-01-01",
"2024-1-1",
}
}
// IsZero reports whether "t" is zero time.
// It completes the pg.Zeroer interface.
func (t SimpleDate) IsZero() bool {
@ -85,6 +107,7 @@ func (t SimpleDate) Add(d time.Duration) SimpleDate {
return SimpleDateFromTime(t.ToTime().Add(d))
}
// CountPastDays returns the count of days between "t" and "pastDate".
func (t SimpleDate) CountPastDays(pastDate SimpleDate) int {
t1, t2 := t.ToTime(), pastDate.ToTime()
return int(t1.Sub(t2).Hours() / 24)
@ -113,6 +136,7 @@ func (t SimpleDate) ToTime() time.Time {
return time.Time(t)
}
// Value completes the pg and native sql driver.Valuer interface.
func (t SimpleDate) Value() (driver.Value, error) {
return t.String(), nil
}

View File

@ -57,3 +57,37 @@ func TestSimpleDateAterBefore(t *testing.T) {
t.Fatalf("[after] expected d2 to be after d1")
}
}
func TestCountPastDays(t *testing.T) {
tests := []struct {
AfterDate SimpleDate
BeforeDate SimpleDate
ExpectedPastDays int
}{
{
AfterDate: mustParseSimpleDate("2023-01-01"),
BeforeDate: mustParseSimpleDate("2022-12-31"),
ExpectedPastDays: 1,
},
{
AfterDate: mustParseSimpleDate("2023-01-01"),
BeforeDate: mustParseSimpleDate("2022-01-01"),
ExpectedPastDays: 365,
},
}
for i, tt := range tests {
if expected, got := tt.ExpectedPastDays, tt.AfterDate.CountPastDays(tt.BeforeDate); expected != got {
t.Fatalf("[%d] expected past days count: %d but got: %d", i, expected, got)
}
}
}
func mustParseSimpleDate(s string) SimpleDate {
d, err := ParseSimpleDate(s)
if err != nil {
panic(err)
}
return d
}