mirror of
https://github.com/pocketbase/pocketbase.git
synced 2024-11-21 13:35:49 +02:00
501 lines
14 KiB
Go
501 lines
14 KiB
Go
package daos
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/pocketbase/dbx"
|
|
"github.com/pocketbase/pocketbase/models"
|
|
"github.com/pocketbase/pocketbase/models/schema"
|
|
"github.com/pocketbase/pocketbase/tools/list"
|
|
)
|
|
|
|
// CollectionQuery returns a new Collection select query.
|
|
func (dao *Dao) CollectionQuery() *dbx.SelectQuery {
|
|
return dao.ModelQuery(&models.Collection{})
|
|
}
|
|
|
|
// FindCollectionsByType finds all collections by the given type.
|
|
func (dao *Dao) FindCollectionsByType(collectionType string) ([]*models.Collection, error) {
|
|
collections := []*models.Collection{}
|
|
|
|
err := dao.CollectionQuery().
|
|
AndWhere(dbx.HashExp{"type": collectionType}).
|
|
OrderBy("created ASC").
|
|
All(&collections)
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return collections, nil
|
|
}
|
|
|
|
// FindCollectionByNameOrId finds a single collection by its name (case insensitive) or id.
|
|
func (dao *Dao) FindCollectionByNameOrId(nameOrId string) (*models.Collection, error) {
|
|
model := &models.Collection{}
|
|
|
|
err := dao.CollectionQuery().
|
|
AndWhere(dbx.NewExp("[[id]] = {:id} OR LOWER([[name]])={:name}", dbx.Params{
|
|
"id": nameOrId,
|
|
"name": strings.ToLower(nameOrId),
|
|
})).
|
|
Limit(1).
|
|
One(model)
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return model, nil
|
|
}
|
|
|
|
// IsCollectionNameUnique checks that there is no existing collection
|
|
// with the provided name (case insensitive!).
|
|
//
|
|
// Note: case insensitive check because the name is used also as a table name for the records.
|
|
func (dao *Dao) IsCollectionNameUnique(name string, excludeIds ...string) bool {
|
|
if name == "" {
|
|
return false
|
|
}
|
|
|
|
query := dao.CollectionQuery().
|
|
Select("count(*)").
|
|
AndWhere(dbx.NewExp("LOWER([[name]])={:name}", dbx.Params{"name": strings.ToLower(name)})).
|
|
Limit(1)
|
|
|
|
if uniqueExcludeIds := list.NonzeroUniques(excludeIds); len(uniqueExcludeIds) > 0 {
|
|
query.AndWhere(dbx.NotIn("id", list.ToInterfaceSlice(uniqueExcludeIds)...))
|
|
}
|
|
|
|
var exists bool
|
|
|
|
return query.Row(&exists) == nil && !exists
|
|
}
|
|
|
|
// FindCollectionReferences returns information for all
|
|
// relation schema fields referencing the provided collection.
|
|
//
|
|
// If the provided collection has reference to itself then it will be
|
|
// also included in the result. To exclude it, pass the collection id
|
|
// as the excludeId argument.
|
|
func (dao *Dao) FindCollectionReferences(collection *models.Collection, excludeIds ...string) (map[*models.Collection][]*schema.SchemaField, error) {
|
|
collections := []*models.Collection{}
|
|
|
|
query := dao.CollectionQuery()
|
|
|
|
if uniqueExcludeIds := list.NonzeroUniques(excludeIds); len(uniqueExcludeIds) > 0 {
|
|
query.AndWhere(dbx.NotIn("id", list.ToInterfaceSlice(uniqueExcludeIds)...))
|
|
}
|
|
|
|
if err := query.All(&collections); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
result := map[*models.Collection][]*schema.SchemaField{}
|
|
|
|
for _, c := range collections {
|
|
for _, f := range c.Schema.Fields() {
|
|
if f.Type != schema.FieldTypeRelation {
|
|
continue
|
|
}
|
|
f.InitOptions()
|
|
options, _ := f.Options.(*schema.RelationOptions)
|
|
if options != nil && options.CollectionId == collection.Id {
|
|
result[c] = append(result[c], f)
|
|
}
|
|
}
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
// DeleteCollection deletes the provided Collection model.
|
|
// This method automatically deletes the related collection records table.
|
|
//
|
|
// NB! The collection cannot be deleted, if:
|
|
// - is system collection (aka. collection.System is true)
|
|
// - is referenced as part of a relation field in another collection
|
|
func (dao *Dao) DeleteCollection(collection *models.Collection) error {
|
|
if collection.System {
|
|
return fmt.Errorf("system collection %q cannot be deleted", collection.Name)
|
|
}
|
|
|
|
// ensure that there aren't any existing references.
|
|
// note: the select is outside of the transaction to prevent SQLITE_LOCKED error when mixing read&write in a single transaction
|
|
result, err := dao.FindCollectionReferences(collection, collection.Id)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if total := len(result); total > 0 {
|
|
names := make([]string, 0, len(result))
|
|
for ref := range result {
|
|
names = append(names, ref.Name)
|
|
}
|
|
return fmt.Errorf("the collection %q has external relation field references (%s)", collection.Name, strings.Join(names, ", "))
|
|
}
|
|
|
|
return dao.RunInTransaction(func(txDao *Dao) error {
|
|
// delete the related view or records table
|
|
if collection.IsView() {
|
|
if err := txDao.DeleteView(collection.Name); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
if err := txDao.DeleteTable(collection.Name); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// trigger views resave to check for dependencies
|
|
if err := txDao.resaveViewsWithChangedSchema(collection.Id); err != nil {
|
|
return fmt.Errorf("the collection has a view dependency - %w", err)
|
|
}
|
|
|
|
return txDao.Delete(collection)
|
|
})
|
|
}
|
|
|
|
// SaveCollection persists the provided Collection model and updates
|
|
// its related records table schema.
|
|
//
|
|
// If collection.IsNew() is true, the method will perform a create, otherwise an update.
|
|
// To explicitly mark a collection for update you can use collection.MarkAsNotNew().
|
|
func (dao *Dao) SaveCollection(collection *models.Collection) error {
|
|
var oldCollection *models.Collection
|
|
|
|
if !collection.IsNew() {
|
|
// get the existing collection state to compare with the new one
|
|
// note: the select is outside of the transaction to prevent SQLITE_LOCKED error when mixing read&write in a single transaction
|
|
var findErr error
|
|
oldCollection, findErr = dao.FindCollectionByNameOrId(collection.Id)
|
|
if findErr != nil {
|
|
return findErr
|
|
}
|
|
}
|
|
|
|
txErr := dao.RunInTransaction(func(txDao *Dao) error {
|
|
// set default collection type
|
|
if collection.Type == "" {
|
|
collection.Type = models.CollectionTypeBase
|
|
}
|
|
|
|
switch collection.Type {
|
|
case models.CollectionTypeView:
|
|
if err := txDao.saveViewCollection(collection, oldCollection); err != nil {
|
|
return err
|
|
}
|
|
default:
|
|
// persist the collection model
|
|
if err := txDao.Save(collection); err != nil {
|
|
return err
|
|
}
|
|
|
|
// sync the changes with the related records table
|
|
if err := txDao.SyncRecordTableSchema(collection, oldCollection); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
})
|
|
|
|
if txErr != nil {
|
|
return txErr
|
|
}
|
|
|
|
// trigger an update for all views with changed schema as a result of the current collection save
|
|
// (ignoring view errors to allow users to update the query from the UI)
|
|
dao.resaveViewsWithChangedSchema(collection.Id)
|
|
|
|
return nil
|
|
}
|
|
|
|
// ImportCollections imports the provided collections list within a single transaction.
|
|
//
|
|
// NB1! If deleteMissing is set, all local collections and schema fields, that are not present
|
|
// in the imported configuration, WILL BE DELETED (including their related records data).
|
|
//
|
|
// NB2! This method doesn't perform validations on the imported collections data!
|
|
// If you need validations, use [forms.CollectionsImport].
|
|
func (dao *Dao) ImportCollections(
|
|
importedCollections []*models.Collection,
|
|
deleteMissing bool,
|
|
afterSync func(txDao *Dao, mappedImported, mappedExisting map[string]*models.Collection) error,
|
|
) error {
|
|
if len(importedCollections) == 0 {
|
|
return errors.New("no collections to import")
|
|
}
|
|
|
|
return dao.RunInTransaction(func(txDao *Dao) error {
|
|
existingCollections := []*models.Collection{}
|
|
if err := txDao.CollectionQuery().OrderBy("updated ASC").All(&existingCollections); err != nil {
|
|
return err
|
|
}
|
|
mappedExisting := make(map[string]*models.Collection, len(existingCollections))
|
|
for _, existing := range existingCollections {
|
|
mappedExisting[existing.GetId()] = existing
|
|
}
|
|
|
|
mappedImported := make(map[string]*models.Collection, len(importedCollections))
|
|
for _, imported := range importedCollections {
|
|
// generate id if not set
|
|
if !imported.HasId() {
|
|
imported.MarkAsNew()
|
|
imported.RefreshId()
|
|
}
|
|
|
|
// set default type if missing
|
|
if imported.Type == "" {
|
|
imported.Type = models.CollectionTypeBase
|
|
}
|
|
|
|
if existing, ok := mappedExisting[imported.GetId()]; ok {
|
|
imported.MarkAsNotNew()
|
|
|
|
// preserve original created date
|
|
if !existing.Created.IsZero() {
|
|
imported.Created = existing.Created
|
|
}
|
|
|
|
// extend existing schema
|
|
if !deleteMissing {
|
|
schemaClone, _ := existing.Schema.Clone()
|
|
for _, f := range imported.Schema.Fields() {
|
|
schemaClone.AddField(f) // add or replace
|
|
}
|
|
imported.Schema = *schemaClone
|
|
}
|
|
} else {
|
|
imported.MarkAsNew()
|
|
}
|
|
|
|
mappedImported[imported.GetId()] = imported
|
|
}
|
|
|
|
// delete old collections not available in the new configuration
|
|
// (before saving the imports in case a deleted collection name is being reused)
|
|
if deleteMissing {
|
|
for _, existing := range existingCollections {
|
|
if mappedImported[existing.GetId()] != nil {
|
|
continue // exist
|
|
}
|
|
|
|
if existing.System {
|
|
return fmt.Errorf("system collection %q cannot be deleted", existing.Name)
|
|
}
|
|
|
|
// delete the related records table or view
|
|
if existing.IsView() {
|
|
if err := txDao.DeleteView(existing.Name); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
if err := txDao.DeleteTable(existing.Name); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// delete the collection
|
|
if err := txDao.Delete(existing); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
// upsert imported collections
|
|
for _, imported := range importedCollections {
|
|
if err := txDao.Save(imported); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// sync record tables
|
|
for _, imported := range importedCollections {
|
|
if imported.IsView() {
|
|
continue
|
|
}
|
|
|
|
existing := mappedExisting[imported.GetId()]
|
|
|
|
if err := txDao.SyncRecordTableSchema(imported, existing); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// sync views
|
|
for _, imported := range importedCollections {
|
|
if !imported.IsView() {
|
|
continue
|
|
}
|
|
|
|
existing := mappedExisting[imported.GetId()]
|
|
|
|
if err := txDao.saveViewCollection(imported, existing); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if afterSync != nil {
|
|
if err := afterSync(txDao, mappedImported, mappedExisting); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
})
|
|
}
|
|
|
|
// saveViewCollection persists the provided View collection changes:
|
|
// - deletes the old related SQL view (if any)
|
|
// - creates a new SQL view with the latest newCollection.Options.Query
|
|
// - generates a new schema based on newCollection.Options.Query
|
|
// - updates newCollection.Schema based on the generated view table info and query
|
|
// - saves the newCollection
|
|
//
|
|
// This method returns an error if newCollection is not a "view".
|
|
func (dao *Dao) saveViewCollection(newCollection, oldCollection *models.Collection) error {
|
|
if !newCollection.IsView() {
|
|
return errors.New("not a view collection")
|
|
}
|
|
|
|
return dao.RunInTransaction(func(txDao *Dao) error {
|
|
query := newCollection.ViewOptions().Query
|
|
|
|
// generate collection schema from the query
|
|
viewSchema, err := txDao.CreateViewSchema(query)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// delete old renamed view
|
|
if oldCollection != nil {
|
|
if err := txDao.DeleteView(oldCollection.Name); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// wrap view query if necessary
|
|
query, err = txDao.normalizeViewQueryId(query)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to normalize view query id: %w", err)
|
|
}
|
|
|
|
// (re)create the view
|
|
if err := txDao.SaveView(newCollection.Name, query); err != nil {
|
|
return err
|
|
}
|
|
|
|
newCollection.Schema = viewSchema
|
|
|
|
return txDao.Save(newCollection)
|
|
})
|
|
}
|
|
|
|
// @todo consider removing once custom id types are supported
|
|
//
|
|
// normalizeViewQueryId wraps (if necessary) the provided view query
|
|
// with a subselect to ensure that the id column is a text since
|
|
// currently we don't support non-string model ids
|
|
// (see https://github.com/pocketbase/pocketbase/issues/3110).
|
|
func (dao *Dao) normalizeViewQueryId(query string) (string, error) {
|
|
query = strings.Trim(strings.TrimSpace(query), ";")
|
|
|
|
parsed, err := dao.parseQueryToFields(query)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
needWrapping := true
|
|
|
|
idField := parsed[schema.FieldNameId]
|
|
if idField != nil && idField.field != nil &&
|
|
idField.field.Type != schema.FieldTypeJson &&
|
|
idField.field.Type != schema.FieldTypeNumber &&
|
|
idField.field.Type != schema.FieldTypeBool {
|
|
needWrapping = false
|
|
}
|
|
|
|
if !needWrapping {
|
|
return query, nil // no changes needed
|
|
}
|
|
|
|
// raw parse to preserve the columns order
|
|
rawParsed := new(identifiersParser)
|
|
if err := rawParsed.parse(query); err != nil {
|
|
return "", err
|
|
}
|
|
|
|
columns := make([]string, 0, len(rawParsed.columns))
|
|
for _, col := range rawParsed.columns {
|
|
if col.alias == schema.FieldNameId {
|
|
columns = append(columns, fmt.Sprintf("cast([[%s]] as text) [[%s]]", col.alias, col.alias))
|
|
} else {
|
|
columns = append(columns, "[["+col.alias+"]]")
|
|
}
|
|
}
|
|
|
|
query = fmt.Sprintf("SELECT %s FROM (%s)", strings.Join(columns, ","), query)
|
|
|
|
return query, nil
|
|
}
|
|
|
|
// resaveViewsWithChangedSchema updates all view collections with changed schemas.
|
|
func (dao *Dao) resaveViewsWithChangedSchema(excludeIds ...string) error {
|
|
collections, err := dao.FindCollectionsByType(models.CollectionTypeView)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return dao.RunInTransaction(func(txDao *Dao) error {
|
|
for _, collection := range collections {
|
|
if len(excludeIds) > 0 && list.ExistInSlice(collection.Id, excludeIds) {
|
|
continue
|
|
}
|
|
|
|
// clone the existing schema so that it is safe for temp modifications
|
|
oldSchema, err := collection.Schema.Clone()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// generate a new schema from the query
|
|
newSchema, err := txDao.CreateViewSchema(collection.ViewOptions().Query)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// unset the schema field ids to exclude from the comparison
|
|
for _, f := range oldSchema.Fields() {
|
|
f.Id = ""
|
|
}
|
|
for _, f := range newSchema.Fields() {
|
|
f.Id = ""
|
|
}
|
|
|
|
encodedNewSchema, err := json.Marshal(newSchema)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
encodedOldSchema, err := json.Marshal(oldSchema)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if bytes.EqualFold(encodedNewSchema, encodedOldSchema) {
|
|
continue // no changes
|
|
}
|
|
|
|
if err := txDao.saveViewCollection(collection, nil); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
})
|
|
}
|