mirror of
https://github.com/pocketbase/pocketbase.git
synced 2025-01-24 06:16:43 +02:00
630 lines
13 KiB
Go
630 lines
13 KiB
Go
package daos_test
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"testing"
|
|
|
|
"github.com/pocketbase/dbx"
|
|
"github.com/pocketbase/pocketbase/core"
|
|
"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"
|
|
)
|
|
|
|
func ensureNoTempViews(app core.App, t *testing.T) {
|
|
var total int
|
|
|
|
err := app.Dao().DB().Select("count(*)").
|
|
From("sqlite_schema").
|
|
AndWhere(dbx.HashExp{"type": "view"}).
|
|
AndWhere(dbx.NewExp(`[[name]] LIKE '%\_temp\_%' ESCAPE '\'`)).
|
|
Limit(1).
|
|
Row(&total)
|
|
if err != nil {
|
|
t.Fatalf("Failed to check for temp views: %v", err)
|
|
}
|
|
|
|
if total > 0 {
|
|
t.Fatalf("Expected all temp views to be deleted, got %d", total)
|
|
}
|
|
}
|
|
|
|
func TestDeleteView(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
app, _ := tests.NewTestApp()
|
|
defer app.Cleanup()
|
|
|
|
scenarios := []struct {
|
|
viewName string
|
|
expectError bool
|
|
}{
|
|
{"", true},
|
|
{"demo1", true}, // not a view table
|
|
{"missing", false}, // missing or already deleted
|
|
{"view1", false}, // existing
|
|
{"VieW1", false}, // view names are case insensitives
|
|
}
|
|
|
|
for i, s := range scenarios {
|
|
err := app.Dao().DeleteView(s.viewName)
|
|
|
|
hasErr := err != nil
|
|
if hasErr != s.expectError {
|
|
t.Errorf("[%d - %q] Expected hasErr %v, got %v (%v)", i, s.viewName, s.expectError, hasErr, err)
|
|
}
|
|
}
|
|
|
|
ensureNoTempViews(app, t)
|
|
}
|
|
|
|
func TestSaveView(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
app, _ := tests.NewTestApp()
|
|
defer app.Cleanup()
|
|
|
|
scenarios := []struct {
|
|
scenarioName string
|
|
viewName string
|
|
query string
|
|
expectError bool
|
|
expectColumns []string
|
|
}{
|
|
{
|
|
"empty name and query",
|
|
"",
|
|
"",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"empty name",
|
|
"",
|
|
"select * from _admins",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"empty query",
|
|
"123Test",
|
|
"",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"invalid query",
|
|
"123Test",
|
|
"123 456",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"missing table",
|
|
"123Test",
|
|
"select id from missing",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"non select query",
|
|
"123Test",
|
|
"drop table _admins",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"multiple select queries",
|
|
"123Test",
|
|
"select *, count(id) as c from _admins; select * from demo1;",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"try to break the parent parenthesis",
|
|
"123Test",
|
|
"select *, count(id) as c from `_admins`)",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"simple select query (+ trimmed semicolon)",
|
|
"123Test",
|
|
";select *, count(id) as c from _admins;",
|
|
false,
|
|
[]string{
|
|
"id", "created", "updated",
|
|
"passwordHash", "tokenKey", "email",
|
|
"lastResetSentAt", "avatar", "c",
|
|
},
|
|
},
|
|
{
|
|
"update old view with new query",
|
|
"123Test",
|
|
"select 1 as test from _admins",
|
|
false,
|
|
[]string{"test"},
|
|
},
|
|
}
|
|
|
|
for _, s := range scenarios {
|
|
t.Run(s.scenarioName, func(t *testing.T) {
|
|
err := app.Dao().SaveView(s.viewName, s.query)
|
|
|
|
hasErr := err != nil
|
|
if hasErr != s.expectError {
|
|
t.Fatalf("Expected hasErr %v, got %v (%v)", s.expectError, hasErr, err)
|
|
}
|
|
|
|
if hasErr {
|
|
return
|
|
}
|
|
|
|
infoRows, err := app.Dao().TableInfo(s.viewName)
|
|
if err != nil {
|
|
t.Fatalf("Failed to fetch table info for %s: %v", s.viewName, err)
|
|
}
|
|
|
|
if len(s.expectColumns) != len(infoRows) {
|
|
t.Fatalf("Expected %d columns, got %d", len(s.expectColumns), len(infoRows))
|
|
}
|
|
|
|
for _, row := range infoRows {
|
|
if !list.ExistInSlice(row.Name, s.expectColumns) {
|
|
t.Fatalf("Missing %q column in %v", row.Name, s.expectColumns)
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
ensureNoTempViews(app, t)
|
|
}
|
|
|
|
func TestCreateViewSchemaWithDiscardedNestedTransaction(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
app, _ := tests.NewTestApp()
|
|
defer app.Cleanup()
|
|
|
|
app.Dao().RunInTransaction(func(txDao *daos.Dao) error {
|
|
_, err := txDao.CreateViewSchema("select id from missing")
|
|
if err == nil {
|
|
t.Fatal("Expected error, got nil")
|
|
}
|
|
|
|
return nil
|
|
})
|
|
|
|
ensureNoTempViews(app, t)
|
|
}
|
|
|
|
func TestCreateViewSchema(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
app, _ := tests.NewTestApp()
|
|
defer app.Cleanup()
|
|
|
|
scenarios := []struct {
|
|
name string
|
|
query string
|
|
expectError bool
|
|
expectFields map[string]string // name-type pairs
|
|
}{
|
|
{
|
|
"empty query",
|
|
"",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"invalid query",
|
|
"test 123456",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"missing table",
|
|
"select id from missing",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"query with wildcard column",
|
|
"select a.id, a.* from demo1 a",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"query without id",
|
|
"select text, url, created, updated from demo1",
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
"query with comments",
|
|
`
|
|
select
|
|
-- test single line
|
|
id,
|
|
text,
|
|
/* multi
|
|
line comment */
|
|
url, created, updated from demo1
|
|
`,
|
|
false,
|
|
map[string]string{
|
|
"text": schema.FieldTypeText,
|
|
"url": schema.FieldTypeUrl,
|
|
},
|
|
},
|
|
{
|
|
"query with all fields and quoted identifiers",
|
|
`
|
|
select
|
|
"id",
|
|
"created",
|
|
"updated",
|
|
[text],
|
|
` + "`bool`" + `,
|
|
"url",
|
|
"select_one",
|
|
"select_many",
|
|
"file_one",
|
|
"demo1"."file_many",
|
|
` + "`demo1`." + "`number`" + ` number_alias,
|
|
"email",
|
|
"datetime",
|
|
"json",
|
|
"rel_one",
|
|
"rel_many",
|
|
'single_quoted_custom_literal' as 'single_quoted_column'
|
|
from demo1
|
|
`,
|
|
false,
|
|
map[string]string{
|
|
"text": schema.FieldTypeText,
|
|
"bool": schema.FieldTypeBool,
|
|
"url": schema.FieldTypeUrl,
|
|
"select_one": schema.FieldTypeSelect,
|
|
"select_many": schema.FieldTypeSelect,
|
|
"file_one": schema.FieldTypeFile,
|
|
"file_many": schema.FieldTypeFile,
|
|
"number_alias": schema.FieldTypeNumber,
|
|
"email": schema.FieldTypeEmail,
|
|
"datetime": schema.FieldTypeDate,
|
|
"json": schema.FieldTypeJson,
|
|
"rel_one": schema.FieldTypeRelation,
|
|
"rel_many": schema.FieldTypeRelation,
|
|
"single_quoted_column": schema.FieldTypeJson,
|
|
},
|
|
},
|
|
{
|
|
"query with indirect relations fields",
|
|
"select a.id, b.id as bid, b.created from demo1 as a left join demo2 b",
|
|
false,
|
|
map[string]string{
|
|
"bid": schema.FieldTypeRelation,
|
|
},
|
|
},
|
|
{
|
|
"query with multiple froms, joins and style of aliasses",
|
|
`
|
|
select
|
|
a.id as id,
|
|
b.id as bid,
|
|
lj.id cid,
|
|
ij.id as did,
|
|
a.bool,
|
|
_admins.id as eid,
|
|
_admins.email
|
|
from demo1 a, demo2 as b
|
|
left join demo3 lj on lj.id = 123
|
|
inner join demo4 as ij on ij.id = 123
|
|
join _admins
|
|
where 1=1
|
|
group by a.id
|
|
limit 10
|
|
`,
|
|
false,
|
|
map[string]string{
|
|
"bid": schema.FieldTypeRelation,
|
|
"cid": schema.FieldTypeRelation,
|
|
"did": schema.FieldTypeRelation,
|
|
"bool": schema.FieldTypeBool,
|
|
"eid": schema.FieldTypeJson, // not from collection
|
|
"email": schema.FieldTypeJson, // not from collection
|
|
},
|
|
},
|
|
{
|
|
"query with casts",
|
|
`select
|
|
a.id,
|
|
count(a.id) count,
|
|
cast(a.id as int) cast_int,
|
|
cast(a.id as integer) cast_integer,
|
|
cast(a.id as real) cast_real,
|
|
cast(a.id as decimal) cast_decimal,
|
|
cast(a.id as numeric) cast_numeric,
|
|
cast(a.id as text) cast_text,
|
|
cast(a.id as bool) cast_bool,
|
|
cast(a.id as boolean) cast_boolean,
|
|
avg(a.id) avg,
|
|
sum(a.id) sum,
|
|
total(a.id) total,
|
|
min(a.id) min,
|
|
max(a.id) max
|
|
from demo1 a`,
|
|
false,
|
|
map[string]string{
|
|
"count": schema.FieldTypeNumber,
|
|
"total": schema.FieldTypeNumber,
|
|
"cast_int": schema.FieldTypeNumber,
|
|
"cast_integer": schema.FieldTypeNumber,
|
|
"cast_real": schema.FieldTypeNumber,
|
|
"cast_decimal": schema.FieldTypeNumber,
|
|
"cast_numeric": schema.FieldTypeNumber,
|
|
"cast_text": schema.FieldTypeText,
|
|
"cast_bool": schema.FieldTypeBool,
|
|
"cast_boolean": schema.FieldTypeBool,
|
|
// json because they are nullable
|
|
"sum": schema.FieldTypeJson,
|
|
"avg": schema.FieldTypeJson,
|
|
"min": schema.FieldTypeJson,
|
|
"max": schema.FieldTypeJson,
|
|
},
|
|
},
|
|
{
|
|
"query with reserved auth collection fields",
|
|
`
|
|
select
|
|
a.id,
|
|
a.username,
|
|
a.email,
|
|
a.emailVisibility,
|
|
a.verified,
|
|
demo1.id relid
|
|
from users a
|
|
left join demo1
|
|
`,
|
|
false,
|
|
map[string]string{
|
|
"username": schema.FieldTypeText,
|
|
"email": schema.FieldTypeEmail,
|
|
"emailVisibility": schema.FieldTypeBool,
|
|
"verified": schema.FieldTypeBool,
|
|
"relid": schema.FieldTypeRelation,
|
|
},
|
|
},
|
|
{
|
|
"query with unknown fields and aliases",
|
|
`select
|
|
id,
|
|
id as id2,
|
|
text as text_alias,
|
|
url as url_alias,
|
|
"demo1"."bool" as bool_alias,
|
|
number as number_alias,
|
|
created created_alias,
|
|
updated updated_alias,
|
|
123 as custom
|
|
from demo1`,
|
|
false,
|
|
map[string]string{
|
|
"id2": schema.FieldTypeRelation,
|
|
"text_alias": schema.FieldTypeText,
|
|
"url_alias": schema.FieldTypeUrl,
|
|
"bool_alias": schema.FieldTypeBool,
|
|
"number_alias": schema.FieldTypeNumber,
|
|
"created_alias": schema.FieldTypeDate,
|
|
"updated_alias": schema.FieldTypeDate,
|
|
"custom": schema.FieldTypeJson,
|
|
},
|
|
},
|
|
{
|
|
"query with distinct and reordered id column",
|
|
`select distinct
|
|
id as id2,
|
|
id,
|
|
123 as custom
|
|
from demo1`,
|
|
false,
|
|
map[string]string{
|
|
"id2": schema.FieldTypeRelation,
|
|
"custom": schema.FieldTypeJson,
|
|
},
|
|
},
|
|
{
|
|
"query with aliasing the same field multiple times",
|
|
`select
|
|
a.id as id,
|
|
a.text as alias1,
|
|
a.text as alias2,
|
|
b.text as alias3,
|
|
b.text as alias4
|
|
from demo1 a
|
|
left join demo1 as b`,
|
|
false,
|
|
map[string]string{
|
|
"alias1": schema.FieldTypeText,
|
|
"alias2": schema.FieldTypeText,
|
|
"alias3": schema.FieldTypeText,
|
|
"alias4": schema.FieldTypeText,
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, s := range scenarios {
|
|
result, err := app.Dao().CreateViewSchema(s.query)
|
|
|
|
hasErr := err != nil
|
|
if hasErr != s.expectError {
|
|
t.Errorf("[%s] Expected hasErr %v, got %v (%v)", s.name, s.expectError, hasErr, err)
|
|
continue
|
|
}
|
|
|
|
if hasErr {
|
|
continue
|
|
}
|
|
|
|
if len(s.expectFields) != len(result.Fields()) {
|
|
serialized, _ := json.Marshal(result)
|
|
t.Errorf("[%s] Expected %d fields, got %d: \n%s", s.name, len(s.expectFields), len(result.Fields()), serialized)
|
|
continue
|
|
}
|
|
|
|
for name, typ := range s.expectFields {
|
|
field := result.GetFieldByName(name)
|
|
|
|
if field == nil {
|
|
t.Errorf("[%s] Expected to find field %s, got nil", s.name, name)
|
|
continue
|
|
}
|
|
|
|
if field.Type != typ {
|
|
t.Errorf("[%s] Expected field %s to be %q, got %s", s.name, name, typ, field.Type)
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
|
|
ensureNoTempViews(app, t)
|
|
}
|
|
|
|
func TestFindRecordByViewFile(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
app, _ := tests.NewTestApp()
|
|
defer app.Cleanup()
|
|
|
|
prevCollection, err := app.Dao().FindCollectionByNameOrId("demo1")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
totalLevels := 6
|
|
|
|
// create collection view mocks
|
|
fileOneAlias := "file_one one0"
|
|
fileManyAlias := "file_many many0"
|
|
mockCollections := make([]*models.Collection, 0, totalLevels)
|
|
for i := 0; i <= totalLevels; i++ {
|
|
view := new(models.Collection)
|
|
view.Type = models.CollectionTypeView
|
|
view.Name = fmt.Sprintf("_test_view%d", i)
|
|
view.SetOptions(&models.CollectionViewOptions{
|
|
Query: fmt.Sprintf(
|
|
"select id, %s, %s from %s",
|
|
fileOneAlias,
|
|
fileManyAlias,
|
|
prevCollection.Name,
|
|
),
|
|
})
|
|
|
|
// save view
|
|
if err := app.Dao().SaveCollection(view); err != nil {
|
|
t.Fatalf("Failed to save view%d: %v", i, err)
|
|
}
|
|
|
|
mockCollections = append(mockCollections, view)
|
|
prevCollection = view
|
|
fileOneAlias = fmt.Sprintf("one%d one%d", i, i+1)
|
|
fileManyAlias = fmt.Sprintf("many%d many%d", i, i+1)
|
|
}
|
|
|
|
fileOneName := "test_d61b33QdDU.txt"
|
|
fileManyName := "test_QZFjKjXchk.txt"
|
|
expectedRecordId := "84nmscqy84lsi1t"
|
|
|
|
scenarios := []struct {
|
|
name string
|
|
collectionNameOrId string
|
|
fileFieldName string
|
|
filename string
|
|
expectError bool
|
|
expectRecordId string
|
|
}{
|
|
{
|
|
"missing collection",
|
|
"missing",
|
|
"a",
|
|
fileOneName,
|
|
true,
|
|
"",
|
|
},
|
|
{
|
|
"non-view collection",
|
|
"demo1",
|
|
"file_one",
|
|
fileOneName,
|
|
true,
|
|
"",
|
|
},
|
|
{
|
|
"view collection after the max recursion limit",
|
|
mockCollections[totalLevels-1].Name,
|
|
fmt.Sprintf("one%d", totalLevels-1),
|
|
fileOneName,
|
|
true,
|
|
"",
|
|
},
|
|
{
|
|
"first view collection (single file)",
|
|
mockCollections[0].Name,
|
|
"one0",
|
|
fileOneName,
|
|
false,
|
|
expectedRecordId,
|
|
},
|
|
{
|
|
"first view collection (many files)",
|
|
mockCollections[0].Name,
|
|
"many0",
|
|
fileManyName,
|
|
false,
|
|
expectedRecordId,
|
|
},
|
|
|
|
{
|
|
"last view collection before the recursion limit (single file)",
|
|
mockCollections[totalLevels-2].Name,
|
|
fmt.Sprintf("one%d", totalLevels-2),
|
|
fileOneName,
|
|
false,
|
|
expectedRecordId,
|
|
},
|
|
{
|
|
"last view collection before the recursion limit (many files)",
|
|
mockCollections[totalLevels-2].Name,
|
|
fmt.Sprintf("many%d", totalLevels-2),
|
|
fileManyName,
|
|
false,
|
|
expectedRecordId,
|
|
},
|
|
}
|
|
|
|
for _, s := range scenarios {
|
|
record, err := app.Dao().FindRecordByViewFile(
|
|
s.collectionNameOrId,
|
|
s.fileFieldName,
|
|
s.filename,
|
|
)
|
|
|
|
hasErr := err != nil
|
|
if hasErr != s.expectError {
|
|
t.Errorf("[%s] Expected hasErr %v, got %v (%v)", s.name, s.expectError, hasErr, err)
|
|
continue
|
|
}
|
|
|
|
if hasErr {
|
|
continue
|
|
}
|
|
|
|
if record.Id != s.expectRecordId {
|
|
t.Errorf("[%s] Expected recordId %q, got %q", s.name, s.expectRecordId, record.Id)
|
|
}
|
|
}
|
|
}
|