Skip to content

Commit b391cca

Browse files
author
Danny Hermes
committed
Add go-date package source.
1 parent e9095c6 commit b391cca

16 files changed

+1570
-2
lines changed

README.md

+45-2
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,45 @@
1-
# go-date
2-
Native type for dealing with dates in Go
1+
# `go-date`
2+
3+
[![GoDoc][1]][2]
4+
[![Go ReportCard][3]][4]
5+
6+
The `go-date` package provides a dedicated `Date{}` struct to emulate the
7+
standard library `time.Time{}` behavior.
8+
9+
## API
10+
11+
This package provides helpers for:
12+
13+
- conversion: `ToTime()`, `date.FromTime()`, `date.FromString()`
14+
- serialization: JSON and SQL
15+
- emulating `time.Time{}`: `After()`, `Before()`, `Sub()`, etc.
16+
- explicit null handling: `NullDate{}` and an analog of `sql.NullTime{}`
17+
- emulating `time` helpers: `Today()` as an analog of `time.Now()`
18+
19+
## Background
20+
21+
The Go standard library contains no native type for dates without times.
22+
Instead, common convention is to use a `time.Time{}` with only the year, month,
23+
and day set. For example, this convention is followed when a timestamp of the
24+
form YYYY-MM-DD is parsed via `time.Parse(time.DateOnly, s)`.
25+
26+
## Alternatives
27+
28+
This package is intended to be simple to understand and only needs to cover
29+
"modern" dates (i.e. dates between 1900 and 2100). As a result, the core
30+
`Date{}` struct directly exposes the year, month, and day as fields.
31+
32+
There are several alternative date packages which cover wider date ranges.
33+
(These packages all use the [proleptic Gregorian calendar][6] to cover the
34+
historical date ranges.) Some existing packages:
35+
36+
- `github.com/fxtlabs/date` [package][7]
37+
- `github.com/rickb777/date` [package][5]
38+
39+
[1]: https://godoc.org/github.com/hardfinhq/go-date?status.svg
40+
[2]: http://godoc.org/github.com/hardfinhq/go-date
41+
[3]: https://goreportcard.com/badge/hardfinhq/go-date
42+
[4]: https://goreportcard.com/report/hardfinhq/go-date
43+
[5]: https://pkg.go.dev/github.com/rickb777/date
44+
[6]: https://en.wikipedia.org/wiki/Proleptic_Gregorian_calendar
45+
[7]: https://pkg.go.dev/github.com/fxtlabs/date

convert.go

+95
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,95 @@
1+
// Copyright 2024 Hardfin, Inc.
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License");
4+
// you may not use this file except in compliance with the License.
5+
// You may obtain a copy of the License at
6+
//
7+
// https://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS,
11+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
// See the License for the specific language governing permissions and
13+
// limitations under the License.
14+
15+
package date
16+
17+
import (
18+
"database/sql"
19+
"fmt"
20+
"time"
21+
)
22+
23+
// ConvertConfig helps customize the behavior of conversion functions like
24+
// `NullTimeFromPtr()`.
25+
type ConvertConfig struct {
26+
Timezone *time.Location
27+
}
28+
29+
// ConvertOption defines a function that will be applied to a convert config.
30+
type ConvertOption func(*ConvertConfig)
31+
32+
// OptConvertTimezone returns an option that sets the timezone on a convert
33+
// config.
34+
func OptConvertTimezone(tz *time.Location) ConvertOption {
35+
return func(cc *ConvertConfig) {
36+
cc.Timezone = tz
37+
}
38+
}
39+
40+
// NullDateFromPtr converts a `Date` pointer into a `NullDate`.
41+
func NullDateFromPtr(d *Date) NullDate {
42+
if d == nil {
43+
return NullDate{Valid: false}
44+
}
45+
46+
return NullDate{Date: *d, Valid: true}
47+
}
48+
49+
// NullTimeFromPtr converts a date to a native Go `sql.NullTime`; the
50+
// convention in Go is that a **date-only** is parsed (via `time.DateOnly`) as
51+
// `time.Date(YYYY, MM, DD, 0, 0, 0, 0, time.UTC)`.
52+
func NullTimeFromPtr(d *Date, opts ...ConvertOption) sql.NullTime {
53+
if d == nil {
54+
return sql.NullTime{Valid: false}
55+
}
56+
57+
t := d.ToTime(opts...)
58+
return sql.NullTime{Time: t, Valid: true}
59+
}
60+
61+
// FromString parses a string of the form YYYY-MM-DD into a `Date{}`.
62+
func FromString(s string) (Date, error) {
63+
t, err := time.Parse(time.DateOnly, s)
64+
if err != nil {
65+
return Date{}, err
66+
}
67+
68+
year, month, day := t.Date()
69+
d := Date{Year: year, Month: month, Day: day}
70+
return d, nil
71+
}
72+
73+
// FromTime validates that a `time.Time{}` contains a date and converts it to a
74+
// `Date{}`.
75+
func FromTime(t time.Time) (Date, error) {
76+
if t.Hour() != 0 ||
77+
t.Minute() != 0 ||
78+
t.Second() != 0 ||
79+
t.Nanosecond() != 0 ||
80+
t.Location() != time.UTC {
81+
return Date{}, fmt.Errorf("timestamp contains more than just date information; %s", t.Format(time.RFC3339Nano))
82+
}
83+
84+
year, month, day := t.Date()
85+
d := Date{Year: year, Month: month, Day: day}
86+
return d, nil
87+
}
88+
89+
// InTimezone translates a timestamp into a timezone and then captures the date
90+
// in that timezone.
91+
func InTimezone(t time.Time, tz *time.Location) Date {
92+
tLocal := t.In(tz)
93+
year, month, day := tLocal.Date()
94+
return Date{Year: year, Month: month, Day: day}
95+
}

