basedb.go 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. // Copyright (C) 2025 The Syncthing Authors.
  2. //
  3. // This Source Code Form is subject to the terms of the Mozilla Public
  4. // License, v. 2.0. If a copy of the MPL was not distributed with this file,
  5. // You can obtain one at https://mozilla.org/MPL/2.0/.
  6. package sqlite
  7. import (
  8. "database/sql"
  9. "embed"
  10. "io/fs"
  11. "log/slog"
  12. "net/url"
  13. "path/filepath"
  14. "strconv"
  15. "strings"
  16. "sync"
  17. "text/template"
  18. "time"
  19. "github.com/jmoiron/sqlx"
  20. "github.com/syncthing/syncthing/internal/slogutil"
  21. "github.com/syncthing/syncthing/lib/build"
  22. "github.com/syncthing/syncthing/lib/protocol"
  23. )
  24. const (
  25. currentSchemaVersion = 4
  26. applicationIDMain = 0x53546d6e // "STmn", Syncthing main database
  27. applicationIDFolder = 0x53546664 // "STfd", Syncthing folder database
  28. )
  29. //go:embed sql/**
  30. var embedded embed.FS
  31. type baseDB struct {
  32. path string
  33. baseName string
  34. sql *sqlx.DB
  35. updateLock sync.Mutex
  36. updatePoints int
  37. checkpointsCount int
  38. statementsMut sync.RWMutex
  39. statements map[string]*sqlx.Stmt
  40. tplInput map[string]any
  41. }
  42. func openBase(path string, maxConns int, pragmas, schemaScripts, migrationScripts []string) (*baseDB, error) {
  43. // Open the database with options to enable foreign keys and recursive
  44. // triggers (needed for the delete+insert triggers on row replace).
  45. pathURL := url.URL{
  46. Scheme: "file",
  47. Path: fileToUriPath(path),
  48. RawQuery: commonOptions,
  49. }
  50. sqlDB, err := sqlx.Open(dbDriver, pathURL.String())
  51. if err != nil {
  52. return nil, wrap(err)
  53. }
  54. sqlDB.SetMaxOpenConns(maxConns)
  55. for _, pragma := range pragmas {
  56. if _, err := sqlDB.Exec("PRAGMA " + pragma); err != nil {
  57. return nil, wrap(err, "PRAGMA "+pragma)
  58. }
  59. }
  60. db := &baseDB{
  61. path: path,
  62. baseName: filepath.Base(path),
  63. sql: sqlDB,
  64. statements: make(map[string]*sqlx.Stmt),
  65. tplInput: map[string]any{
  66. "FlagLocalUnsupported": protocol.FlagLocalUnsupported,
  67. "FlagLocalIgnored": protocol.FlagLocalIgnored,
  68. "FlagLocalMustRescan": protocol.FlagLocalMustRescan,
  69. "FlagLocalReceiveOnly": protocol.FlagLocalReceiveOnly,
  70. "FlagLocalGlobal": protocol.FlagLocalGlobal,
  71. "FlagLocalNeeded": protocol.FlagLocalNeeded,
  72. "FlagLocalRemoteInvalid": protocol.FlagLocalRemoteInvalid,
  73. "LocalInvalidFlags": protocol.LocalInvalidFlags,
  74. "SyncthingVersion": build.LongVersion,
  75. },
  76. }
  77. tx, err := db.sql.Beginx()
  78. if err != nil {
  79. return nil, wrap(err)
  80. }
  81. defer tx.Rollback()
  82. for _, script := range schemaScripts {
  83. if err := db.runScripts(tx, script); err != nil {
  84. return nil, wrap(err)
  85. }
  86. }
  87. ver, _ := db.getAppliedSchemaVersion(tx)
  88. shouldVacuum := false
  89. if ver.SchemaVersion > 0 {
  90. filter := func(scr string) bool {
  91. scr = filepath.Base(scr)
  92. nstr, _, ok := strings.Cut(scr, "-")
  93. if !ok {
  94. return false
  95. }
  96. n, err := strconv.ParseInt(nstr, 10, 32)
  97. if err != nil {
  98. return false
  99. }
  100. if int(n) > ver.SchemaVersion {
  101. slog.Info("Applying database migration", slogutil.FilePath(db.baseName), slog.String("script", scr))
  102. shouldVacuum = true
  103. return true
  104. }
  105. return false
  106. }
  107. for _, script := range migrationScripts {
  108. if err := db.runScripts(tx, script, filter); err != nil {
  109. return nil, wrap(err)
  110. }
  111. }
  112. }
  113. // Set the current schema version, if not already set
  114. if err := db.setAppliedSchemaVersion(tx, currentSchemaVersion); err != nil {
  115. return nil, wrap(err)
  116. }
  117. if err := tx.Commit(); err != nil {
  118. return nil, wrap(err)
  119. }
  120. if shouldVacuum {
  121. // We applied migrations and should take the opportunity to vaccuum
  122. // the database.
  123. if err := db.vacuumAndOptimize(); err != nil {
  124. return nil, wrap(err)
  125. }
  126. }
  127. return db, nil
  128. }
  129. func fileToUriPath(path string) string {
  130. path = filepath.ToSlash(path)
  131. if (build.IsWindows && len(path) >= 2 && path[1] == ':') ||
  132. (strings.HasPrefix(path, "//") && !strings.HasPrefix(path, "///")) {
  133. // Add an extra leading slash for Windows drive letter or UNC path
  134. path = "/" + path
  135. }
  136. return path
  137. }
  138. func (s *baseDB) Close() error {
  139. s.updateLock.Lock()
  140. s.statementsMut.Lock()
  141. defer s.updateLock.Unlock()
  142. defer s.statementsMut.Unlock()
  143. for _, stmt := range s.statements {
  144. stmt.Close()
  145. }
  146. return wrap(s.sql.Close())
  147. }
  148. var tplFuncs = template.FuncMap{
  149. "or": func(vs ...int) int {
  150. v := vs[0]
  151. for _, ov := range vs[1:] {
  152. v |= ov
  153. }
  154. return v
  155. },
  156. }
  157. // stmt returns a prepared statement for the given SQL string, after
  158. // applying local template expansions. The statement is cached.
  159. func (s *baseDB) stmt(tpl string) stmt {
  160. tpl = strings.TrimSpace(tpl)
  161. // Fast concurrent lookup of cached statement
  162. s.statementsMut.RLock()
  163. stmt, ok := s.statements[tpl]
  164. s.statementsMut.RUnlock()
  165. if ok {
  166. return stmt
  167. }
  168. // On miss, take the full lock, check again
  169. s.statementsMut.Lock()
  170. defer s.statementsMut.Unlock()
  171. stmt, ok = s.statements[tpl]
  172. if ok {
  173. return stmt
  174. }
  175. // Prepare and cache
  176. stmt, err := s.sql.Preparex(s.expandTemplateVars(tpl))
  177. if err != nil {
  178. return failedStmt{err}
  179. }
  180. s.statements[tpl] = stmt
  181. return stmt
  182. }
  183. // expandTemplateVars just applies template expansions to the template
  184. // string, or dies trying
  185. func (s *baseDB) expandTemplateVars(tpl string) string {
  186. var sb strings.Builder
  187. compTpl := template.Must(template.New("tpl").Funcs(tplFuncs).Parse(tpl))
  188. if err := compTpl.Execute(&sb, s.tplInput); err != nil {
  189. panic("bug: bad template: " + err.Error())
  190. }
  191. return sb.String()
  192. }
  193. func (s *baseDB) vacuumAndOptimize() error {
  194. stmts := []string{
  195. "VACUUM;",
  196. "PRAGMA optimize;",
  197. "PRAGMA wal_checkpoint(truncate);",
  198. }
  199. for _, stmt := range stmts {
  200. if _, err := s.sql.Exec(stmt); err != nil {
  201. return wrap(err, stmt)
  202. }
  203. }
  204. return nil
  205. }
  206. type stmt interface {
  207. Exec(args ...any) (sql.Result, error)
  208. Get(dest any, args ...any) error
  209. Queryx(args ...any) (*sqlx.Rows, error)
  210. Select(dest any, args ...any) error
  211. }
  212. type failedStmt struct {
  213. err error
  214. }
  215. func (f failedStmt) Exec(_ ...any) (sql.Result, error) { return nil, f.err }
  216. func (f failedStmt) Get(_ any, _ ...any) error { return f.err }
  217. func (f failedStmt) Queryx(_ ...any) (*sqlx.Rows, error) { return nil, f.err }
  218. func (f failedStmt) Select(_ any, _ ...any) error { return f.err }
  219. func (s *baseDB) runScripts(tx *sqlx.Tx, glob string, filter ...func(s string) bool) error {
  220. scripts, err := fs.Glob(embedded, glob)
  221. if err != nil {
  222. return wrap(err)
  223. }
  224. nextScript:
  225. for _, scr := range scripts {
  226. for _, fn := range filter {
  227. if !fn(scr) {
  228. continue nextScript
  229. }
  230. }
  231. bs, err := fs.ReadFile(embedded, scr)
  232. if err != nil {
  233. return wrap(err, scr)
  234. }
  235. // SQLite requires one statement per exec, so we split the init
  236. // files on lines containing only a semicolon and execute them
  237. // separately. We require it on a separate line because there are
  238. // also statement-internal semicolons in the triggers.
  239. for _, stmt := range strings.Split(string(bs), "\n;") {
  240. if _, err := tx.Exec(s.expandTemplateVars(stmt)); err != nil {
  241. return wrap(err, stmt)
  242. }
  243. }
  244. }
  245. return nil
  246. }
  247. type schemaVersion struct {
  248. SchemaVersion int
  249. AppliedAt int64
  250. SyncthingVersion string
  251. }
  252. func (s *schemaVersion) AppliedTime() time.Time {
  253. return time.Unix(0, s.AppliedAt)
  254. }
  255. func (s *baseDB) setAppliedSchemaVersion(tx *sqlx.Tx, ver int) error {
  256. _, err := tx.Exec(`
  257. INSERT OR IGNORE INTO schemamigrations (schema_version, applied_at, syncthing_version)
  258. VALUES (?, ?, ?)
  259. `, ver, time.Now().UnixNano(), build.LongVersion)
  260. return wrap(err)
  261. }
  262. func (s *baseDB) getAppliedSchemaVersion(tx *sqlx.Tx) (schemaVersion, error) {
  263. var v schemaVersion
  264. err := tx.Get(&v, `
  265. SELECT schema_version as schemaversion, applied_at as appliedat, syncthing_version as syncthingversion FROM schemamigrations
  266. ORDER BY schema_version DESC
  267. LIMIT 1
  268. `)
  269. return v, wrap(err)
  270. }