2022-07-06 23:19:05 +02:00
|
|
|
package forms_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
2022-07-12 12:42:06 +02:00
|
|
|
"errors"
|
2022-07-06 23:19:05 +02:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
validation "github.com/go-ozzo/ozzo-validation/v4"
|
|
|
|
"github.com/pocketbase/pocketbase/forms"
|
|
|
|
"github.com/pocketbase/pocketbase/models"
|
|
|
|
"github.com/pocketbase/pocketbase/models/schema"
|
|
|
|
"github.com/pocketbase/pocketbase/tests"
|
2023-03-21 15:31:20 +02:00
|
|
|
"github.com/pocketbase/pocketbase/tools/dbutils"
|
2022-08-08 18:16:33 +02:00
|
|
|
"github.com/pocketbase/pocketbase/tools/security"
|
2022-07-06 23:19:05 +02:00
|
|
|
"github.com/spf13/cast"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestNewCollectionUpsert(t *testing.T) {
|
|
|
|
app, _ := tests.NewTestApp()
|
|
|
|
defer app.Cleanup()
|
|
|
|
|
|
|
|
collection := &models.Collection{}
|
2022-10-30 10:28:14 +02:00
|
|
|
collection.Name = "test_name"
|
|
|
|
collection.Type = "test_type"
|
2022-07-06 23:19:05 +02:00
|
|
|
collection.System = true
|
2023-02-18 19:33:42 +02:00
|
|
|
listRule := "test_list"
|
2022-07-06 23:19:05 +02:00
|
|
|
collection.ListRule = &listRule
|
|
|
|
viewRule := "test_view"
|
|
|
|
collection.ViewRule = &viewRule
|
|
|
|
createRule := "test_create"
|
|
|
|
collection.CreateRule = &createRule
|
|
|
|
updateRule := "test_update"
|
|
|
|
collection.UpdateRule = &updateRule
|
|
|
|
deleteRule := "test_delete"
|
|
|
|
collection.DeleteRule = &deleteRule
|
|
|
|
collection.Schema = schema.NewSchema(&schema.SchemaField{
|
|
|
|
Name: "test",
|
|
|
|
Type: schema.FieldTypeText,
|
|
|
|
})
|
|
|
|
|
|
|
|
form := forms.NewCollectionUpsert(app, collection)
|
|
|
|
|
|
|
|
if form.Name != collection.Name {
|
|
|
|
t.Errorf("Expected Name %q, got %q", collection.Name, form.Name)
|
|
|
|
}
|
|
|
|
|
2022-10-30 10:28:14 +02:00
|
|
|
if form.Type != collection.Type {
|
|
|
|
t.Errorf("Expected Type %q, got %q", collection.Type, form.Type)
|
|
|
|
}
|
|
|
|
|
2022-07-06 23:19:05 +02:00
|
|
|
if form.System != collection.System {
|
|
|
|
t.Errorf("Expected System %v, got %v", collection.System, form.System)
|
|
|
|
}
|
|
|
|
|
|
|
|
if form.ListRule != collection.ListRule {
|
|
|
|
t.Errorf("Expected ListRule %v, got %v", collection.ListRule, form.ListRule)
|
|
|
|
}
|
|
|
|
|
|
|
|
if form.ViewRule != collection.ViewRule {
|
|
|
|
t.Errorf("Expected ViewRule %v, got %v", collection.ViewRule, form.ViewRule)
|
|
|
|
}
|
|
|
|
|
|
|
|
if form.CreateRule != collection.CreateRule {
|
|
|
|
t.Errorf("Expected CreateRule %v, got %v", collection.CreateRule, form.CreateRule)
|
|
|
|
}
|
|
|
|
|
|
|
|
if form.UpdateRule != collection.UpdateRule {
|
|
|
|
t.Errorf("Expected UpdateRule %v, got %v", collection.UpdateRule, form.UpdateRule)
|
|
|
|
}
|
|
|
|
|
|
|
|
if form.DeleteRule != collection.DeleteRule {
|
|
|
|
t.Errorf("Expected DeleteRule %v, got %v", collection.DeleteRule, form.DeleteRule)
|
|
|
|
}
|
|
|
|
|
|
|
|
// store previous state and modify the collection schema to verify
|
|
|
|
// that the form.Schema is a deep clone
|
|
|
|
loadedSchema, _ := collection.Schema.MarshalJSON()
|
|
|
|
collection.Schema.AddField(&schema.SchemaField{
|
|
|
|
Name: "new_field",
|
|
|
|
Type: schema.FieldTypeBool,
|
|
|
|
})
|
|
|
|
|
|
|
|
formSchema, _ := form.Schema.MarshalJSON()
|
|
|
|
|
|
|
|
if string(formSchema) != string(loadedSchema) {
|
|
|
|
t.Errorf("Expected Schema %v, got %v", string(loadedSchema), string(formSchema))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-30 10:28:14 +02:00
|
|
|
func TestCollectionUpsertValidateAndSubmit(t *testing.T) {
|
2022-07-06 23:19:05 +02:00
|
|
|
app, _ := tests.NewTestApp()
|
|
|
|
defer app.Cleanup()
|
|
|
|
|
|
|
|
scenarios := []struct {
|
2022-10-30 10:28:14 +02:00
|
|
|
testName string
|
2022-07-06 23:19:05 +02:00
|
|
|
existingName string
|
|
|
|
jsonData string
|
|
|
|
expectedErrors []string
|
|
|
|
}{
|
2022-11-16 15:13:04 +02:00
|
|
|
{"empty create (base)", "", "{}", []string{"name", "schema"}},
|
|
|
|
{"empty create (auth)", "", `{"type":"auth"}`, []string{"name"}},
|
2023-02-18 19:33:42 +02:00
|
|
|
{"empty create (view)", "", `{"type":"view"}`, []string{"name", "options"}},
|
2022-10-30 10:28:14 +02:00
|
|
|
{"empty update", "demo2", "{}", []string{}},
|
2022-07-06 23:19:05 +02:00
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"create failure",
|
2022-07-06 23:19:05 +02:00
|
|
|
"",
|
|
|
|
`{
|
|
|
|
"name": "test ?!@#$",
|
2022-10-30 10:28:14 +02:00
|
|
|
"type": "invalid",
|
2022-07-06 23:19:05 +02:00
|
|
|
"system": true,
|
|
|
|
"schema": [
|
|
|
|
{"name":"","type":"text"}
|
|
|
|
],
|
|
|
|
"listRule": "missing = '123'",
|
|
|
|
"viewRule": "missing = '123'",
|
|
|
|
"createRule": "missing = '123'",
|
|
|
|
"updateRule": "missing = '123'",
|
2023-03-21 15:31:20 +02:00
|
|
|
"deleteRule": "missing = '123'",
|
|
|
|
"indexes": ["create index '' on '' ()"]
|
2022-07-06 23:19:05 +02:00
|
|
|
}`,
|
2023-03-21 15:31:20 +02:00
|
|
|
[]string{"name", "type", "schema", "listRule", "viewRule", "createRule", "updateRule", "deleteRule", "indexes"},
|
2022-07-06 23:19:05 +02:00
|
|
|
},
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"create failure - existing name",
|
2022-07-06 23:19:05 +02:00
|
|
|
"",
|
|
|
|
`{
|
2022-10-30 10:28:14 +02:00
|
|
|
"name": "demo1",
|
2022-07-06 23:19:05 +02:00
|
|
|
"system": true,
|
|
|
|
"schema": [
|
|
|
|
{"name":"test","type":"text"}
|
|
|
|
],
|
|
|
|
"listRule": "test='123'",
|
|
|
|
"viewRule": "test='123'",
|
|
|
|
"createRule": "test='123'",
|
|
|
|
"updateRule": "test='123'",
|
|
|
|
"deleteRule": "test='123'"
|
|
|
|
}`,
|
|
|
|
[]string{"name"},
|
|
|
|
},
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"create failure - existing internal table",
|
2022-07-06 23:19:05 +02:00
|
|
|
"",
|
|
|
|
`{
|
2022-10-30 10:28:14 +02:00
|
|
|
"name": "_admins",
|
2022-07-06 23:19:05 +02:00
|
|
|
"schema": [
|
|
|
|
{"name":"test","type":"text"}
|
|
|
|
]
|
|
|
|
}`,
|
|
|
|
[]string{"name"},
|
|
|
|
},
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"create failure - name starting with underscore",
|
2022-07-06 23:19:05 +02:00
|
|
|
"",
|
|
|
|
`{
|
|
|
|
"name": "_test_new",
|
|
|
|
"schema": [
|
|
|
|
{"name":"test","type":"text"}
|
|
|
|
]
|
|
|
|
}`,
|
|
|
|
[]string{"name"},
|
|
|
|
},
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"create failure - duplicated field names (case insensitive)",
|
2022-07-06 23:19:05 +02:00
|
|
|
"",
|
|
|
|
`{
|
|
|
|
"name": "test_new",
|
|
|
|
"schema": [
|
|
|
|
{"name":"test","type":"text"},
|
|
|
|
{"name":"tESt","type":"text"}
|
|
|
|
]
|
|
|
|
}`,
|
|
|
|
[]string{"schema"},
|
|
|
|
},
|
2023-01-23 21:57:35 +02:00
|
|
|
{
|
|
|
|
"create failure - missing relation display field",
|
|
|
|
"",
|
|
|
|
`{
|
|
|
|
"name": "test_new",
|
|
|
|
"type": "base",
|
|
|
|
"schema": [
|
|
|
|
{
|
|
|
|
"name":"test",
|
|
|
|
"type":"relation",
|
|
|
|
"options":{
|
|
|
|
"collectionId":"wsmn24bux7wo113",
|
|
|
|
"displayFields":["text", "missing"]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}`,
|
|
|
|
[]string{"schema"},
|
|
|
|
},
|
2022-07-06 23:19:05 +02:00
|
|
|
{
|
2023-02-18 19:33:42 +02:00
|
|
|
"create failure - check auth options validators",
|
2022-10-30 10:28:14 +02:00
|
|
|
"",
|
|
|
|
`{
|
|
|
|
"name": "test_new",
|
|
|
|
"type": "auth",
|
|
|
|
"schema": [
|
|
|
|
{"name":"test","type":"text"}
|
|
|
|
],
|
|
|
|
"options": { "minPasswordLength": 3 }
|
|
|
|
}`,
|
|
|
|
[]string{"options"},
|
|
|
|
},
|
2023-02-18 19:33:42 +02:00
|
|
|
{
|
|
|
|
"create failure - check view options validators",
|
|
|
|
"",
|
|
|
|
`{
|
|
|
|
"name": "test_new",
|
|
|
|
"type": "view",
|
|
|
|
"options": { "query": "invalid query" }
|
|
|
|
}`,
|
|
|
|
[]string{"options"},
|
|
|
|
},
|
2022-10-30 10:28:14 +02:00
|
|
|
{
|
|
|
|
"create success",
|
2022-07-06 23:19:05 +02:00
|
|
|
"",
|
|
|
|
`{
|
|
|
|
"name": "test_new",
|
2022-11-15 15:06:46 +02:00
|
|
|
"type": "auth",
|
2022-07-06 23:19:05 +02:00
|
|
|
"system": true,
|
|
|
|
"schema": [
|
|
|
|
{"id":"a123456","name":"test1","type":"text"},
|
2023-01-23 21:57:35 +02:00
|
|
|
{"id":"b123456","name":"test2","type":"email"},
|
|
|
|
{
|
|
|
|
"name":"test3",
|
|
|
|
"type":"relation",
|
|
|
|
"options":{
|
|
|
|
"collectionId":"v851q4r790rhknl",
|
|
|
|
"displayFields":["name","id","created","updated","username","email","emailVisibility","verified"]
|
|
|
|
}
|
|
|
|
}
|
2022-07-06 23:19:05 +02:00
|
|
|
],
|
2022-11-15 15:06:46 +02:00
|
|
|
"listRule": "test1='123' && verified = true",
|
|
|
|
"viewRule": "test1='123' && emailVisibility = true",
|
|
|
|
"createRule": "test1='123' && email != ''",
|
|
|
|
"updateRule": "test1='123' && username != ''",
|
2023-03-21 15:31:20 +02:00
|
|
|
"deleteRule": "test1='123' && id != ''",
|
|
|
|
"indexes": ["create index idx_test_new on anything (test1)"]
|
2022-07-06 23:19:05 +02:00
|
|
|
}`,
|
|
|
|
[]string{},
|
|
|
|
},
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"update failure - changing field type",
|
2022-07-06 23:19:05 +02:00
|
|
|
"test_new",
|
|
|
|
`{
|
|
|
|
"schema": [
|
|
|
|
{"id":"a123456","name":"test1","type":"url"},
|
|
|
|
{"id":"b123456","name":"test2","type":"bool"}
|
2023-03-21 15:31:20 +02:00
|
|
|
],
|
|
|
|
"indexes": ["create index idx_test_new on test_new (test1)", "invalid"]
|
2022-07-06 23:19:05 +02:00
|
|
|
}`,
|
2023-03-21 15:31:20 +02:00
|
|
|
[]string{"schema", "indexes"},
|
2022-07-06 23:19:05 +02:00
|
|
|
},
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"update success - rename fields to existing field names (aka. reusing field names)",
|
2022-07-06 23:19:05 +02:00
|
|
|
"test_new",
|
|
|
|
`{
|
|
|
|
"schema": [
|
|
|
|
{"id":"a123456","name":"test2","type":"text"},
|
|
|
|
{"id":"b123456","name":"test1","type":"email"}
|
|
|
|
]
|
|
|
|
}`,
|
2022-07-18 15:26:37 +02:00
|
|
|
[]string{},
|
2022-07-06 23:19:05 +02:00
|
|
|
},
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"update failure - existing name",
|
|
|
|
"demo2",
|
|
|
|
`{"name": "demo3"}`,
|
2022-07-06 23:19:05 +02:00
|
|
|
[]string{"name"},
|
|
|
|
},
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"update failure - changing system collection",
|
|
|
|
"nologin",
|
2022-07-06 23:19:05 +02:00
|
|
|
`{
|
|
|
|
"name": "update",
|
|
|
|
"system": false,
|
|
|
|
"schema": [
|
2022-10-30 10:28:14 +02:00
|
|
|
{"id":"koih1lqx","name":"abc","type":"text"}
|
2022-07-06 23:19:05 +02:00
|
|
|
],
|
2022-10-30 10:28:14 +02:00
|
|
|
"listRule": "abc = '123'",
|
|
|
|
"viewRule": "abc = '123'",
|
|
|
|
"createRule": "abc = '123'",
|
|
|
|
"updateRule": "abc = '123'",
|
|
|
|
"deleteRule": "abc = '123'"
|
|
|
|
}`,
|
|
|
|
[]string{"name", "system"},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"update failure - changing collection type",
|
|
|
|
"demo3",
|
|
|
|
`{
|
|
|
|
"type": "auth"
|
2022-07-06 23:19:05 +02:00
|
|
|
}`,
|
2022-10-30 10:28:14 +02:00
|
|
|
[]string{"type"},
|
2022-07-06 23:19:05 +02:00
|
|
|
},
|
2023-02-24 14:12:27 +02:00
|
|
|
{
|
|
|
|
"update failure - changing relation collection",
|
|
|
|
"users",
|
|
|
|
`{
|
|
|
|
"schema": [
|
|
|
|
{
|
|
|
|
"id": "lkeigvv3",
|
|
|
|
"name": "rel",
|
|
|
|
"type": "relation",
|
|
|
|
"options": {
|
|
|
|
"collectionId": "wzlqyes4orhoygb",
|
|
|
|
"cascadeDelete": false,
|
|
|
|
"maxSelect": 1,
|
|
|
|
"displayFields": null
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}`,
|
|
|
|
[]string{"schema"},
|
|
|
|
},
|
2022-07-06 23:19:05 +02:00
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"update failure - all fields",
|
|
|
|
"demo2",
|
2022-07-06 23:19:05 +02:00
|
|
|
`{
|
|
|
|
"name": "test ?!@#$",
|
2022-10-30 10:28:14 +02:00
|
|
|
"type": "invalid",
|
2022-07-06 23:19:05 +02:00
|
|
|
"system": true,
|
|
|
|
"schema": [
|
|
|
|
{"name":"","type":"text"}
|
|
|
|
],
|
|
|
|
"listRule": "missing = '123'",
|
|
|
|
"viewRule": "missing = '123'",
|
|
|
|
"createRule": "missing = '123'",
|
|
|
|
"updateRule": "missing = '123'",
|
2022-10-30 10:28:14 +02:00
|
|
|
"deleteRule": "missing = '123'",
|
2023-03-21 15:31:20 +02:00
|
|
|
"options": {"test": 123},
|
|
|
|
"indexes": ["create index '' from demo2 on (id)"]
|
2022-07-06 23:19:05 +02:00
|
|
|
}`,
|
2023-03-21 15:31:20 +02:00
|
|
|
[]string{"name", "type", "system", "schema", "listRule", "viewRule", "createRule", "updateRule", "deleteRule", "indexes"},
|
2022-07-06 23:19:05 +02:00
|
|
|
},
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"update success - update all fields",
|
|
|
|
"clients",
|
2022-07-06 23:19:05 +02:00
|
|
|
`{
|
|
|
|
"name": "demo_update",
|
2022-10-30 10:28:14 +02:00
|
|
|
"type": "auth",
|
2022-07-06 23:19:05 +02:00
|
|
|
"schema": [
|
|
|
|
{"id":"_2hlxbmp","name":"test","type":"text"}
|
|
|
|
],
|
2022-11-15 12:03:12 +02:00
|
|
|
"listRule": "test='123' && verified = true",
|
|
|
|
"viewRule": "test='123' && emailVisibility = true",
|
|
|
|
"createRule": "test='123' && email != ''",
|
|
|
|
"updateRule": "test='123' && username != ''",
|
|
|
|
"deleteRule": "test='123' && id != ''",
|
2023-03-21 15:31:20 +02:00
|
|
|
"options": {"minPasswordLength": 10},
|
|
|
|
"indexes": [
|
|
|
|
"create index idx_clients_test1 on anything (id, email, test)",
|
|
|
|
"create unique index idx_clients_test2 on clients (id, username, email)"
|
|
|
|
]
|
2022-07-06 23:19:05 +02:00
|
|
|
}`,
|
|
|
|
[]string{},
|
|
|
|
},
|
|
|
|
// (fail due to filters old field references)
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"update failure - rename the schema field of the last updated collection",
|
2022-07-06 23:19:05 +02:00
|
|
|
"demo_update",
|
|
|
|
`{
|
|
|
|
"schema": [
|
|
|
|
{"id":"_2hlxbmp","name":"test_renamed","type":"text"}
|
|
|
|
]
|
|
|
|
}`,
|
|
|
|
[]string{"listRule", "viewRule", "createRule", "updateRule", "deleteRule"},
|
|
|
|
},
|
|
|
|
// (cleared filter references)
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"update success - rename the schema field of the last updated collection",
|
2022-07-06 23:19:05 +02:00
|
|
|
"demo_update",
|
|
|
|
`{
|
|
|
|
"schema": [
|
|
|
|
{"id":"_2hlxbmp","name":"test_renamed","type":"text"}
|
|
|
|
],
|
|
|
|
"listRule": null,
|
|
|
|
"viewRule": null,
|
|
|
|
"createRule": null,
|
|
|
|
"updateRule": null,
|
2023-03-21 15:31:20 +02:00
|
|
|
"deleteRule": null,
|
|
|
|
"indexes": []
|
2022-07-06 23:19:05 +02:00
|
|
|
}`,
|
|
|
|
[]string{},
|
|
|
|
},
|
|
|
|
{
|
2022-10-30 10:28:14 +02:00
|
|
|
"update success - system collection",
|
|
|
|
"nologin",
|
2022-07-06 23:19:05 +02:00
|
|
|
`{
|
2022-10-30 10:28:14 +02:00
|
|
|
"listRule": "name='123'",
|
|
|
|
"viewRule": "name='123'",
|
|
|
|
"createRule": "name='123'",
|
|
|
|
"updateRule": "name='123'",
|
|
|
|
"deleteRule": "name='123'"
|
2022-07-06 23:19:05 +02:00
|
|
|
}`,
|
|
|
|
[]string{},
|
|
|
|
},
|
2023-02-18 19:33:42 +02:00
|
|
|
|
|
|
|
// view tests
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
{
|
|
|
|
"view create failure",
|
|
|
|
"",
|
|
|
|
`{
|
|
|
|
"name": "upsert_view",
|
|
|
|
"type": "view",
|
|
|
|
"listRule": "id='123' && verified = true",
|
|
|
|
"viewRule": "id='123' && emailVisibility = true",
|
|
|
|
"schema": [
|
|
|
|
{"id":"abc123","name":"some invalid field name that will be overwritten !@#$","type":"bool"}
|
|
|
|
],
|
|
|
|
"options": {
|
|
|
|
"query": "select id, email from users; drop table _admins;"
|
2023-03-21 15:31:20 +02:00
|
|
|
},
|
|
|
|
"indexes": ["create index idx_test_view on upsert_view (id)"]
|
2023-02-18 19:33:42 +02:00
|
|
|
}`,
|
|
|
|
[]string{
|
|
|
|
"listRule",
|
|
|
|
"viewRule",
|
|
|
|
"options",
|
2023-03-21 15:31:20 +02:00
|
|
|
"indexes", // views don't have indexes
|
2023-02-18 19:33:42 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"view create success",
|
|
|
|
"",
|
|
|
|
`{
|
|
|
|
"name": "upsert_view",
|
|
|
|
"type": "view",
|
|
|
|
"listRule": "id='123' && verified = true",
|
|
|
|
"viewRule": "id='123' && emailVisibility = true",
|
|
|
|
"schema": [
|
|
|
|
{"id":"abc123","name":"some invalid field name that will be overwritten !@#$","type":"bool"}
|
|
|
|
],
|
|
|
|
"options": {
|
|
|
|
"query": "select id, emailVisibility, verified from users"
|
|
|
|
}
|
|
|
|
}`,
|
|
|
|
[]string{
|
|
|
|
// "schema", should be overwritten by an autogenerated from the query
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"view update failure (schema autogeneration and rule fields check)",
|
|
|
|
"upsert_view",
|
|
|
|
`{
|
|
|
|
"name": "upsert_view_2",
|
|
|
|
"listRule": "id='456' && verified = true",
|
|
|
|
"viewRule": "id='456'",
|
|
|
|
"createRule": "id='123'",
|
|
|
|
"updateRule": "id='123'",
|
|
|
|
"deleteRule": "id='123'",
|
|
|
|
"schema": [
|
|
|
|
{"id":"abc123","name":"verified","type":"bool"}
|
|
|
|
],
|
|
|
|
"options": {
|
|
|
|
"query": "select 1 as id"
|
|
|
|
}
|
|
|
|
}`,
|
|
|
|
[]string{
|
|
|
|
"listRule", // missing field (ignoring the old or explicit schema)
|
|
|
|
"createRule", // not allowed
|
|
|
|
"updateRule", // not allowed
|
|
|
|
"deleteRule", // not allowed
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"view update failure (check query identifiers format)",
|
|
|
|
"upsert_view",
|
|
|
|
`{
|
|
|
|
"listRule": null,
|
|
|
|
"viewRule": null,
|
|
|
|
"options": {
|
|
|
|
"query": "select 1 as id, 2 as [invalid!@#]"
|
|
|
|
}
|
|
|
|
}`,
|
|
|
|
[]string{
|
|
|
|
"schema", // should fail due to invalid field name
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"view update success",
|
|
|
|
"upsert_view",
|
|
|
|
`{
|
|
|
|
"listRule": null,
|
|
|
|
"viewRule": null,
|
|
|
|
"options": {
|
|
|
|
"query": "select 1 as id, 2 as valid"
|
|
|
|
}
|
|
|
|
}`,
|
|
|
|
[]string{},
|
|
|
|
},
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
|
2022-10-30 10:28:14 +02:00
|
|
|
for _, s := range scenarios {
|
2022-07-06 23:19:05 +02:00
|
|
|
collection := &models.Collection{}
|
|
|
|
if s.existingName != "" {
|
|
|
|
var err error
|
|
|
|
collection, err = app.Dao().FindCollectionByNameOrId(s.existingName)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
form := forms.NewCollectionUpsert(app, collection)
|
|
|
|
|
|
|
|
// load data
|
|
|
|
loadErr := json.Unmarshal([]byte(s.jsonData), form)
|
|
|
|
if loadErr != nil {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Failed to load form data: %v", s.testName, loadErr)
|
2022-07-06 23:19:05 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2022-07-12 12:42:06 +02:00
|
|
|
interceptorCalls := 0
|
2023-01-15 17:00:28 +02:00
|
|
|
interceptor := func(next forms.InterceptorNextFunc[*models.Collection]) forms.InterceptorNextFunc[*models.Collection] {
|
|
|
|
return func(c *models.Collection) error {
|
2022-07-12 12:42:06 +02:00
|
|
|
interceptorCalls++
|
2023-01-15 17:00:28 +02:00
|
|
|
return next(c)
|
2022-07-12 12:42:06 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-06 23:19:05 +02:00
|
|
|
// parse errors
|
2022-07-12 12:42:06 +02:00
|
|
|
result := form.Submit(interceptor)
|
2022-07-06 23:19:05 +02:00
|
|
|
errs, ok := result.(validation.Errors)
|
|
|
|
if !ok && result != nil {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Failed to parse errors %v", s.testName, result)
|
2022-07-06 23:19:05 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2022-07-12 12:42:06 +02:00
|
|
|
// check interceptor calls
|
2022-12-03 14:50:02 +02:00
|
|
|
expectInterceptorCalls := 1
|
2022-07-12 12:42:06 +02:00
|
|
|
if len(s.expectedErrors) > 0 {
|
2022-12-03 14:50:02 +02:00
|
|
|
expectInterceptorCalls = 0
|
2022-07-12 12:42:06 +02:00
|
|
|
}
|
2022-12-03 14:50:02 +02:00
|
|
|
if interceptorCalls != expectInterceptorCalls {
|
|
|
|
t.Errorf("[%s] Expected interceptor to be called %d, got %d", s.testName, expectInterceptorCalls, interceptorCalls)
|
2022-07-12 12:42:06 +02:00
|
|
|
}
|
|
|
|
|
2022-07-06 23:19:05 +02:00
|
|
|
// check errors
|
|
|
|
if len(errs) > len(s.expectedErrors) {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Expected error keys %v, got %v", s.testName, s.expectedErrors, errs)
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
for _, k := range s.expectedErrors {
|
|
|
|
if _, ok := errs[k]; !ok {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Missing expected error key %q in %v", s.testName, k, errs)
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(s.expectedErrors) > 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
collection, _ = app.Dao().FindCollectionByNameOrId(form.Name)
|
|
|
|
if collection == nil {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Expected to find collection %q, got nil", s.testName, form.Name)
|
2022-07-06 23:19:05 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if form.Name != collection.Name {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Expected Name %q, got %q", s.testName, collection.Name, form.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
if form.Type != collection.Type {
|
|
|
|
t.Errorf("[%s] Expected Type %q, got %q", s.testName, collection.Type, form.Type)
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if form.System != collection.System {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Expected System %v, got %v", s.testName, collection.System, form.System)
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if cast.ToString(form.ListRule) != cast.ToString(collection.ListRule) {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Expected ListRule %v, got %v", s.testName, collection.ListRule, form.ListRule)
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if cast.ToString(form.ViewRule) != cast.ToString(collection.ViewRule) {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Expected ViewRule %v, got %v", s.testName, collection.ViewRule, form.ViewRule)
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if cast.ToString(form.CreateRule) != cast.ToString(collection.CreateRule) {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Expected CreateRule %v, got %v", s.testName, collection.CreateRule, form.CreateRule)
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if cast.ToString(form.UpdateRule) != cast.ToString(collection.UpdateRule) {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Expected UpdateRule %v, got %v", s.testName, collection.UpdateRule, form.UpdateRule)
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if cast.ToString(form.DeleteRule) != cast.ToString(collection.DeleteRule) {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Expected DeleteRule %v, got %v", s.testName, collection.DeleteRule, form.DeleteRule)
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
|
2023-02-18 19:33:42 +02:00
|
|
|
rawFormSchema, _ := form.Schema.MarshalJSON()
|
|
|
|
rawCollectionSchema, _ := collection.Schema.MarshalJSON()
|
|
|
|
|
|
|
|
if len(form.Schema.Fields()) != len(collection.Schema.Fields()) {
|
|
|
|
t.Errorf("[%s] Expected Schema \n%v, \ngot \n%v", s.testName, string(rawCollectionSchema), string(rawFormSchema))
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, f := range form.Schema.Fields() {
|
|
|
|
if collection.Schema.GetFieldByName(f.Name) == nil {
|
|
|
|
t.Errorf("[%s] Missing field %s \nin \n%v", s.testName, f.Name, string(rawFormSchema))
|
|
|
|
continue
|
|
|
|
}
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
2023-03-21 15:31:20 +02:00
|
|
|
|
|
|
|
// check indexes (if any)
|
|
|
|
allIndexes, _ := app.Dao().TableIndexes(form.Name)
|
|
|
|
for _, formIdx := range form.Indexes {
|
|
|
|
parsed := dbutils.ParseIndex(formIdx)
|
|
|
|
parsed.TableName = form.Name
|
|
|
|
normalizedIdx := parsed.Build()
|
|
|
|
|
|
|
|
var exists bool
|
|
|
|
for _, idx := range allIndexes {
|
|
|
|
if dbutils.ParseIndex(idx).Build() == normalizedIdx {
|
|
|
|
exists = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !exists {
|
|
|
|
t.Errorf(
|
|
|
|
"[%s] Missing index %s \nin \n%v", s.testName, normalizedIdx, allIndexes)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
2022-07-06 23:19:05 +02:00
|
|
|
}
|
|
|
|
}
|
2022-07-12 12:42:06 +02:00
|
|
|
|
|
|
|
func TestCollectionUpsertSubmitInterceptors(t *testing.T) {
|
|
|
|
app, _ := tests.NewTestApp()
|
|
|
|
defer app.Cleanup()
|
|
|
|
|
2022-10-30 10:28:14 +02:00
|
|
|
collection, err := app.Dao().FindCollectionByNameOrId("demo2")
|
2022-07-12 12:42:06 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
form := forms.NewCollectionUpsert(app, collection)
|
|
|
|
form.Name = "test_new"
|
|
|
|
|
|
|
|
testErr := errors.New("test_error")
|
|
|
|
interceptorCollectionName := ""
|
|
|
|
|
|
|
|
interceptor1Called := false
|
2023-01-15 17:00:28 +02:00
|
|
|
interceptor1 := func(next forms.InterceptorNextFunc[*models.Collection]) forms.InterceptorNextFunc[*models.Collection] {
|
|
|
|
return func(c *models.Collection) error {
|
2022-07-12 12:42:06 +02:00
|
|
|
interceptor1Called = true
|
2023-01-15 17:00:28 +02:00
|
|
|
return next(c)
|
2022-07-12 12:42:06 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
interceptor2Called := false
|
2023-01-15 17:00:28 +02:00
|
|
|
interceptor2 := func(next forms.InterceptorNextFunc[*models.Collection]) forms.InterceptorNextFunc[*models.Collection] {
|
|
|
|
return func(c *models.Collection) error {
|
2022-07-12 12:42:06 +02:00
|
|
|
interceptorCollectionName = collection.Name // to check if the record was filled
|
|
|
|
interceptor2Called = true
|
|
|
|
return testErr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
submitErr := form.Submit(interceptor1, interceptor2)
|
|
|
|
if submitErr != testErr {
|
|
|
|
t.Fatalf("Expected submitError %v, got %v", testErr, submitErr)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !interceptor1Called {
|
|
|
|
t.Fatalf("Expected interceptor1 to be called")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !interceptor2Called {
|
|
|
|
t.Fatalf("Expected interceptor2 to be called")
|
|
|
|
}
|
|
|
|
|
|
|
|
if interceptorCollectionName != form.Name {
|
|
|
|
t.Fatalf("Expected the form model to be filled before calling the interceptors")
|
|
|
|
}
|
|
|
|
}
|
2022-08-08 18:16:33 +02:00
|
|
|
|
|
|
|
func TestCollectionUpsertWithCustomId(t *testing.T) {
|
|
|
|
app, _ := tests.NewTestApp()
|
|
|
|
defer app.Cleanup()
|
|
|
|
|
2022-10-30 10:28:14 +02:00
|
|
|
existingCollection, err := app.Dao().FindCollectionByNameOrId("demo2")
|
2022-08-08 18:16:33 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
newCollection := func() *models.Collection {
|
|
|
|
return &models.Collection{
|
2022-11-06 15:28:41 +02:00
|
|
|
Name: "c_" + security.PseudorandomString(4),
|
2022-08-08 18:16:33 +02:00
|
|
|
Schema: existingCollection.Schema,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
scenarios := []struct {
|
|
|
|
name string
|
|
|
|
jsonData string
|
|
|
|
collection *models.Collection
|
|
|
|
expectError bool
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
"empty data",
|
|
|
|
"{}",
|
|
|
|
newCollection(),
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"empty id",
|
|
|
|
`{"id":""}`,
|
|
|
|
newCollection(),
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"id < 15 chars",
|
|
|
|
`{"id":"a23"}`,
|
|
|
|
newCollection(),
|
|
|
|
true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"id > 15 chars",
|
|
|
|
`{"id":"a234567890123456"}`,
|
|
|
|
newCollection(),
|
|
|
|
true,
|
|
|
|
},
|
|
|
|
{
|
2022-08-11 09:29:01 +02:00
|
|
|
"id = 15 chars (invalid chars)",
|
|
|
|
`{"id":"a@3456789012345"}`,
|
|
|
|
newCollection(),
|
|
|
|
true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"id = 15 chars (valid chars)",
|
2022-08-08 18:16:33 +02:00
|
|
|
`{"id":"a23456789012345"}`,
|
|
|
|
newCollection(),
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"changing the id of an existing item",
|
|
|
|
`{"id":"b23456789012345"}`,
|
|
|
|
existingCollection,
|
|
|
|
true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"using the same existing item id",
|
|
|
|
`{"id":"` + existingCollection.Id + `"}`,
|
|
|
|
existingCollection,
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"skipping the id for existing item",
|
|
|
|
`{}`,
|
|
|
|
existingCollection,
|
|
|
|
false,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2022-10-30 10:28:14 +02:00
|
|
|
for _, s := range scenarios {
|
|
|
|
form := forms.NewCollectionUpsert(app, s.collection)
|
2022-08-08 18:16:33 +02:00
|
|
|
|
|
|
|
// load data
|
2022-10-30 10:28:14 +02:00
|
|
|
loadErr := json.Unmarshal([]byte(s.jsonData), form)
|
2022-08-08 18:16:33 +02:00
|
|
|
if loadErr != nil {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Failed to load form data: %v", s.name, loadErr)
|
2022-08-08 18:16:33 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
submitErr := form.Submit()
|
|
|
|
hasErr := submitErr != nil
|
|
|
|
|
2022-10-30 10:28:14 +02:00
|
|
|
if hasErr != s.expectError {
|
|
|
|
t.Errorf("[%s] Expected hasErr to be %v, got %v (%v)", s.name, s.expectError, hasErr, submitErr)
|
2022-08-08 18:16:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if !hasErr && form.Id != "" {
|
|
|
|
_, err := app.Dao().FindCollectionByNameOrId(form.Id)
|
|
|
|
if err != nil {
|
2022-10-30 10:28:14 +02:00
|
|
|
t.Errorf("[%s] Expected to find record with id %s, got %v", s.name, form.Id, err)
|
2022-08-08 18:16:33 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|