| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419 | 
							- // Copyright (C) 2025 The Syncthing Authors.
 
- //
 
- // This Source Code Form is subject to the terms of the Mozilla Public
 
- // License, v. 2.0. If a copy of the MPL was not distributed with this file,
 
- // You can obtain one at https://mozilla.org/MPL/2.0/.
 
- package sqlite
 
- import (
 
- 	"database/sql"
 
- 	"errors"
 
- 	"fmt"
 
- 	"io"
 
- 	"iter"
 
- 	"log/slog"
 
- 	"path/filepath"
 
- 	"strings"
 
- 	"time"
 
- 	"github.com/syncthing/syncthing/internal/db"
 
- 	"github.com/syncthing/syncthing/lib/config"
 
- 	"github.com/syncthing/syncthing/lib/protocol"
 
- 	"github.com/syncthing/syncthing/lib/rand"
 
- )
 
- var errNoSuchFolder = errors.New("no such folder")
 
- func (s *DB) getFolderDB(folder string, create bool) (*folderDB, error) {
 
- 	// Check for an already open database
 
- 	s.folderDBsMut.RLock()
 
- 	fdb, ok := s.folderDBs[folder]
 
- 	s.folderDBsMut.RUnlock()
 
- 	if ok {
 
- 		return fdb, nil
 
- 	}
 
- 	// Check for an existing database. If we're not supposed to create the
 
- 	// folder, we don't move on if it doesn't already have a database name.
 
- 	var dbns sql.NullString
 
- 	if err := s.stmt(`
 
- 		SELECT database_name FROM folders
 
- 		WHERE folder_id = ?
 
- 	`).Get(&dbns, folder); err != nil && !errors.Is(err, sql.ErrNoRows) {
 
- 		return nil, wrap(err)
 
- 	}
 
- 	var dbName string
 
- 	if dbns.Valid {
 
- 		dbName = dbns.String
 
- 	}
 
- 	if dbName == "" && !create {
 
- 		return nil, errNoSuchFolder
 
- 	}
 
- 	// Create a folder ID and database if it does not already exist
 
- 	s.folderDBsMut.Lock()
 
- 	defer s.folderDBsMut.Unlock()
 
- 	if fdb, ok := s.folderDBs[folder]; ok {
 
- 		return fdb, nil
 
- 	}
 
- 	if dbName == "" {
 
- 		// First time we want to access this folder, need to create a new
 
- 		// folder ID
 
- 		s.updateLock.Lock()
 
- 		defer s.updateLock.Unlock()
 
- 		idx, err := s.folderIdxLocked(folder)
 
- 		if err != nil {
 
- 			return nil, wrap(err)
 
- 		}
 
- 		// The database name is the folder index ID and a random slug.
 
- 		slug := strings.ToLower(rand.String(8))
 
- 		dbName = fmt.Sprintf("folder.%04x-%s.db", idx, slug)
 
- 		if _, err := s.stmt(`UPDATE folders SET database_name = ? WHERE idx = ?`).Exec(dbName, idx); err != nil {
 
- 			return nil, wrap(err, "set name")
 
- 		}
 
- 	}
 
- 	slog.Debug("Folder database opened", "folder", folder, "db", dbName)
 
- 	path := dbName
 
- 	if !filepath.IsAbs(path) {
 
- 		path = filepath.Join(s.pathBase, dbName)
 
- 	}
 
- 	fdb, err := s.folderDBOpener(folder, path, s.deleteRetention)
 
- 	if err != nil {
 
- 		return nil, wrap(err)
 
- 	}
 
- 	s.folderDBs[folder] = fdb
 
- 	return fdb, nil
 
- }
 
- func (s *DB) Update(folder string, device protocol.DeviceID, fs []protocol.FileInfo) error {
 
- 	fdb, err := s.getFolderDB(folder, true)
 
- 	if err != nil {
 
- 		return err
 
- 	}
 
- 	return fdb.Update(device, fs)
 
- }
 
- func (s *DB) GetDeviceFile(folder string, device protocol.DeviceID, file string) (protocol.FileInfo, bool, error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return protocol.FileInfo{}, false, nil
 
- 	}
 
- 	if err != nil {
 
- 		return protocol.FileInfo{}, false, err
 
- 	}
 
- 	return fdb.GetDeviceFile(device, file)
 
- }
 
- func (s *DB) GetGlobalAvailability(folder, file string) ([]protocol.DeviceID, error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return nil, nil
 
- 	}
 
- 	if err != nil {
 
- 		return nil, err
 
- 	}
 
- 	return fdb.GetGlobalAvailability(file)
 
- }
 
- func (s *DB) GetGlobalFile(folder string, file string) (protocol.FileInfo, bool, error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return protocol.FileInfo{}, false, nil
 
- 	}
 
- 	if err != nil {
 
- 		return protocol.FileInfo{}, false, err
 
- 	}
 
