Files
sqlite3/tests/time_test.go

255 lines
8.1 KiB
Go
Raw Permalink Normal View History

2023-03-07 14:19:22 +00:00
package tests
import (
2023-11-23 03:28:56 +00:00
"context"
2023-10-29 11:28:29 +00:00
"fmt"
2023-03-07 14:19:22 +00:00
"reflect"
"testing"
"time"
"github.com/ncruces/go-sqlite3"
2023-10-17 23:52:04 +01:00
"github.com/ncruces/go-sqlite3/driver"
2023-11-23 03:28:56 +00:00
_ "github.com/ncruces/go-sqlite3/embed"
2024-04-16 17:33:48 +01:00
_ "github.com/ncruces/go-sqlite3/tests/testcfg"
2023-03-07 14:19:22 +00:00
)
func TestTimeFormat_Encode(t *testing.T) {
t.Parallel()
reference := time.Date(2013, 10, 7, 4, 23, 19, 120_000_000, time.FixedZone("", -4*3600))
tests := []struct {
fmt sqlite3.TimeFormat
time time.Time
want any
}{
{sqlite3.TimeFormatDefault, reference, "2013-10-07T04:23:19.12-04:00"},
{sqlite3.TimeFormatJulianDay, reference, 2456572.849526851851852},
{sqlite3.TimeFormatUnix, reference, int64(1381134199)},
{sqlite3.TimeFormatUnixFrac, reference, 1381134199.120},
{sqlite3.TimeFormatUnixMilli, reference, int64(1381134199_120)},
{sqlite3.TimeFormatUnixMicro, reference, int64(1381134199_120000)},
{sqlite3.TimeFormatUnixNano, reference, int64(1381134199_120000000)},
{sqlite3.TimeFormat7, reference, "2013-10-07T08:23:19.120"},
}
for _, tt := range tests {
t.Run("", func(t *testing.T) {
if got := tt.fmt.Encode(tt.time); !reflect.DeepEqual(got, tt.want) {
t.Errorf("%q.Encode(%v) = %v, want %v", tt.fmt, tt.time, got, tt.want)
}
})
}
}
func TestTimeFormat_Decode(t *testing.T) {
t.Parallel()
2024-05-03 18:08:41 +01:00
const offset = -4 * 3600
zone := time.FixedZone("", offset)
2023-10-19 11:53:31 +01:00
reference := time.Date(2013, 10, 7, 4, 23, 19, 120_000_000, zone)
refnodate := time.Date(2000, 01, 1, 4, 23, 19, 120_000_000, zone)
2023-03-07 14:19:22 +00:00
tests := []struct {
fmt sqlite3.TimeFormat
val any
want time.Time
wantDelta time.Duration
2024-05-03 18:08:41 +01:00
wantOff int
2023-03-07 14:19:22 +00:00
wantErr bool
}{
2024-05-03 18:08:41 +01:00
{sqlite3.TimeFormatJulianDay, "2456572.849526851851852", reference, 0, 0, false},
{sqlite3.TimeFormatJulianDay, 2456572.849526851851852, reference, time.Millisecond, 0, false},
{sqlite3.TimeFormatJulianDay, int64(2456572), reference, 24 * time.Hour, 0, false},
{sqlite3.TimeFormatJulianDay, false, time.Time{}, 0, 0, true},
{sqlite3.TimeFormatUnix, "1381134199.120", reference, time.Microsecond, 0, false},
{sqlite3.TimeFormatUnix, 1381134199.120, reference, time.Microsecond, 0, false},
{sqlite3.TimeFormatUnix, int64(1381134199), reference, time.Second, 0, false},
{sqlite3.TimeFormatUnix, "abc", time.Time{}, 0, 0, true},
{sqlite3.TimeFormatUnix, false, time.Time{}, 0, 0, true},
{sqlite3.TimeFormatUnixMilli, "1381134199120", reference, 0, 0, false},
{sqlite3.TimeFormatUnixMilli, 1381134199.120e3, reference, 0, 0, false},
{sqlite3.TimeFormatUnixMilli, int64(1381134199_120), reference, 0, 0, false},
{sqlite3.TimeFormatUnixMilli, "abc", time.Time{}, 0, 0, true},
{sqlite3.TimeFormatUnixMilli, false, time.Time{}, 0, 0, true},
{sqlite3.TimeFormatUnixMicro, "1381134199120000", reference, 0, 0, false},
{sqlite3.TimeFormatUnixMicro, 1381134199.120e6, reference, 0, 0, false},
{sqlite3.TimeFormatUnixMicro, int64(1381134199_120000), reference, 0, 0, false},
{sqlite3.TimeFormatUnixMicro, "abc", time.Time{}, 0, 0, true},
{sqlite3.TimeFormatUnixMicro, false, time.Time{}, 0, 0, true},
{sqlite3.TimeFormatUnixNano, "1381134199120000000", reference, 0, 0, false},
{sqlite3.TimeFormatUnixNano, 1381134199.120e9, reference, 0, 0, false},
{sqlite3.TimeFormatUnixNano, int64(1381134199_120000000), reference, 0, 0, false},
{sqlite3.TimeFormatUnixNano, "abc", time.Time{}, 0, 0, true},
{sqlite3.TimeFormatUnixNano, false, time.Time{}, 0, 0, true},
{sqlite3.TimeFormatAuto, "2456572.849526851851852", reference, time.Millisecond, 0, false},
{sqlite3.TimeFormatAuto, "2456572", reference, 24 * time.Hour, 0, false},
{sqlite3.TimeFormatAuto, "1381134199.120", reference, time.Microsecond, 0, false},
{sqlite3.TimeFormatAuto, "1381134199.120e3", reference, time.Microsecond, 0, false},
{sqlite3.TimeFormatAuto, "1381134199.120e6", reference, time.Microsecond, 0, false},
{sqlite3.TimeFormatAuto, "1381134199.120e9", reference, time.Microsecond, 0, false},
{sqlite3.TimeFormatAuto, "1381134199", reference, time.Second, 0, false},
{sqlite3.TimeFormatAuto, "1381134199120", reference, 0, 0, false},
{sqlite3.TimeFormatAuto, "1381134199120000", reference, 0, 0, false},
{sqlite3.TimeFormatAuto, "1381134199120000000", reference, 0, 0, false},
{sqlite3.TimeFormatAuto, "2013-10-07 04:23:19.12-04:00", reference, 0, offset, false},
{sqlite3.TimeFormatAuto, "04:23:19.12-04:00", refnodate, 0, offset, false},
{sqlite3.TimeFormatAuto, "abc", time.Time{}, 0, 0, true},
{sqlite3.TimeFormatAuto, false, time.Time{}, 0, 0, true},
{sqlite3.TimeFormat3, "2013-10-07 04:23:19.12-04:00", reference, 0, offset, false},
{sqlite3.TimeFormat3, "2013-10-07 08:23:19.12", reference, 0, 0, false},
{sqlite3.TimeFormat9, "04:23:19.12-04:00", refnodate, 0, offset, false},
{sqlite3.TimeFormat9, "08:23:19.12", refnodate, 0, 0, false},
{sqlite3.TimeFormat3, false, time.Time{}, 0, 0, true},
{sqlite3.TimeFormat9, false, time.Time{}, 0, 0, true},
{sqlite3.TimeFormatDefault, "2013-10-07T04:23:19.12-04:00", reference, 0, offset, false},
{sqlite3.TimeFormatDefault, "2013-10-07T08:23:19.12Z", reference, 0, 0, false},
{sqlite3.TimeFormatDefault, false, time.Time{}, 0, 0, true},
2023-03-07 14:19:22 +00:00
}
for _, tt := range tests {
t.Run("", func(t *testing.T) {
got, err := tt.fmt.Decode(tt.val)
if (err != nil) != tt.wantErr {
t.Errorf("%q.Decode(%v) error = %v, wantErr %v", tt.fmt, tt.val, err, tt.wantErr)
return
}
2023-10-19 11:53:31 +01:00
if got.Sub(tt.want).Abs() > tt.wantDelta {
2023-03-07 14:19:22 +00:00
t.Errorf("%q.Decode(%v) = %v, want %v", tt.fmt, tt.val, got, tt.want)
}
2024-05-03 18:08:41 +01:00
if _, off := got.Zone(); off != tt.wantOff {
t.Errorf("%q.Decode(%v) = %v, want %v", tt.fmt, tt.val, off, tt.wantOff)
2023-10-19 11:53:31 +01:00
}
2023-03-07 14:19:22 +00:00
})
}
}
2023-03-10 16:25:07 +00:00
2023-10-17 23:52:04 +01:00
func TestTimeFormat_Scanner(t *testing.T) {
t.Parallel()
2023-11-23 03:28:56 +00:00
ctx, cancel := context.WithCancel(context.Background())
defer cancel()
2023-10-17 23:52:04 +01:00
db, err := driver.Open(":memory:", nil)
if err != nil {
t.Fatal(err)
}
defer db.Close()
2023-11-23 03:28:56 +00:00
conn, err := db.Conn(ctx)
if err != nil {
t.Fatal(err)
}
defer conn.Close()
2024-04-04 01:25:52 +01:00
_, err = conn.ExecContext(ctx, `CREATE TABLE test (col)`)
2023-10-17 23:52:04 +01:00
if err != nil {
t.Fatal(err)
}
reference := time.Date(2013, 10, 7, 4, 23, 19, 120_000_000, time.FixedZone("", -4*3600))
2023-11-23 03:28:56 +00:00
_, err = conn.ExecContext(ctx, `INSERT INTO test VALUES (?)`, sqlite3.TimeFormat7TZ.Encode(reference))
2023-10-17 23:52:04 +01:00
if err != nil {
t.Fatal(err)
}
var got time.Time
2023-11-23 03:28:56 +00:00
err = conn.QueryRowContext(ctx, "SELECT * FROM test").Scan(sqlite3.TimeFormatAuto.Scanner(&got))
2023-10-17 23:52:04 +01:00
if err != nil {
t.Fatal(err)
}
if !got.Equal(reference) {
t.Errorf("got %v, want %v", got, reference)
}
}
2023-03-10 16:25:07 +00:00
func TestDB_timeCollation(t *testing.T) {
t.Parallel()
db, err := sqlite3.Open(":memory:")
if err != nil {
t.Fatal(err)
}
defer db.Close()
2024-04-04 01:25:52 +01:00
err = db.Exec(`CREATE TABLE times (tstamp COLLATE TIME)`)
2023-03-10 16:25:07 +00:00
if err != nil {
t.Fatal(err)
}
stmt, _, err := db.Prepare(`INSERT INTO times VALUES (?), (?), (?)`)
if err != nil {
t.Fatal(err)
}
defer stmt.Close()
stmt.BindTime(1, time.Unix(0, 0).UTC(), sqlite3.TimeFormatDefault)
stmt.BindTime(2, time.Unix(0, -1).UTC(), sqlite3.TimeFormatDefault)
stmt.BindTime(3, time.Unix(0, +1).UTC(), sqlite3.TimeFormatDefault)
stmt.Step()
err = stmt.Close()
if err != nil {
t.Fatal(err)
}
stmt, _, err = db.Prepare(`SELECT tstamp FROM times ORDER BY tstamp`)
if err != nil {
t.Fatal(err)
}
var t0 time.Time
for stmt.Step() {
t1 := stmt.ColumnTime(0, sqlite3.TimeFormatAuto)
if t0.After(t1) {
t.Errorf("got %v after %v", t0, t1)
}
t0 = t1
}
err = stmt.Close()
if err != nil {
t.Fatal(err)
}
}
2023-10-29 11:28:29 +00:00
func TestDB_isoWeek(t *testing.T) {
t.Parallel()
db, err := sqlite3.Open(":memory:")
if err != nil {
t.Fatal(err)
}
defer db.Close()
stmt, _, err := db.Prepare(`SELECT strftime('%G-W%V-%u', ?)`)
if err != nil {
t.Fatal(err)
}
defer stmt.Close()
2024-01-22 09:38:39 +00:00
tend := time.Date(2500, 1, 1, 0, 0, 0, 0, time.UTC)
tstart := time.Date(1500, 1, 1, 12, 0, 0, 0, time.UTC)
for tm := tstart; tm.Before(tend); tm = tm.AddDate(0, 0, 1) {
2023-10-29 11:28:29 +00:00
stmt.BindTime(1, tm, sqlite3.TimeFormatDefault)
if stmt.Step() {
y, w := tm.ISOWeek()
d := tm.Weekday()
if d == 0 {
d = 7
}
want := fmt.Sprintf("%04d-W%02d-%d", y, w, d)
if got := stmt.ColumnText(0); got != want {
t.Errorf("got %q, want %q (%v)", got, want, tm)
}
}
stmt.Reset()
}
}