package core_test import ( "context" "fmt" "testing" "time" "github.com/pocketbase/pocketbase/core" "github.com/pocketbase/pocketbase/tests" "github.com/pocketbase/pocketbase/tools/types" ) func TestDateFieldBaseMethods(t *testing.T) { testFieldBaseMethods(t, core.FieldTypeDate) } func TestDateFieldColumnType(t *testing.T) { app, _ := tests.NewTestApp() defer app.Cleanup() f := &core.DateField{} expected := "TEXT DEFAULT '' NOT NULL" if v := f.ColumnType(app); v != expected { t.Fatalf("Expected\n%q\ngot\n%q", expected, v) } } func TestDateFieldPrepareValue(t *testing.T) { app, _ := tests.NewTestApp() defer app.Cleanup() f := &core.DateField{} record := core.NewRecord(core.NewBaseCollection("test")) scenarios := []struct { raw any expected string }{ {"", ""}, {"invalid", ""}, {"2024-01-01 00:11:22.345Z", "2024-01-01 00:11:22.345Z"}, {time.Date(2024, 1, 2, 3, 4, 5, 0, time.UTC), "2024-01-02 03:04:05.000Z"}, } for i, s := range scenarios { t.Run(fmt.Sprintf("%d_%#v", i, s.raw), func(t *testing.T) { v, err := f.PrepareValue(record, s.raw) if err != nil { t.Fatal(err) } vDate, ok := v.(types.DateTime) if !ok { t.Fatalf("Expected types.DateTime instance, got %T", v) } if vDate.String() != s.expected { t.Fatalf("Expected %v, got %v", s.expected, v) } }) } } func TestDateFieldValidateValue(t *testing.T) { app, _ := tests.NewTestApp() defer app.Cleanup() collection := core.NewBaseCollection("test_collection") scenarios := []struct { name string field *core.DateField record func() *core.Record expectError bool }{ { "invalid raw value", &core.DateField{Name: "test"}, func() *core.Record { record := core.NewRecord(collection) record.SetRaw("test", 123) return record }, true, }, { "zero field value (not required)", &core.DateField{Name: "test"}, func() *core.Record { record := core.NewRecord(collection) record.SetRaw("test", types.DateTime{}) return record }, false, }, { "zero field value (required)", &core.DateField{Name: "test", Required: true}, func() *core.Record { record := core.NewRecord(collection) record.SetRaw("test", types.DateTime{}) return record }, true, }, { "non-zero field value (required)", &core.DateField{Name: "test", Required: true}, func() *core.Record { record := core.NewRecord(collection) record.SetRaw("test", types.NowDateTime()) return record }, false, }, } for _, s := range scenarios { t.Run(s.name, func(t *testing.T) { err := s.field.ValidateValue(context.Background(), app, s.record()) hasErr := err != nil if hasErr != s.expectError { t.Fatalf("Expected hasErr %v, got %v (%v)", s.expectError, hasErr, err) } }) } } func TestDateFieldValidateSettings(t *testing.T) { testDefaultFieldIdValidation(t, core.FieldTypeDate) testDefaultFieldNameValidation(t, core.FieldTypeDate) app, _ := tests.NewTestApp() defer app.Cleanup() collection := core.NewBaseCollection("test_collection") scenarios := []struct { name string field func() *core.DateField expectErrors []string }{ { "zero Min/Max", func() *core.DateField { return &core.DateField{ Id: "test", Name: "test", } }, []string{}, }, { "non-empty Min with empty Max", func() *core.DateField { return &core.DateField{ Id: "test", Name: "test", Min: types.NowDateTime(), } }, []string{}, }, { "empty Min non-empty Max", func() *core.DateField { return &core.DateField{ Id: "test", Name: "test", Max: types.NowDateTime(), } }, []string{}, }, { "Min = Max", func() *core.DateField { date := types.NowDateTime() return &core.DateField{ Id: "test", Name: "test", Min: date, Max: date, } }, []string{}, }, { "Min > Max", func() *core.DateField { min := types.NowDateTime() max := min.Add(-5 * time.Second) return &core.DateField{ Id: "test", Name: "test", Min: min, Max: max, } }, []string{}, }, { "Min < Max", func() *core.DateField { max := types.NowDateTime() min := max.Add(-5 * time.Second) return &core.DateField{ Id: "test", Name: "test", Min: min, Max: max, } }, []string{}, }, } for _, s := range scenarios { t.Run(s.name, func(t *testing.T) { errs := s.field().ValidateSettings(context.Background(), app, collection) tests.TestValidationErrors(t, errs, s.expectErrors) }) } }