convert_test.go

+150
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,150 @@
1+
// Copyright 2024 Hardfin, Inc.
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License");
4+
// you may not use this file except in compliance with the License.
5+
// You may obtain a copy of the License at
6+
//
7+
// https://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS,
11+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
// See the License for the specific language governing permissions and
13+
// limitations under the License.
14+
15+
package date_test
16+
17+
import (
18+
"database/sql"
19+
"fmt"
20+
"testing"
21+
"time"
22+
23+
testifyrequire "github.com/stretchr/testify/require"
24+
25+
date "github.com/hardfinhq/go-date"
26+
)
27+
28+
func TestNullDateFromPtr(t *testing.T) {
29+
t.Parallel()
30+
assert := testifyrequire.New(t)
31+
32+
d1 := &date.Date{Year: 2000, Month: time.January, Day: 1}
33+
nd1 := date.NullDateFromPtr(d1)
34+
expected := date.NullDate{Date: *d1, Valid: true}
35+
assert.Equal(expected, nd1)
36+
37+
var d2 *date.Date
38+
nd2 := date.NullDateFromPtr(d2)
39+
expected = date.NullDate{Valid: false}
40+
assert.Equal(expected, nd2)
41+
}
42+
43+
func TestNullTimeFromPtr(t *testing.T) {
44+
t.Parallel()
45+
assert := testifyrequire.New(t)
46+
47+
var d *date.Date
48+
nt := date.NullTimeFromPtr(d)
49+
expected := sql.NullTime{Valid: false}
50+
assert.Equal(expected, nt)
51+
52+
d = &date.Date{Year: 2000, Month: time.January, Day: 1}
53+
nt = date.NullTimeFromPtr(d)
54+
expected = sql.NullTime{Time: time.Date(2000, time.January, 1, 0, 0, 0, 0, time.UTC), Valid: true}
55+
assert.Equal(expected, nt)
56+
57+
tz, err := time.LoadLocation("America/Chicago")
58+
assert.Nil(err)
59+
nt = date.NullTimeFromPtr(d, date.OptConvertTimezone(tz))
60+
expected = sql.NullTime{Time: time.Date(2000, time.January, 1, 0, 0, 0, 0, tz), Valid: true}
61+
assert.Equal(expected, nt)
62+
}
63+
64+
func TestFromTime(base *testing.T) {
65+
base.Parallel()
66+
67+
type testCase struct {
68+
Time string
69+
Date date.Date
70+
Error string
71+
}
72+
73+
cases := []testCase{
74+
{
75+
Time: "2020-05-11T07:10:55.209309302Z",
76+
Error: "timestamp contains more than just date information; 2020-05-11T07:10:55.209309302Z",
77+
},
78+
{
79+
Time: "2022-01-31T00:00:00.000Z",
80+
Date: date.Date{Year: 2022, Month: time.January, Day: 31},
81+
},
82+
{
83+
Time: "2022-01-31T00:00:00.000-05:00",
84+
Error: "timestamp contains more than just date information; 2022-01-31T00:00:00-05:00",
85+
},
86+
}
87+
88+
for i := range cases {
89+
// NOTE: Assign to loop-local (instead of declaring the `tc` variable in
90+
// `range`) to avoid capturing reference to loop variable.
91+
tc := cases[i]
92+
base.Run(tc.Time, func(t *testing.T) {
93+
t.Parallel()
94+
assert := testifyrequire.New(t)
95+
96+
timestamp, err := time.Parse(time.RFC3339Nano, tc.Time)
97+
assert.Nil(err)
98+
99+
d, err := date.FromTime(timestamp)
100+
if tc.Error == "" {
101+
assert.Nil(err)
102+
assert.Equal(tc.Date, d)
103+
} else {
104+
assert.Equal(tc.Error, fmt.Sprintf("%v", err))
105+
assert.Equal(date.Date{}, d)
106+
}
107+
})
108+
}
109+
}
110+
111+
func TestInTimezone(base *testing.T) {
112+
base.Parallel()
113+
114+
type testCase struct {
115+
Time string
116+
Timezone string
117+
Date string
118+
}
119+
120+
cases := []testCase{
121+
{Time: "2024-02-01T06:41:35.540349Z", Timezone: "America/Los_Angeles", Date: "2024-01-31"},
122+
{Time: "2024-02-01T06:41:35.540349Z", Timezone: "America/Denver", Date: "2024-01-31"},
123+
{Time: "2024-02-01T06:41:35.540349Z", Timezone: "America/Chicago", Date: "2024-02-01"},
124+
{Time: "2024-02-01T06:41:35.540349Z", Timezone: "America/New_York", Date: "2024-02-01"},
125+
{Time: "2024-02-01T06:41:35.540349Z", Timezone: "UTC", Date: "2024-02-01"},
126+
}
127+
128+
for i := range cases {
129+
// NOTE: Assign to loop-local (instead of declaring the `tc` variable in
130+
// `range`) to avoid capturing reference to loop variable.
131+
tc := cases[i]
132+
description := fmt.Sprintf("%s::%s", tc.Time, tc.Timezone)
133+
base.Run(description, func(t *testing.T) {
134+
t.Parallel()
135+
assert := testifyrequire.New(t)
136+
137+
timestamp, err := time.Parse(time.RFC3339Nano, tc.Time)
138+
assert.Nil(err)
139+
140+
tz, err := time.LoadLocation(tc.Timezone)
141+
assert.Nil(err)
142+
143+
expected, err := date.FromString(tc.Date)
144+
assert.Nil(err)
145+
146+
d := date.InTimezone(timestamp, tz)
147+
assert.Equal(expected, d)
148+
})
149+
}
150+
}

0 commit comments

Comments
 (0)