- 	return fdb.GetGlobalFile(file)
 
- }
 
- func (s *DB) AllGlobalFiles(folder string) (iter.Seq[db.FileMetadata], func() error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return func(yield func(db.FileMetadata) bool) {}, func() error { return nil }
 
- 	}
 
- 	if err != nil {
 
- 		return func(yield func(db.FileMetadata) bool) {}, func() error { return err }
 
- 	}
 
- 	return fdb.AllGlobalFiles()
 
- }
 
- func (s *DB) AllGlobalFilesPrefix(folder string, prefix string) (iter.Seq[db.FileMetadata], func() error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return func(yield func(db.FileMetadata) bool) {}, func() error { return nil }
 
- 	}
 
- 	if err != nil {
 
- 		return func(yield func(db.FileMetadata) bool) {}, func() error { return err }
 
- 	}
 
- 	return fdb.AllGlobalFilesPrefix(prefix)
 
- }
 
- func (s *DB) AllLocalBlocksWithHash(folder string, hash []byte) (iter.Seq[db.BlockMapEntry], func() error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return func(yield func(db.BlockMapEntry) bool) {}, func() error { return nil }
 
- 	}
 
- 	if err != nil {
 
- 		return func(yield func(db.BlockMapEntry) bool) {}, func() error { return err }
 
- 	}
 
- 	return fdb.AllLocalBlocksWithHash(hash)
 
- }
 
- func (s *DB) AllLocalFiles(folder string, device protocol.DeviceID) (iter.Seq[protocol.FileInfo], func() error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return func(yield func(protocol.FileInfo) bool) {}, func() error { return nil }
 
- 	}
 
- 	if err != nil {
 
- 		return func(yield func(protocol.FileInfo) bool) {}, func() error { return err }
 
- 	}
 
- 	return fdb.AllLocalFiles(device)
 
- }
 
- func (s *DB) AllLocalFilesBySequence(folder string, device protocol.DeviceID, startSeq int64, limit int) (iter.Seq[protocol.FileInfo], func() error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return func(yield func(protocol.FileInfo) bool) {}, func() error { return nil }
 
- 	}
 
- 	if err != nil {
 
- 		return func(yield func(protocol.FileInfo) bool) {}, func() error { return err }
 
- 	}
 
- 	return fdb.AllLocalFilesBySequence(device, startSeq, limit)
 
- }
 
- func (s *DB) AllLocalFilesWithPrefix(folder string, device protocol.DeviceID, prefix string) (iter.Seq[protocol.FileInfo], func() error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return func(yield func(protocol.FileInfo) bool) {}, func() error { return nil }
 
- 	}
 
- 	if err != nil {
 
- 		return func(yield func(protocol.FileInfo) bool) {}, func() error { return err }
 
- 	}
 
- 	return fdb.AllLocalFilesWithPrefix(device, prefix)
 
- }
 
- func (s *DB) AllLocalFilesWithBlocksHash(folder string, h []byte) (iter.Seq[db.FileMetadata], func() error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return func(yield func(db.FileMetadata) bool) {}, func() error { return nil }
 
- 	}
 
- 	if err != nil {
 
- 		return func(yield func(db.FileMetadata) bool) {}, func() error { return err }
 
- 	}
 
- 	return fdb.AllLocalFilesWithBlocksHash(h)
 
- }
 
- func (s *DB) AllNeededGlobalFiles(folder string, device protocol.DeviceID, order config.PullOrder, limit, offset int) (iter.Seq[protocol.FileInfo], func() error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return func(yield func(protocol.FileInfo) bool) {}, func() error { return nil }
 
- 	}
 
- 	if err != nil {
 
- 		return func(yield func(protocol.FileInfo) bool) {}, func() error { return err }
 
- 	}
 
- 	return fdb.AllNeededGlobalFiles(device, order, limit, offset)
 
- }
 
- func (s *DB) DropAllFiles(folder string, device protocol.DeviceID) error {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return nil
 
- 	}
 
- 	if err != nil {
 
- 		return err
 
- 	}
 
- 	return fdb.DropAllFiles(device)
 
- }
 
- func (s *DB) DropFilesNamed(folder string, device protocol.DeviceID, names []string) error {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return nil
 
- 	}
 
- 	if err != nil {
 
- 		return err
 
- 	}
 
- 	return fdb.DropFilesNamed(device, names)
 
- }
 
- func (s *DB) ListDevicesForFolder(folder string) ([]protocol.DeviceID, error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return nil, nil
 
- 	}
 
- 	if err != nil {
 
- 		return nil, err
 
- 	}
 
- 	return fdb.ListDevicesForFolder()
 
- }
 
- func (s *DB) RemoteSequences(folder string) (map[protocol.DeviceID]int64, error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return nil, nil //nolint:nilnil
 
- 	}
 
