package daos_test import ( "encoding/json" "errors" "strings" "testing" "github.com/pocketbase/dbx" "github.com/pocketbase/pocketbase/daos" "github.com/pocketbase/pocketbase/models" "github.com/pocketbase/pocketbase/models/schema" "github.com/pocketbase/pocketbase/tests" "github.com/pocketbase/pocketbase/tools/list" "github.com/pocketbase/pocketbase/tools/types" ) func TestCollectionQuery(t *testing.T) { t.Parallel() app, _ := tests.NewTestApp() defer app.Cleanup() expected := "SELECT {{_collections}}.* FROM `_collections`" sql := app.Dao().CollectionQuery().Build().SQL() if sql != expected { t.Errorf("Expected sql %s, got %s", expected, sql) } } func TestFindCollectionsByType(t *testing.T) { t.Parallel() app, _ := tests.NewTestApp() defer app.Cleanup() scenarios := []struct { collectionType string expectError bool expectTotal int }{ {"", false, 0}, {"unknown", false, 0}, {models.CollectionTypeAuth, false, 3}, {models.CollectionTypeBase, false, 5}, } for i, scenario := range scenarios { collections, err := app.Dao().FindCollectionsByType(scenario.collectionType) hasErr := err != nil if hasErr != scenario.expectError { t.Errorf("[%d] Expected hasErr to be %v, got %v (%v)", i, scenario.expectError, hasErr, err) } if len(collections) != scenario.expectTotal { t.Errorf("[%d] Expected %d collections, got %d", i, scenario.expectTotal, len(collections)) } for _, c := range collections { if c.Type != scenario.collectionType { t.Errorf("[%d] Expected collection with type %s, got %s: \n%v", i, scenario.collectionType, c.Type, c) } } } } func TestFindCollectionByNameOrId(t *testing.T) { t.Parallel() app, _ := tests.NewTestApp() defer app.Cleanup() scenarios := []struct { nameOrId string expectError bool }{ {"", true}, {"missing", true}, {"wsmn24bux7wo113", false}, {"demo1", false}, {"DEMO1", false}, // case insensitive check } for i, scenario := range scenarios { model, err := app.Dao().FindCollectionByNameOrId(scenario.nameOrId) hasErr := err != nil if hasErr != scenario.expectError { t.Errorf("[%d] Expected hasErr to be %v, got %v (%v)", i, scenario.expectError, hasErr, err) } if model != nil && model.Id != scenario.nameOrId && !strings.EqualFold(model.Name, scenario.nameOrId) { t.Errorf("[%d] Expected model with identifier %s, got %v", i, scenario.nameOrId, model) } } } func TestIsCollectionNameUnique(t *testing.T) { t.Parallel() app, _ := tests.NewTestApp() defer app.Cleanup() scenarios := []struct { name string excludeId string expected bool }{ {"", "", false}, {"demo1", "", false}, {"Demo1", "", false}, {"new", "", true}, {"demo1", "wsmn24bux7wo113", true}, } for i, scenario := range scenarios { result := app.Dao().IsCollectionNameUnique(scenario.name, scenario.excludeId) if result != scenario.expected { t.Errorf("[%d] Expected %v, got %v", i, scenario.expected, result) } } } func TestFindCollectionReferences(t *testing.T) { t.Parallel() app, _ := tests.NewTestApp() defer app.Cleanup() collection, err := app.Dao().FindCollectionByNameOrId("demo3") if err != nil { t.Fatal(err) } result, err := app.Dao().FindCollectionReferences( collection, collection.Id, // test whether "nonempty" exclude ids condition will be skipped "", "", ) if err != nil { t.Fatal(err) } if len(result) != 1 { t.Fatalf("Expected 1 collection, got %d: %v", len(result), result) } expectedFields := []string{ "rel_one_no_cascade", "rel_one_no_cascade_required", "rel_one_cascade", "rel_many_no_cascade", "rel_many_no_cascade_required", "rel_many_cascade", } for col, fields := range result { if col.Name != "demo4" { t.Fatalf("Expected collection demo4, got %s", col.Name) } if len(fields) != len(expectedFields) { t.Fatalf("Expected fields %v, got %v", expectedFields, fields) } for i, f := range fields { if !list.ExistInSlice(f.Name, expectedFields) { t.Fatalf("[%d] Didn't expect field %v", i, f) } } } } func TestDeleteCollection(t *testing.T) { t.Parallel() app, _ := tests.NewTestApp() defer app.Cleanup() colUnsaved := &models.Collection{} colAuth, err := app.Dao().FindCollectionByNameOrId("users") if err != nil { t.Fatal(err) } colReferenced, err := app.Dao().FindCollectionByNameOrId("demo2") if err != nil { t.Fatal(err) } colSystem, err := app.Dao().FindCollectionByNameOrId("demo3") if err != nil { t.Fatal(err) } colSystem.System = true if err := app.Dao().Save(colSystem); err != nil { t.Fatal(err) } colBase, err := app.Dao().FindCollectionByNameOrId("demo1") if err != nil { t.Fatal(err) } colView1, err := app.Dao().FindCollectionByNameOrId("view1") if err != nil { t.Fatal(err) } colView2, err := app.Dao().FindCollectionByNameOrId("view2") if err != nil { t.Fatal(err) } scenarios := []struct { model *models.Collection expectError bool }{ {colUnsaved, true}, {colReferenced, true}, {colSystem, true}, {colBase, true}, // depend on view1, view2 and view2 {colView1, true}, // view2 depend on it {colView2, false}, {colView1, false}, // no longer has dependent collections {colBase, false}, // no longer has dependent views {colAuth, false}, // should delete also its related external auths } for i, s := range scenarios { err := app.Dao().DeleteCollection(s.model) hasErr := err != nil if hasErr != s.expectError { t.Errorf("[%d] Expected hasErr %v, got %v (%v)", i, s.expectError, hasErr, err) continue } if hasErr { continue } if app.Dao().HasTable(s.model.Name) { t.Errorf("[%d] Expected table/view %s to be deleted", i, s.model.Name) } // check if the external auths were deleted if s.model.IsAuth() { var total int err := app.Dao().ExternalAuthQuery(). Select("count(*)"). AndWhere(dbx.HashExp{"collectionId": s.model.Id}). Row(&total) if err != nil || total > 0 { t.Fatalf("[%d] Expected external auths to be deleted, got %v (%v)", i, total, err) } } } } func TestSaveCollectionCreate(t *testing.T) { t.Parallel() app, _ := tests.NewTestApp() defer app.Cleanup() collection := &models.Collection{ Name: "new_test", Type: models.CollectionTypeBase, Schema: schema.NewSchema( &schema.SchemaField{ Type: schema.FieldTypeText, Name: "test", }, ), } err := app.Dao().SaveCollection(collection) if err != nil { t.Fatal(err) } if collection.Id == "" { t.Fatal("Expected collection id to be set") } // check if the records table was created hasTable := app.Dao().HasTable(collection.Name) if !hasTable { t.Fatalf("Expected records table %s to be created", collection.Name) } // check if the records table has the schema fields columns, err := app.Dao().TableColumns(collection.Name) if err != nil { t.Fatal(err) } expectedColumns := []string{"id", "created", "updated", "test"} if len(columns) != len(expectedColumns) { t.Fatalf("Expected columns %v, got %v", expectedColumns, columns) } for i, c := range columns { if !list.ExistInSlice(c, expectedColumns) { t.Fatalf("[%d] Didn't expect record column %s", i, c) } } } func TestSaveCollectionUpdate(t *testing.T) { t.Parallel() app, _ := tests.NewTestApp() defer app.Cleanup() collection, err := app.Dao().FindCollectionByNameOrId("demo3") if err != nil { t.Fatal(err) } // rename an existing schema field and add a new one oldField := collection.Schema.GetFieldByName("title") oldField.Name = "title_update" collection.Schema.AddField(&schema.SchemaField{ Type: schema.FieldTypeText, Name: "test", }) saveErr := app.Dao().SaveCollection(collection) if saveErr != nil { t.Fatal(saveErr) } // check if the records table has the schema fields expectedColumns := []string{"id", "created", "updated", "title_update", "test", "files"} columns, err := app.Dao().TableColumns(collection.Name) if err != nil { t.Fatal(err) } if len(columns) != len(expectedColumns) { t.Fatalf("Expected columns %v, got %v", expectedColumns, columns) } for i, c := range columns { if !list.ExistInSlice(c, expectedColumns) { t.Fatalf("[%d] Didn't expect record column %s", i, c) } } } // indirect update of a field used in view should cause view(s) update func TestSaveCollectionIndirectViewsUpdate(t *testing.T) { t.Parallel() app, _ := tests.NewTestApp() defer app.Cleanup() collection, err := app.Dao().FindCollectionByNameOrId("demo1") if err != nil { t.Fatal(err) } // update MaxSelect fields { relMany := collection.Schema.GetFieldByName("rel_many") relManyOpt := relMany.Options.(*schema.RelationOptions) relManyOpt.MaxSelect = types.Pointer(1) fileOne := collection.Schema.GetFieldByName("file_one") fileOneOpt := fileOne.Options.(*schema.FileOptions) fileOneOpt.MaxSelect = 10 if err := app.Dao().SaveCollection(collection); err != nil { t.Fatal(err) } } // check view1 schema { view1, err := app.Dao().FindCollectionByNameOrId("view1") if err != nil { t.Fatal(err) } relMany := view1.Schema.GetFieldByName("rel_many") relManyOpt := relMany.Options.(*schema.RelationOptions) if relManyOpt.MaxSelect == nil || *relManyOpt.MaxSelect != 1 { t.Fatalf("Expected view1.rel_many MaxSelect to be %d, got %v", 1, relManyOpt.MaxSelect) } fileOne := view1.Schema.GetFieldByName("file_one") fileOneOpt := fileOne.Options.(*schema.FileOptions) if fileOneOpt.MaxSelect != 10 { t.Fatalf("Expected view1.file_one MaxSelect to be %d, got %v", 10, fileOneOpt.MaxSelect) } } // check view2 schema { view2, err := app.Dao().FindCollectionByNameOrId("view2") if err != nil { t.Fatal(err) } relMany := view2.Schema.GetFieldByName("rel_many") relManyOpt := relMany.Options.(*schema.RelationOptions) if relManyOpt.MaxSelect == nil || *relManyOpt.MaxSelect != 1 { t.Fatalf("Expected view2.rel_many MaxSelect to be %d, got %v", 1, relManyOpt.MaxSelect) } } } func TestSaveCollectionViewWrapping(t *testing.T) { t.Parallel() viewName := "test_wrapping" scenarios := []struct { name string query string expected string }{ { "no wrapping - text field", "select text as id, bool from demo1", "CREATE VIEW `test_wrapping` AS SELECT * FROM (select text as id, bool from demo1)", }, { "no wrapping - id field", "select text as id, bool from demo1", "CREATE VIEW `test_wrapping` AS SELECT * FROM (select text as id, bool from demo1)", }, { "no wrapping - relation field", "select rel_one as id, bool from demo1", "CREATE VIEW `test_wrapping` AS SELECT * FROM (select rel_one as id, bool from demo1)", }, { "no wrapping - select field", "select select_many as id, bool from demo1", "CREATE VIEW `test_wrapping` AS SELECT * FROM (select select_many as id, bool from demo1)", }, { "no wrapping - email field", "select email as id, bool from demo1", "CREATE VIEW `test_wrapping` AS SELECT * FROM (select email as id, bool from demo1)", }, { "no wrapping - datetime field", "select datetime as id, bool from demo1", "CREATE VIEW `test_wrapping` AS SELECT * FROM (select datetime as id, bool from demo1)", }, { "no wrapping - url field", "select url as id, bool from demo1", "CREATE VIEW `test_wrapping` AS SELECT * FROM (select url as id, bool from demo1)", }, { "wrapping - bool field", "select bool as id, text as txt, url from demo1", "CREATE VIEW `test_wrapping` AS SELECT * FROM (SELECT cast(`id` as text) `id`,`txt`,`url` FROM (select bool as id, text as txt, url from demo1))", }, { "wrapping - bool field (different order)", "select text as txt, url, bool as id from demo1", "CREATE VIEW `test_wrapping` AS SELECT * FROM (SELECT `txt`,`url`,cast(`id` as text) `id` FROM (select text as txt, url, bool as id from demo1))", }, { "wrapping - json field", "select json as id, text, url from demo1", "CREATE VIEW `test_wrapping` AS SELECT * FROM (SELECT cast(`id` as text) `id`,`text`,`url` FROM (select json as id, text, url from demo1))", }, { "wrapping - numeric id", "select 1 as id", "CREATE VIEW `test_wrapping` AS SELECT * FROM (SELECT cast(`id` as text) `id` FROM (select 1 as id))", }, { "wrapping - expresion", "select ('test') as id", "CREATE VIEW `test_wrapping` AS SELECT * FROM (SELECT cast(`id` as text) `id` FROM (select ('test') as id))", }, { "no wrapping - cast as text", "select cast('test' as text) as id", "CREATE VIEW `test_wrapping` AS SELECT * FROM (select cast('test' as text) as id)", }, } for _, s := range scenarios { t.Run(s.name, func(t *testing.T) { app, _ := tests.NewTestApp() defer app.Cleanup() collection := &models.Collection{ Name: viewName, Type: models.CollectionTypeView, Options: types.JsonMap{ "query": s.query, }, } err := app.Dao().SaveCollection(collection) if err != nil { t.Fatal(err) } var sql string rowErr := app.Dao().DB().NewQuery("SELECT sql FROM sqlite_master WHERE type='view' AND name={:name}"). Bind(dbx.Params{"name": viewName}). Row(&sql) if rowErr != nil { t.Fatalf("Failed to retrieve view sql: %v", rowErr) } if sql != s.expected { t.Fatalf("Expected query \n%v, \ngot \n%v", s.expected, sql) } }) } } func TestImportCollections(t *testing.T) { t.Parallel() totalCollections := 11 scenarios := []struct { name string jsonData string deleteMissing bool beforeRecordsSync func(txDao *daos.Dao, mappedImported, mappedExisting map[string]*models.Collection) error expectError bool expectCollectionsCount int beforeTestFunc func(testApp *tests.TestApp, resultCollections []*models.Collection) afterTestFunc func(testApp *tests.TestApp, resultCollections []*models.Collection) }{ { name: "empty collections", jsonData: `[]`, expectError: true, expectCollectionsCount: totalCollections, }, { name: "minimal collection import", jsonData: `[ {"name": "import_test1", "schema": [{"name":"test", "type": "text"}]}, {"name": "import_test2", "type": "auth"} ]`, deleteMissing: false, expectError: false, expectCollectionsCount: totalCollections + 2, }, { name: "minimal collection import + failed beforeRecordsSync", jsonData: `[ {"name": "import_test", "schema": [{"name":"test", "type": "text"}]} ]`, beforeRecordsSync: func(txDao *daos.Dao, mappedImported, mappedExisting map[string]*models.Collection) error { return errors.New("test_error") }, deleteMissing: false, expectError: true, expectCollectionsCount: totalCollections, }, { name: "minimal collection import + successful beforeRecordsSync", jsonData: `[ {"name": "import_test", "schema": [{"name":"test", "type": "text"}]} ]`, beforeRecordsSync: func(txDao *daos.Dao, mappedImported, mappedExisting map[string]*models.Collection) error { return nil }, deleteMissing: false, expectError: false, expectCollectionsCount: totalCollections + 1, }, { name: "new + update + delete system collection", jsonData: `[ { "id":"wsmn24bux7wo113", "name":"demo", "schema":[ { "id":"_2hlxbmp", "name":"title", "type":"text", "system":false, "required":true, "unique":false, "options":{ "min":3, "max":null, "pattern":"" } } ] }, { "name": "import1", "schema": [ { "name":"active", "type":"bool" } ] } ]`, deleteMissing: true, expectError: true, expectCollectionsCount: totalCollections, }, { name: "new + update + delete non-system collection", jsonData: `[ { "id": "kpv709sk2lqbqk8", "system": true, "name": "nologin", "type": "auth", "options": { "allowEmailAuth": false, "allowOAuth2Auth": false, "allowUsernameAuth": false, "exceptEmailDomains": [], "manageRule": "@request.auth.collectionName = 'users'", "minPasswordLength": 8, "onlyEmailDomains": [], "requireEmail": true }, "listRule": "", "viewRule": "", "createRule": "", "updateRule": "", "deleteRule": "", "schema": [ { "id": "x8zzktwe", "name": "name", "type": "text", "system": false, "required": false, "unique": false, "options": { "min": null, "max": null, "pattern": "" } } ] }, { "id":"wsmn24bux7wo113", "name":"demo1_rename", "schema":[ { "id":"_2hlxbmp", "name":"title", "type":"text", "system":false, "required":true, "unique":false, "options":{ "min":3, "max":null, "pattern":"" } } ] }, { "id": "test_deleted_collection_name_reuse", "name": "demo2", "schema": [ { "id":"fz6iql2m", "name":"active", "type":"bool" } ] }, { "id": "test_new_view", "name": "new_view", "type": "view", "options": { "query": "select id from demo2" } } ]`, deleteMissing: true, expectError: false, expectCollectionsCount: 4, }, { name: "test with deleteMissing: false", jsonData: `[ { "id":"wsmn24bux7wo113", "name":"demo1", "schema":[ { "id":"_2hlxbmp", "name":"title", "type":"text", "system":false, "required":true, "unique":false, "options":{ "min":3, "max":null, "pattern":"" } }, { "id":"_2hlxbmp", "name":"field_with_duplicate_id", "type":"text", "system":false, "required":true, "unique":false, "options":{ "min":3, "max":null, "pattern":"" } }, { "id":"abcd_import", "name":"new_field", "type":"text" } ] }, { "name": "new_import", "schema": [ { "id":"abcd_import", "name":"active", "type":"bool" } ] } ]`, deleteMissing: false, expectError: false, expectCollectionsCount: totalCollections + 1, afterTestFunc: func(testApp *tests.TestApp, resultCollections []*models.Collection) { expectedCollectionFields := map[string]int{ "nologin": 1, "demo1": 15, "demo2": 2, "demo3": 2, "demo4": 11, "demo5": 6, "new_import": 1, } for name, expectedCount := range expectedCollectionFields { collection, err := testApp.Dao().FindCollectionByNameOrId(name) if err != nil { t.Fatal(err) } if totalFields := len(collection.Schema.Fields()); totalFields != expectedCount { t.Errorf("Expected %d %q fields, got %d", expectedCount, collection.Name, totalFields) } } }, }, } for _, scenario := range scenarios { testApp, _ := tests.NewTestApp() defer testApp.Cleanup() importedCollections := []*models.Collection{} // load data loadErr := json.Unmarshal([]byte(scenario.jsonData), &importedCollections) if loadErr != nil { t.Fatalf("[%s] Failed to load data: %v", scenario.name, loadErr) continue } err := testApp.Dao().ImportCollections(importedCollections, scenario.deleteMissing, scenario.beforeRecordsSync) hasErr := err != nil if hasErr != scenario.expectError { t.Errorf("[%s] Expected hasErr to be %v, got %v (%v)", scenario.name, scenario.expectError, hasErr, err) } // check collections count collections := []*models.Collection{} if err := testApp.Dao().CollectionQuery().All(&collections); err != nil { t.Fatal(err) } if len(collections) != scenario.expectCollectionsCount { t.Errorf("[%s] Expected %d collections, got %d", scenario.name, scenario.expectCollectionsCount, len(collections)) } if scenario.afterTestFunc != nil { scenario.afterTestFunc(testApp, collections) } } }