db_open.go 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  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. "os"
  10. "path/filepath"
  11. "strconv"
  12. "strings"
  13. "text/template"
  14. "github.com/jmoiron/sqlx"
  15. "github.com/syncthing/syncthing/lib/build"
  16. "github.com/syncthing/syncthing/lib/protocol"
  17. )
  18. const maxDBConns = 128
  19. func Open(path string) (*DB, error) {
  20. // Open the database with options to enable foreign keys and recursive
  21. // triggers (needed for the delete+insert triggers on row replace).
  22. sqlDB, err := sqlx.Open(dbDriver, "file:"+path+"?"+commonOptions)
  23. if err != nil {
  24. return nil, wrap(err)
  25. }
  26. sqlDB.SetMaxOpenConns(maxDBConns)
  27. if _, err := sqlDB.Exec(`PRAGMA journal_mode = WAL`); err != nil {
  28. return nil, wrap(err, "PRAGMA journal_mode")
  29. }
  30. if _, err := sqlDB.Exec(`PRAGMA optimize = 0x10002`); err != nil {
  31. // https://www.sqlite.org/pragma.html#pragma_optimize
  32. return nil, wrap(err, "PRAGMA optimize")
  33. }
  34. if _, err := sqlDB.Exec(`PRAGMA journal_size_limit = 6144000`); err != nil {
  35. // https://www.powersync.com/blog/sqlite-optimizations-for-ultra-high-performance
  36. return nil, wrap(err, "PRAGMA journal_size_limit")
  37. }
  38. return openCommon(sqlDB)
  39. }
  40. // Open the database with options suitable for the migration inserts. This
  41. // is not a safe mode of operation for normal processing, use only for bulk
  42. // inserts with a close afterwards.
  43. func OpenForMigration(path string) (*DB, error) {
  44. sqlDB, err := sqlx.Open(dbDriver, "file:"+path+"?"+commonOptions)
  45. if err != nil {
  46. return nil, wrap(err, "open")
  47. }
  48. sqlDB.SetMaxOpenConns(1)
  49. if _, err := sqlDB.Exec(`PRAGMA foreign_keys = 0`); err != nil {
  50. return nil, wrap(err, "PRAGMA foreign_keys")
  51. }
  52. if _, err := sqlDB.Exec(`PRAGMA journal_mode = OFF`); err != nil {
  53. return nil, wrap(err, "PRAGMA journal_mode")
  54. }
  55. if _, err := sqlDB.Exec(`PRAGMA synchronous = 0`); err != nil {
  56. return nil, wrap(err, "PRAGMA synchronous")
  57. }
  58. return openCommon(sqlDB)
  59. }
  60. func OpenTemp() (*DB, error) {
  61. // SQLite has a memory mode, but it works differently with concurrency
  62. // compared to what we need with the WAL mode. So, no memory databases
  63. // for now.
  64. dir, err := os.MkdirTemp("", "syncthing-db")
  65. if err != nil {
  66. return nil, wrap(err)
  67. }
  68. path := filepath.Join(dir, "db")
  69. l.Debugln("Test DB in", path)
  70. return Open(path)
  71. }
  72. func openCommon(sqlDB *sqlx.DB) (*DB, error) {
  73. if _, err := sqlDB.Exec(`PRAGMA auto_vacuum = INCREMENTAL`); err != nil {
  74. return nil, wrap(err, "PRAGMA auto_vacuum")
  75. }
  76. if _, err := sqlDB.Exec(`PRAGMA default_temp_store = MEMORY`); err != nil {
  77. return nil, wrap(err, "PRAGMA default_temp_store")
  78. }
  79. if _, err := sqlDB.Exec(`PRAGMA temp_store = MEMORY`); err != nil {
  80. return nil, wrap(err, "PRAGMA temp_store")
  81. }
  82. db := &DB{
  83. sql: sqlDB,
  84. statements: make(map[string]*sqlx.Stmt),
  85. }
  86. if err := db.runScripts("sql/schema/*"); err != nil {
  87. return nil, wrap(err)
  88. }
  89. ver, _ := db.getAppliedSchemaVersion()
  90. if ver.SchemaVersion > 0 {
  91. filter := func(scr string) bool {
  92. scr = filepath.Base(scr)
  93. nstr, _, ok := strings.Cut(scr, "-")
  94. if !ok {
  95. return false
  96. }
  97. n, err := strconv.ParseInt(nstr, 10, 32)
  98. if err != nil {
  99. return false
  100. }
  101. return int(n) > ver.SchemaVersion
  102. }
  103. if err := db.runScripts("sql/migrations/*", filter); err != nil {
  104. return nil, wrap(err)
  105. }
  106. }
  107. // Touch device IDs that should always exist and have a low index
  108. // numbers, and will never change
  109. db.localDeviceIdx, _ = db.deviceIdxLocked(protocol.LocalDeviceID)
  110. // Set the current schema version, if not already set
  111. if err := db.setAppliedSchemaVersion(currentSchemaVersion); err != nil {
  112. return nil, wrap(err)
  113. }
  114. db.tplInput = map[string]any{
  115. "FlagLocalUnsupported": protocol.FlagLocalUnsupported,
  116. "FlagLocalIgnored": protocol.FlagLocalIgnored,
  117. "FlagLocalMustRescan": protocol.FlagLocalMustRescan,
  118. "FlagLocalReceiveOnly": protocol.FlagLocalReceiveOnly,
  119. "FlagLocalGlobal": protocol.FlagLocalGlobal,
  120. "FlagLocalNeeded": protocol.FlagLocalNeeded,
  121. "LocalDeviceIdx": db.localDeviceIdx,
  122. "SyncthingVersion": build.LongVersion,
  123. }
  124. return db, nil
  125. }
  126. var tplFuncs = template.FuncMap{
  127. "or": func(vs ...int) int {
  128. v := vs[0]
  129. for _, ov := range vs[1:] {
  130. v |= ov
  131. }
  132. return v
  133. },
  134. }
  135. // stmt returns a prepared statement for the given SQL string, after
  136. // applying local template expansions. The statement is cached.
  137. func (s *DB) stmt(tpl string) stmt {
  138. tpl = strings.TrimSpace(tpl)
  139. // Fast concurrent lookup of cached statement
  140. s.statementsMut.RLock()
  141. stmt, ok := s.statements[tpl]
  142. s.statementsMut.RUnlock()
  143. if ok {
  144. return stmt
  145. }
  146. // On miss, take the full lock, check again
  147. s.statementsMut.Lock()
  148. defer s.statementsMut.Unlock()
  149. stmt, ok = s.statements[tpl]
  150. if ok {
  151. return stmt
  152. }
  153. // Apply template expansions
  154. var sb strings.Builder
  155. compTpl := template.Must(template.New("tpl").Funcs(tplFuncs).Parse(tpl))
  156. if err := compTpl.Execute(&sb, s.tplInput); err != nil {
  157. panic("bug: bad template: " + err.Error())
  158. }
  159. // Prepare and cache
  160. stmt, err := s.sql.Preparex(sb.String())
  161. if err != nil {
  162. return failedStmt{err}
  163. }
  164. s.statements[tpl] = stmt
  165. return stmt
  166. }
  167. type stmt interface {
  168. Exec(args ...any) (sql.Result, error)
  169. Get(dest any, args ...any) error
  170. Queryx(args ...any) (*sqlx.Rows, error)
  171. Select(dest any, args ...any) error
  172. }
  173. type failedStmt struct {
  174. err error
  175. }
  176. func (f failedStmt) Exec(_ ...any) (sql.Result, error) { return nil, f.err }
  177. func (f failedStmt) Get(_ any, _ ...any) error { return f.err }
  178. func (f failedStmt) Queryx(_ ...any) (*sqlx.Rows, error) { return nil, f.err }
  179. func (f failedStmt) Select(_ any, _ ...any) error { return f.err }