1
0
mirror of https://github.com/pocketbase/pocketbase.git synced 2025-01-25 06:42:43 +02:00
pocketbase/daos/collection.go

291 lines
8.4 KiB
Go
Raw Normal View History

2022-07-07 00:19:05 +03:00
package daos
import (
"errors"
"fmt"
"strings"
"github.com/pocketbase/dbx"
"github.com/pocketbase/pocketbase/models"
"github.com/pocketbase/pocketbase/models/schema"
2022-10-30 10:28:14 +02:00
"github.com/pocketbase/pocketbase/tools/list"
2022-07-07 00:19:05 +03:00
)
// CollectionQuery returns a new Collection select query.
func (dao *Dao) CollectionQuery() *dbx.SelectQuery {
return dao.ModelQuery(&models.Collection{})
}
2022-11-14 14:43:20 +02:00
// FindCollectionsByType finds all collections by the given type.
2022-10-30 10:28:14 +02:00
func (dao *Dao) FindCollectionsByType(collectionType string) ([]*models.Collection, error) {
collections := []*models.Collection{}
2022-10-30 10:28:14 +02:00
err := dao.CollectionQuery().
AndWhere(dbx.HashExp{"type": collectionType}).
OrderBy("created ASC").
All(&collections)
2022-10-30 10:28:14 +02:00
if err != nil {
return nil, err
}
return collections, nil
2022-10-30 10:28:14 +02:00
}
2022-07-07 00:19:05 +03:00
// FindCollectionByNameOrId finds the first collection by its name or id.
func (dao *Dao) FindCollectionByNameOrId(nameOrId string) (*models.Collection, error) {
model := &models.Collection{}
err := dao.CollectionQuery().
AndWhere(dbx.Or(
dbx.HashExp{"id": nameOrId},
dbx.HashExp{"name": 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 sensitive check because the name is used also as a table name for the records.
2022-10-30 10:28:14 +02:00
func (dao *Dao) IsCollectionNameUnique(name string, excludeIds ...string) bool {
2022-07-07 00:19:05 +03:00
if name == "" {
return false
}
2022-10-30 10:28:14 +02:00
query := dao.CollectionQuery().
2022-07-07 00:19:05 +03:00
Select("count(*)").
AndWhere(dbx.NewExp("LOWER([[name]])={:name}", dbx.Params{"name": strings.ToLower(name)})).
2022-10-30 10:28:14 +02:00
Limit(1)
2022-07-07 00:19:05 +03:00
2022-10-30 10:28:14 +02:00
if len(excludeIds) > 0 {
uniqueExcludeIds := list.NonzeroUniques(excludeIds)
query.AndWhere(dbx.NotIn("id", list.ToInterfaceSlice(uniqueExcludeIds)...))
}
2022-07-07 00:19:05 +03:00
2022-10-30 10:28:14 +02:00
var exists bool
2022-07-07 00:19:05 +03:00
2022-10-30 10:28:14 +02:00
return query.Row(&exists) == nil && !exists
2022-07-07 00:19:05 +03:00
}
// FindCollectionReferences returns information for all
// relation schema fields referencing the provided collection.
//
// If the provided collection has reference to itself then it will be
2022-10-17 19:17:44 +02:00
// also included in the result. To exclude it, pass the collection id
2022-07-07 00:19:05 +03:00
// as the excludeId argument.
2022-10-30 10:28:14 +02:00
func (dao *Dao) FindCollectionReferences(collection *models.Collection, excludeIds ...string) (map[*models.Collection][]*schema.SchemaField, error) {
2022-07-07 00:19:05 +03:00
collections := []*models.Collection{}
2022-10-30 10:28:14 +02:00
query := dao.CollectionQuery()
if len(excludeIds) > 0 {
uniqueExcludeIds := list.NonzeroUniques(excludeIds)
query.AndWhere(dbx.NotIn("id", list.ToInterfaceSlice(uniqueExcludeIds)...))
}
if err := query.All(&collections); err != nil {
2022-07-07 00:19:05 +03:00
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 {
2022-08-08 19:16:33 +03:00
return fmt.Errorf("System collection %q cannot be deleted.", collection.Name)
2022-07-07 00:19:05 +03:00
}
// 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 {
2022-08-08 19:16:33 +03:00
return fmt.Errorf("The collection %q has external relation field references (%d).", collection.Name, total)
2022-07-07 00:19:05 +03:00
}
return dao.RunInTransaction(func(txDao *Dao) error {
// delete the related records table
if err := txDao.DeleteTable(collection.Name); err != nil {
return err
}
return txDao.Delete(collection)
})
}
// SaveCollection upserts the provided Collection model and updates
// its related records table schema.
func (dao *Dao) SaveCollection(collection *models.Collection) error {
var oldCollection *models.Collection
2022-08-05 06:00:38 +03:00
if !collection.IsNew() {
2022-07-07 00:19:05 +03:00
// 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
}
}
return dao.RunInTransaction(func(txDao *Dao) error {
2022-10-30 10:28:14 +02:00
// set default collection type
if collection.Type == "" {
collection.Type = models.CollectionTypeBase
}
2022-07-07 00:19:05 +03:00
// persist the collection model
if err := txDao.Save(collection); err != nil {
return err
}
// sync the changes with the related records table
return txDao.SyncRecordTableSchema(collection, oldCollection)
})
}
2022-08-08 19:16:33 +03:00
2022-08-10 20:37:41 +03:00
// ImportCollections imports the provided collections list within a single transaction.
2022-08-08 19:16:33 +03:00
//
2022-08-10 20:37:41 +03:00
// 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).
2022-08-08 19:16:33 +03:00
//
2022-08-10 20:37:41 +03:00
// NB2! This method doesn't perform validations on the imported collections data!
2022-08-08 19:16:33 +03:00
// If you need validations, use [forms.CollectionsImport].
func (dao *Dao) ImportCollections(
importedCollections []*models.Collection,
deleteMissing bool,
beforeRecordsSync 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("created 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
2022-08-08 19:16:33 +03:00
if !imported.HasId() {
imported.MarkAsNew()
imported.RefreshId()
}
2022-10-30 10:28:14 +02:00
// set default type if missing
if imported.Type == "" {
imported.Type = models.CollectionTypeBase
}
if existing, ok := mappedExisting[imported.GetId()]; ok {
// preserve original created date
if !existing.Created.IsZero() {
imported.Created = existing.Created
}
// extend existing schema
if !deleteMissing {
schema, _ := existing.Schema.Clone()
for _, f := range imported.Schema.Fields() {
schema.AddField(f) // add or replace
}
imported.Schema = *schema
2022-08-10 13:22:27 +03:00
}
} else {
imported.MarkAsNew()
2022-08-10 13:22:27 +03:00
}
2022-08-08 19:16:33 +03:00
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 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
}
}
if beforeRecordsSync != nil {
if err := beforeRecordsSync(txDao, mappedImported, mappedExisting); err != nil {
return err
}
}
// delete the record tables of the deleted collections
if deleteMissing {
for _, existing := range existingCollections {
if mappedImported[existing.GetId()] != nil {
continue // exist
}
if err := txDao.DeleteTable(existing.Name); err != nil {
return err
}
}
}
// sync the upserted collections with the related records table
for _, imported := range importedCollections {
existing := mappedExisting[imported.GetId()]
if err := txDao.SyncRecordTableSchema(imported, existing); err != nil {
return err
}
}
return nil
})
}