package daos import ( "context" "database/sql" "errors" "fmt" "strings" "github.com/pocketbase/dbx" "github.com/pocketbase/pocketbase/models" "github.com/pocketbase/pocketbase/models/schema" "github.com/pocketbase/pocketbase/resolvers" "github.com/pocketbase/pocketbase/tools/inflector" "github.com/pocketbase/pocketbase/tools/list" "github.com/pocketbase/pocketbase/tools/search" "github.com/pocketbase/pocketbase/tools/security" "github.com/pocketbase/pocketbase/tools/types" "github.com/spf13/cast" ) // RecordQuery returns a new Record select query from a collection model, id or name. // // In case a collection id or name is provided and that collection doesn't // actually exists, the generated query will be created with a cancelled context // and will fail once an executor (Row(), One(), All(), etc.) is called. func (dao *Dao) RecordQuery(collectionModelOrIdentifier any) *dbx.SelectQuery { var tableName string var collection *models.Collection var collectionErr error switch c := collectionModelOrIdentifier.(type) { case *models.Collection: collection = c tableName = collection.Name case models.Collection: collection = &c tableName = collection.Name case string: collection, collectionErr = dao.FindCollectionByNameOrId(c) if collection != nil { tableName = collection.Name } else { // update with some fake table name for easier debugging tableName = "@@__missing_" + c } default: // update with some fake table name for easier debugging tableName = "@@__invalidCollectionModelOrIdentifier" collectionErr = errors.New("unsupported collection identifier, must be collection model, id or name") } selectCols := fmt.Sprintf("%s.*", dao.DB().QuoteSimpleColumnName(tableName)) query := dao.DB().Select(selectCols).From(tableName) // in case of an error attach a new context and cancel it immediately with the error if collectionErr != nil { // @todo consider changing to WithCancelCause when upgrading // the min Go requirement to 1.20, so that we can pass the error ctx, cancelFunc := context.WithCancel(context.Background()) query.WithContext(ctx) cancelFunc() } return query.WithBuildHook(func(q *dbx.Query) { q.WithExecHook(execLockRetry(dao.ModelQueryTimeout, dao.MaxLockRetries)). WithOneHook(func(q *dbx.Query, a any, op func(b any) error) error { switch v := a.(type) { case *models.Record: if v == nil { return op(a) } row := dbx.NullStringMap{} if err := op(&row); err != nil { return err } record := models.NewRecordFromNullStringMap(collection, row) *v = *record return nil default: return op(a) } }). WithAllHook(func(q *dbx.Query, sliceA any, op func(sliceB any) error) error { switch v := sliceA.(type) { case *[]*models.Record: if v == nil { return op(sliceA) } rows := []dbx.NullStringMap{} if err := op(&rows); err != nil { return err } records := models.NewRecordsFromNullStringMaps(collection, rows) *v = records return nil case *[]models.Record: if v == nil { return op(sliceA) } rows := []dbx.NullStringMap{} if err := op(&rows); err != nil { return err } records := models.NewRecordsFromNullStringMaps(collection, rows) nonPointers := make([]models.Record, len(records)) for i, r := range records { nonPointers[i] = *r } *v = nonPointers return nil default: return op(sliceA) } }) }) } // FindRecordById finds the Record model by its id. func (dao *Dao) FindRecordById( collectionNameOrId string, recordId string, optFilters ...func(q *dbx.SelectQuery) error, ) (*models.Record, error) { collection, err := dao.FindCollectionByNameOrId(collectionNameOrId) if err != nil { return nil, err } query := dao.RecordQuery(collection). AndWhere(dbx.HashExp{collection.Name + ".id": recordId}) for _, filter := range optFilters { if filter == nil { continue } if err := filter(query); err != nil { return nil, err } } record := &models.Record{} if err := query.Limit(1).One(record); err != nil { return nil, err } return record, nil } // FindRecordsByIds finds all Record models by the provided ids. // If no records are found, returns an empty slice. func (dao *Dao) FindRecordsByIds( collectionNameOrId string, recordIds []string, optFilters ...func(q *dbx.SelectQuery) error, ) ([]*models.Record, error) { collection, err := dao.FindCollectionByNameOrId(collectionNameOrId) if err != nil { return nil, err } query := dao.RecordQuery(collection). AndWhere(dbx.In( collection.Name+".id", list.ToInterfaceSlice(recordIds)..., )) for _, filter := range optFilters { if filter == nil { continue } if err := filter(query); err != nil { return nil, err } } records := make([]*models.Record, 0, len(recordIds)) if err := query.All(&records); err != nil { return nil, err } return records, nil } // @todo consider to depricate as it may be easier to just use dao.RecordQuery() // // FindRecordsByExpr finds all records by the specified db expression. // // Returns all collection records if no expressions are provided. // // Returns an empty slice if no records are found. // // Example: // // expr1 := dbx.HashExp{"email": "test@example.com"} // expr2 := dbx.NewExp("LOWER(username) = {:username}", dbx.Params{"username": "test"}) // dao.FindRecordsByExpr("example", expr1, expr2) func (dao *Dao) FindRecordsByExpr(collectionNameOrId string, exprs ...dbx.Expression) ([]*models.Record, error) { query := dao.RecordQuery(collectionNameOrId) // add only the non-nil expressions for _, expr := range exprs { if expr != nil { query.AndWhere(expr) } } var records []*models.Record if err := query.All(&records); err != nil { return nil, err } return records, nil } // FindFirstRecordByData returns the first found record matching // the provided key-value pair. func (dao *Dao) FindFirstRecordByData( collectionNameOrId string, key string, value any, ) (*models.Record, error) { record := &models.Record{} err := dao.RecordQuery(collectionNameOrId). AndWhere(dbx.HashExp{inflector.Columnify(key): value}). Limit(1). One(record) if err != nil { return nil, err } return record, nil } // FindRecordsByFilter returns limit number of records matching the // provided string filter. // // The sort argument is optional and can be empty string OR the same format // used in the web APIs, eg. "-created,title". // // If the limit argument is <= 0, no limit is applied to the query and // all matching records are returned. // // NB! Don't put untrusted user input in the filter string as it // practically would allow the users to inject their own custom filter. // // Example: // // dao.FindRecordsByFilter("posts", "title ~ 'lorem ipsum' && visible = true", "-created", 10) func (dao *Dao) FindRecordsByFilter( collectionNameOrId string, filter string, sort string, limit int, ) ([]*models.Record, error) { collection, err := dao.FindCollectionByNameOrId(collectionNameOrId) if err != nil { return nil, err } q := dao.RecordQuery(collection) // build a fields resolver and attach the generated conditions to the query // --- resolver := resolvers.NewRecordFieldResolver( dao, collection, // the base collection nil, // no request data true, // allow searching hidden/protected fields like "email" ) expr, err := search.FilterData(filter).BuildExpr(resolver) if err != nil || expr == nil { return nil, errors.New("invalid or empty filter expression") } q.AndWhere(expr) if sort != "" { for _, sortField := range search.ParseSortFromString(sort) { expr, err := sortField.BuildExpr(resolver) if err != nil { return nil, err } if expr != "" { q.AndOrderBy(expr) } } } resolver.UpdateQuery(q) // attaches any adhoc joins and aliases // --- if limit > 0 { q.Limit(int64(limit)) } records := []*models.Record{} if err := q.All(&records); err != nil { return nil, err } return records, nil } // FindFirstRecordByFilter returns the first available record matching the provided filter. // // NB! Don't put untrusted user input in the filter string as it // practically would allow the users to inject their own custom filter. // // Example: // // dao.FindFirstRecordByFilter("posts", "slug='test'") func (dao *Dao) FindFirstRecordByFilter(collectionNameOrId string, filter string) (*models.Record, error) { result, err := dao.FindRecordsByFilter(collectionNameOrId, filter, "", 1) if err != nil { return nil, err } if len(result) == 0 { return nil, sql.ErrNoRows } return result[0], nil } // IsRecordValueUnique checks if the provided key-value pair is a unique Record value. // // For correctness, if the collection is "auth" and the key is "username", // the unique check will be case insensitive. // // NB! Array values (eg. from multiple select fields) are matched // as a serialized json strings (eg. `["a","b"]`), so the value uniqueness // depends on the elements order. Or in other words the following values // are considered different: `[]string{"a","b"}` and `[]string{"b","a"}` func (dao *Dao) IsRecordValueUnique( collectionNameOrId string, key string, value any, excludeIds ...string, ) bool { collection, err := dao.FindCollectionByNameOrId(collectionNameOrId) if err != nil { return false } var expr dbx.Expression if collection.IsAuth() && key == schema.FieldNameUsername { expr = dbx.NewExp("LOWER([["+schema.FieldNameUsername+"]])={:username}", dbx.Params{ "username": strings.ToLower(cast.ToString(value)), }) } else { var normalizedVal any switch val := value.(type) { case []string: normalizedVal = append(types.JsonArray[string]{}, val...) case []any: normalizedVal = append(types.JsonArray[any]{}, val...) default: normalizedVal = val } expr = dbx.HashExp{inflector.Columnify(key): normalizedVal} } query := dao.RecordQuery(collection). Select("count(*)"). AndWhere(expr). Limit(1) if uniqueExcludeIds := list.NonzeroUniques(excludeIds); len(uniqueExcludeIds) > 0 { query.AndWhere(dbx.NotIn(collection.Name+".id", list.ToInterfaceSlice(uniqueExcludeIds)...)) } var exists bool return query.Row(&exists) == nil && !exists } // FindAuthRecordByToken finds the auth record associated with the provided JWT token. // // Returns an error if the JWT token is invalid, expired or not associated to an auth collection record. func (dao *Dao) FindAuthRecordByToken(token string, baseTokenKey string) (*models.Record, error) { unverifiedClaims, err := security.ParseUnverifiedJWT(token) if err != nil { return nil, err } // check required claims id, _ := unverifiedClaims["id"].(string) collectionId, _ := unverifiedClaims["collectionId"].(string) if id == "" || collectionId == "" { return nil, errors.New("missing or invalid token claims") } record, err := dao.FindRecordById(collectionId, id) if err != nil { return nil, err } if !record.Collection().IsAuth() { return nil, errors.New("The token is not associated to an auth collection record.") } verificationKey := record.TokenKey() + baseTokenKey // verify token signature if _, err := security.ParseJWT(token, verificationKey); err != nil { return nil, err } return record, nil } // FindAuthRecordByEmail finds the auth record associated with the provided email. // // Returns an error if it is not an auth collection or the record is not found. func (dao *Dao) FindAuthRecordByEmail(collectionNameOrId string, email string) (*models.Record, error) { collection, err := dao.FindCollectionByNameOrId(collectionNameOrId) if err != nil { return nil, fmt.Errorf("failed to fetch auth collection %q (%w)", collectionNameOrId, err) } if !collection.IsAuth() { return nil, fmt.Errorf("%q is not an auth collection", collectionNameOrId) } record := &models.Record{} err = dao.RecordQuery(collection). AndWhere(dbx.HashExp{schema.FieldNameEmail: email}). Limit(1). One(record) if err != nil { return nil, err } return record, nil } // FindAuthRecordByUsername finds the auth record associated with the provided username (case insensitive). // // Returns an error if it is not an auth collection or the record is not found. func (dao *Dao) FindAuthRecordByUsername(collectionNameOrId string, username string) (*models.Record, error) { collection, err := dao.FindCollectionByNameOrId(collectionNameOrId) if err != nil { return nil, fmt.Errorf("failed to fetch auth collection %q (%w)", collectionNameOrId, err) } if !collection.IsAuth() { return nil, fmt.Errorf("%q is not an auth collection", collectionNameOrId) } record := &models.Record{} err = dao.RecordQuery(collection). AndWhere(dbx.NewExp("LOWER([["+schema.FieldNameUsername+"]])={:username}", dbx.Params{ "username": strings.ToLower(username), })). Limit(1). One(record) if err != nil { return nil, err } return record, nil } // SuggestUniqueAuthRecordUsername checks if the provided username is unique // and return a new "unique" username with appended random numeric part // (eg. "existingName" -> "existingName583"). // // The same username will be returned if the provided string is already unique. func (dao *Dao) SuggestUniqueAuthRecordUsername( collectionNameOrId string, baseUsername string, excludeIds ...string, ) string { username := baseUsername for i := 0; i < 10; i++ { // max 10 attempts isUnique := dao.IsRecordValueUnique( collectionNameOrId, schema.FieldNameUsername, username, excludeIds..., ) if isUnique { break // already unique } username = baseUsername + security.RandomStringWithAlphabet(3+i, "123456789") } return username } // CanAccessRecord checks if a record is allowed to be accessed by the // specified requestInfo and accessRule. // // Rule and db checks are ignored in case requestInfo.Admin is set. // // The returned error indicate that something unexpected happened during // the check (eg. invalid rule or db error). // // The method always return false on invalid access rule or db error. // // Example: // // requestInfo := apis.RequestInfo(c /* echo.Context */) // record, _ := dao.FindRecordById("example", "RECORD_ID") // rule := types.Pointer("@request.auth.id != '' || status = 'public'") // // ... or use one of the record collection's rule, eg. record.Collection().ViewRule // // if ok, _ := dao.CanAccessRecord(record, requestInfo, rule); ok { ... } func (dao *Dao) CanAccessRecord(record *models.Record, requestInfo *models.RequestInfo, accessRule *string) (bool, error) { if requestInfo.Admin != nil { // admins can access everything return true, nil } if accessRule == nil { // only admins can access this record return false, nil } if *accessRule == "" { // empty public rule, aka. everyone can access return true, nil } var exists bool query := dao.RecordQuery(record.Collection()). Select("(1)"). AndWhere(dbx.HashExp{record.Collection().Name + ".id": record.Id}) // parse and apply the access rule filter resolver := resolvers.NewRecordFieldResolver(dao, record.Collection(), requestInfo, true) expr, err := search.FilterData(*accessRule).BuildExpr(resolver) if err != nil { return false, err } resolver.UpdateQuery(query) query.AndWhere(expr) if err := query.Limit(1).Row(&exists); err != nil && !errors.Is(err, sql.ErrNoRows) { return false, err } return exists, nil } // SaveRecord persists the provided Record model in the database. // // If record.IsNew() is true, the method will perform a create, otherwise an update. // To explicitly mark a record for update you can use record.MarkAsNotNew(). func (dao *Dao) SaveRecord(record *models.Record) error { if record.Collection().IsAuth() { if record.Username() == "" { return errors.New("unable to save auth record without username") } // Cross-check that the auth record id is unique for all auth collections. // This is to make sure that the filter `@request.auth.id` always returns a unique id. authCollections, err := dao.FindCollectionsByType(models.CollectionTypeAuth) if err != nil { return fmt.Errorf("unable to fetch the auth collections for cross-id unique check: %w", err) } for _, collection := range authCollections { if record.Collection().Id == collection.Id { continue // skip current collection (sqlite will do the check for us) } isUnique := dao.IsRecordValueUnique(collection.Id, schema.FieldNameId, record.Id) if !isUnique { return errors.New("the auth record ID must be unique across all auth collections") } } } return dao.Save(record) } // DeleteRecord deletes the provided Record model. // // This method will also cascade the delete operation to all linked // relational records (delete or unset, depending on the rel settings). // // The delete operation may fail if the record is part of a required // reference in another record (aka. cannot be deleted or unset). func (dao *Dao) DeleteRecord(record *models.Record) error { // fetch rel references (if any) // // note: the select is outside of the transaction to minimize // SQLITE_BUSY errors when mixing read&write in a single transaction refs, err := dao.FindCollectionReferences(record.Collection()) if err != nil { return err } return dao.RunInTransaction(func(txDao *Dao) error { // manually trigger delete on any linked external auth to ensure // that the `OnModel*` hooks are triggered if record.Collection().IsAuth() { // note: the select is outside of the transaction to minimize // SQLITE_BUSY errors when mixing read&write in a single transaction externalAuths, err := dao.FindAllExternalAuthsByRecord(record) if err != nil { return err } for _, auth := range externalAuths { if err := txDao.DeleteExternalAuth(auth); err != nil { return err } } } // delete the record before the relation references to ensure that there // will be no "A<->B" relations to prevent deadlock when calling DeleteRecord recursively if err := txDao.Delete(record); err != nil { return err } return txDao.cascadeRecordDelete(record, refs) }) } // cascadeRecordDelete triggers cascade deletion for the provided references. // // NB! This method is expected to be called inside a transaction. func (dao *Dao) cascadeRecordDelete(mainRecord *models.Record, refs map[*models.Collection][]*schema.SchemaField) error { uniqueJsonEachAlias := "__je__" + security.PseudorandomString(4) for refCollection, fields := range refs { if refCollection.IsView() { continue // skip view collections } for _, field := range fields { recordTableName := inflector.Columnify(refCollection.Name) prefixedFieldName := recordTableName + "." + inflector.Columnify(field.Name) query := dao.RecordQuery(refCollection).Distinct(true) if opt, ok := field.Options.(schema.MultiValuer); !ok || !opt.IsMultiple() { query.AndWhere(dbx.HashExp{prefixedFieldName: mainRecord.Id}) } else { query.InnerJoin(fmt.Sprintf( `json_each(CASE WHEN json_valid([[%s]]) THEN [[%s]] ELSE json_array([[%s]]) END) as {{%s}}`, prefixedFieldName, prefixedFieldName, prefixedFieldName, uniqueJsonEachAlias, ), dbx.HashExp{uniqueJsonEachAlias + ".value": mainRecord.Id}) } if refCollection.Id == mainRecord.Collection().Id { query.AndWhere(dbx.Not(dbx.HashExp{recordTableName + ".id": mainRecord.Id})) } // trigger cascade for each batchSize rel items until there is none batchSize := 4000 rows := make([]dbx.NullStringMap, 0, batchSize) for { if err := query.Limit(int64(batchSize)).All(&rows); err != nil { return err } total := len(rows) if total == 0 { break } refRecords := models.NewRecordsFromNullStringMaps(refCollection, rows) err := dao.deleteRefRecords(mainRecord, refRecords, field) if err != nil { return err } if total < batchSize { break // no more items } rows = rows[:0] // keep allocated memory } } } return nil } // deleteRefRecords checks if related records has to be deleted (if `CascadeDelete` is set) // OR // just unset the record id from any relation field values (if they are not required). // // NB! This method is expected to be called inside a transaction. func (dao *Dao) deleteRefRecords(mainRecord *models.Record, refRecords []*models.Record, field *schema.SchemaField) error { options, _ := field.Options.(*schema.RelationOptions) if options == nil { return errors.New("relation field options are not initialized") } for _, refRecord := range refRecords { ids := refRecord.GetStringSlice(field.Name) // unset the record id for i := len(ids) - 1; i >= 0; i-- { if ids[i] == mainRecord.Id { ids = append(ids[:i], ids[i+1:]...) break } } // cascade delete the reference // (only if there are no other active references in case of multiple select) if options.CascadeDelete && len(ids) == 0 { if err := dao.DeleteRecord(refRecord); err != nil { return err } // no further actions are needed (the reference is deleted) continue } if field.Required && len(ids) == 0 { return fmt.Errorf("the record cannot be deleted because it is part of a required reference in record %s (%s collection)", refRecord.Id, refRecord.Collection().Name) } // save the reference changes refRecord.Set(field.Name, field.PrepareValue(ids)) if err := dao.SaveRecord(refRecord); err != nil { return err } } return nil }