basedb.go 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  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. "path/filepath"
  12. "strconv"
  13. "strings"
  14. "sync"
  15. "text/template"
  16. "time"
  17. "github.com/jmoiron/sqlx"
  18. "github.com/syncthing/syncthing/lib/build"
  19. "github.com/syncthing/syncthing/lib/protocol"
  20. )
  21. const currentSchemaVersion = 3
  22. //go:embed sql/**
  23. var embedded embed.FS
  24. type baseDB struct {
  25. path string
  26. baseName string
  27. sql *sqlx.DB
  28. updateLock sync.Mutex
  29. updatePoints int
  30. checkpointsCount int
  31. statementsMut sync.RWMutex
  32. statements map[string]*sqlx.Stmt
  33. tplInput map[string]any
  34. }
  35. func openBase(path string, maxConns int, pragmas, schemaScripts, migrationScripts []string) (*baseDB, error) {
  36. // Open the database with options to enable foreign keys and recursive
  37. // triggers (needed for the delete+insert triggers on row replace).
  38. sqlDB, err := sqlx.Open(dbDriver, "file:"+path+"?"+commonOptions)
  39. if err != nil {
  40. return nil, wrap(err)
  41. }
  42. sqlDB.SetMaxOpenConns(maxConns)
  43. for _, pragma := range pragmas {
  44. if _, err := sqlDB.Exec("PRAGMA " + pragma); err != nil {
  45. return nil, wrap(err, "PRAGMA "+pragma)
  46. }
  47. }
  48. db := &baseDB{
  49. path: path,
  50. baseName: filepath.Base(path),
  51. sql: sqlDB,
  52. statements: make(map[string]*sqlx.Stmt),
  53. tplInput: map[string]any{
  54. "FlagLocalUnsupported": protocol.FlagLocalUnsupported,
  55. "FlagLocalIgnored": protocol.FlagLocalIgnored,
  56. "FlagLocalMustRescan": protocol.FlagLocalMustRescan,
  57. "FlagLocalReceiveOnly": protocol.FlagLocalReceiveOnly,
  58. "FlagLocalGlobal": protocol.FlagLocalGlobal,
  59. "FlagLocalNeeded": protocol.FlagLocalNeeded,
  60. "FlagLocalRemoteInvalid": protocol.FlagLocalRemoteInvalid,
  61. "LocalInvalidFlags": protocol.LocalInvalidFlags,
  62. "SyncthingVersion": build.LongVersion,
  63. },
  64. }
  65. for _, script := range schemaScripts {
  66. if err := db.runScripts(script); err != nil {
  67. return nil, wrap(err)
  68. }
  69. }
  70. ver, _ := db.getAppliedSchemaVersion()
  71. if ver.SchemaVersion > 0 {
  72. filter := func(scr string) bool {
  73. scr = filepath.Base(scr)
  74. nstr, _, ok := strings.Cut(scr, "-")
  75. if !ok {
  76. return false
  77. }
  78. n, err := strconv.ParseInt(nstr, 10, 32)
  79. if err != nil {
  80. return false
  81. }
  82. return int(n) > ver.SchemaVersion
  83. }
  84. for _, script := range migrationScripts {
  85. if err := db.runScripts(script, filter); err != nil {
  86. return nil, wrap(err)
  87. }
  88. }
  89. }
  90. // Set the current schema version, if not already set
  91. if err := db.setAppliedSchemaVersion(currentSchemaVersion); err != nil {
  92. return nil, wrap(err)
  93. }
  94. return db, nil
  95. }
  96. func (s *baseDB) Close() error {
  97. s.updateLock.Lock()
  98. s.statementsMut.Lock()
  99. defer s.updateLock.Unlock()
  100. defer s.statementsMut.Unlock()
  101. for _, stmt := range s.statements {
  102. stmt.Close()
  103. }
  104. return wrap(s.sql.Close())
  105. }
  106. var tplFuncs = template.FuncMap{
  107. "or": func(vs ...int) int {
  108. v := vs[0]
  109. for _, ov := range vs[1:] {
  110. v |= ov
  111. }
  112. return v
  113. },
  114. }
  115. // stmt returns a prepared statement for the given SQL string, after
  116. // applying local template expansions. The statement is cached.
  117. func (s *baseDB) stmt(tpl string) stmt {
  118. tpl = strings.TrimSpace(tpl)
  119. // Fast concurrent lookup of cached statement
  120. s.statementsMut.RLock()
  121. stmt, ok := s.statements[tpl]
  122. s.statementsMut.RUnlock()
  123. if ok {
  124. return stmt
  125. }
  126. // On miss, take the full lock, check again
  127. s.statementsMut.Lock()
  128. defer s.statementsMut.Unlock()
  129. stmt, ok = s.statements[tpl]
  130. if ok {
  131. return stmt
  132. }
  133. // Prepare and cache
  134. stmt, err := s.sql.Preparex(s.expandTemplateVars(tpl))
  135. if err != nil {
  136. return failedStmt{err}
  137. }
  138. s.statements[tpl] = stmt
  139. return stmt
  140. }
  141. // expandTemplateVars just applies template expansions to the template
  142. // string, or dies trying
  143. func (s *baseDB) expandTemplateVars(tpl string) string {
  144. var sb strings.Builder
  145. compTpl := template.Must(template.New("tpl").Funcs(tplFuncs).Parse(tpl))
  146. if err := compTpl.Execute(&sb, s.tplInput); err != nil {
  147. panic("bug: bad template: " + err.Error())
  148. }
  149. return sb.String()
  150. }
  151. type stmt interface {
  152. Exec(args ...any) (sql.Result, error)
  153. Get(dest any, args ...any) error
  154. Queryx(args ...any) (*sqlx.Rows, error)
  155. Select(dest any, args ...any) error
  156. }
  157. type failedStmt struct {
  158. err error
  159. }
  160. func (f failedStmt) Exec(_ ...any) (sql.Result, error) { return nil, f.err }
  161. func (f failedStmt) Get(_ any, _ ...any) error { return f.err }
  162. func (f failedStmt) Queryx(_ ...any) (*sqlx.Rows, error) { return nil, f.err }
  163. func (f failedStmt) Select(_ any, _ ...any) error { return f.err }
  164. func (s *baseDB) runScripts(glob string, filter ...func(s string) bool) error {
  165. scripts, err := fs.Glob(embedded, glob)
  166. if err != nil {
  167. return wrap(err)
  168. }
  169. tx, err := s.sql.Begin()
  170. if err != nil {
  171. return wrap(err)
  172. }
  173. defer tx.Rollback() //nolint:errcheck
  174. nextScript:
  175. for _, scr := range scripts {
  176. for _, fn := range filter {
  177. if !fn(scr) {
  178. continue nextScript
  179. }
  180. }
  181. bs, err := fs.ReadFile(embedded, scr)
  182. if err != nil {
  183. return wrap(err, scr)
  184. }
  185. // SQLite requires one statement per exec, so we split the init
  186. // files on lines containing only a semicolon and execute them
  187. // separately. We require it on a separate line because there are
  188. // also statement-internal semicolons in the triggers.
  189. for _, stmt := range strings.Split(string(bs), "\n;") {
  190. if _, err := tx.Exec(s.expandTemplateVars(stmt)); err != nil {
  191. return wrap(err, stmt)
  192. }
  193. }
  194. }
  195. return wrap(tx.Commit())
  196. }
  197. type schemaVersion struct {
  198. SchemaVersion int
  199. AppliedAt int64
  200. SyncthingVersion string
  201. }
  202. func (s *schemaVersion) AppliedTime() time.Time {
  203. return time.Unix(0, s.AppliedAt)
  204. }
  205. func (s *baseDB) setAppliedSchemaVersion(ver int) error {
  206. _, err := s.stmt(`
  207. INSERT OR IGNORE INTO schemamigrations (schema_version, applied_at, syncthing_version)
  208. VALUES (?, ?, ?)
  209. `).Exec(ver, time.Now().UnixNano(), build.LongVersion)
  210. return wrap(err)
  211. }
  212. func (s *baseDB) getAppliedSchemaVersion() (schemaVersion, error) {
  213. var v schemaVersion
  214. err := s.stmt(`
  215. SELECT schema_version as schemaversion, applied_at as appliedat, syncthing_version as syncthingversion FROM schemamigrations
  216. ORDER BY schema_version DESC
  217. LIMIT 1
  218. `).Get(&v)
  219. return v, wrap(err)
  220. }