- 	if err != nil {
 
- 		return nil, err
 
- 	}
 
- 	return fdb.RemoteSequences()
 
- }
 
- func (s *DB) CountGlobal(folder string) (db.Counts, error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return db.Counts{}, nil
 
- 	}
 
- 	if err != nil {
 
- 		return db.Counts{}, err
 
- 	}
 
- 	return fdb.CountGlobal()
 
- }
 
- func (s *DB) CountLocal(folder string, device protocol.DeviceID) (db.Counts, error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return db.Counts{}, nil
 
- 	}
 
- 	if err != nil {
 
- 		return db.Counts{}, err
 
- 	}
 
- 	return fdb.CountLocal(device)
 
- }
 
- func (s *DB) CountNeed(folder string, device protocol.DeviceID) (db.Counts, error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return db.Counts{}, nil
 
- 	}
 
- 	if err != nil {
 
- 		return db.Counts{}, err
 
- 	}
 
- 	return fdb.CountNeed(device)
 
- }
 
- func (s *DB) CountReceiveOnlyChanged(folder string) (db.Counts, error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return db.Counts{}, nil
 
- 	}
 
- 	if err != nil {
 
- 		return db.Counts{}, err
 
- 	}
 
- 	return fdb.CountReceiveOnlyChanged()
 
- }
 
- func (s *DB) DropAllIndexIDs() error {
 
- 	return s.forEachFolder(func(fdb *folderDB) error {
 
- 		return fdb.DropAllIndexIDs()
 
- 	})
 
- }
 
- func (s *DB) GetIndexID(folder string, device protocol.DeviceID) (protocol.IndexID, error) {
 
- 	fdb, err := s.getFolderDB(folder, true)
 
- 	if err != nil {
 
- 		return 0, err
 
- 	}
 
- 	return fdb.GetIndexID(device)
 
- }
 
- func (s *DB) SetIndexID(folder string, device protocol.DeviceID, id protocol.IndexID) error {
 
- 	fdb, err := s.getFolderDB(folder, true)
 
- 	if err != nil {
 
- 		return err
 
- 	}
 
- 	return fdb.SetIndexID(device, id)
 
- }
 
- func (s *DB) GetDeviceSequence(folder string, device protocol.DeviceID) (int64, error) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return 0, nil
 
- 	}
 
- 	if err != nil {
 
- 		return 0, err
 
- 	}
 
- 	return fdb.GetDeviceSequence(device)
 
- }
 
- func (s *DB) DeleteMtime(folder, name string) error {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return nil
 
- 	}
 
- 	if err != nil {
 
- 		return err
 
- 	}
 
- 	return fdb.DeleteMtime(name)
 
- }
 
- func (s *DB) GetMtime(folder, name string) (ondisk, virtual time.Time) {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if errors.Is(err, errNoSuchFolder) {
 
- 		return time.Time{}, time.Time{}
 
- 	}
 
- 	if err != nil {
 
- 		return time.Time{}, time.Time{}
 
- 	}
 
- 	return fdb.GetMtime(name)
 
- }
 
- func (s *DB) PutMtime(folder, name string, ondisk, virtual time.Time) error {
 
- 	fdb, err := s.getFolderDB(folder, true)
 
- 	if err != nil {
 
- 		return err
 
- 	}
 
- 	return fdb.PutMtime(name, ondisk, virtual)
 
- }
 
- func (s *DB) DropDevice(device protocol.DeviceID) error {
 
- 	return s.forEachFolder(func(fdb *folderDB) error {
 
- 		return fdb.DropDevice(device)
 
- 	})
 
- }
 
- func (s *DB) DebugCounts(out io.Writer, folder string) error {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if err != nil {
 
- 		return err
 
- 	}
 
- 	return fdb.DebugCounts(out)
 
- }
 
- func (s *DB) DebugFilePattern(out io.Writer, folder, name string) error {
 
- 	fdb, err := s.getFolderDB(folder, false)
 
- 	if err != nil {
 
- 		return err
 
- 	}
 
- 	return fdb.DebugFilePattern(out, name)
 
- }
 
- // forEachFolder runs the function for each currently open folderDB,
 
- // returning the first error that was encountered.
 
- func (s *DB) forEachFolder(fn func(fdb *folderDB) error) error {
 
- 	folders, err := s.ListFolders()
 
- 	if err != nil {
 
- 		return err
 
- 	}
 
- 	var firstError error
 
- 	for _, folder := range folders {
 
- 		fdb, err := s.getFolderDB(folder, false)
 
- 		if err != nil {
 
- 			if firstError == nil {
 
- 				firstError = err
 
- 			}
 
- 			continue
 
- 		}
 
- 		if err := fn(fdb); err != nil && firstError == nil {
 
- 			firstError = err
 
- 		}
 
- 	}
 
- 	return firstError
 
- }
 
 
  |