db_service.go 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  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. "context"
  9. "fmt"
  10. "log/slog"
  11. "time"
  12. "github.com/jmoiron/sqlx"
  13. "github.com/syncthing/syncthing/internal/db"
  14. "github.com/syncthing/syncthing/internal/slogutil"
  15. "github.com/thejerf/suture/v4"
  16. )
  17. const (
  18. internalMetaPrefix = "dbsvc"
  19. lastMaintKey = "lastMaint"
  20. )
  21. func (s *DB) Service(maintenanceInterval time.Duration) suture.Service {
  22. return newService(s, maintenanceInterval)
  23. }
  24. type Service struct {
  25. sdb *DB
  26. maintenanceInterval time.Duration
  27. internalMeta *db.Typed
  28. }
  29. func (s *Service) String() string {
  30. return fmt.Sprintf("sqlite.service@%p", s)
  31. }
  32. func newService(sdb *DB, maintenanceInterval time.Duration) *Service {
  33. return &Service{
  34. sdb: sdb,
  35. maintenanceInterval: maintenanceInterval,
  36. internalMeta: db.NewTyped(sdb, internalMetaPrefix),
  37. }
  38. }
  39. func (s *Service) Serve(ctx context.Context) error {
  40. // Run periodic maintenance
  41. // Figure out when we last ran maintenance and schedule accordingly. If
  42. // it was never, do it now.
  43. lastMaint, _, _ := s.internalMeta.Time(lastMaintKey)
  44. nextMaint := lastMaint.Add(s.maintenanceInterval)
  45. wait := time.Until(nextMaint)
  46. if wait < 0 {
  47. wait = time.Minute
  48. }
  49. slog.DebugContext(ctx, "Next periodic run due", "after", wait)
  50. timer := time.NewTimer(wait)
  51. for {
  52. select {
  53. case <-ctx.Done():
  54. return ctx.Err()
  55. case <-timer.C:
  56. }
  57. if err := s.periodic(ctx); err != nil {
  58. return wrap(err)
  59. }
  60. timer.Reset(s.maintenanceInterval)
  61. slog.DebugContext(ctx, "Next periodic run due", "after", s.maintenanceInterval)
  62. _ = s.internalMeta.PutTime(lastMaintKey, time.Now())
  63. }
  64. }
  65. func (s *Service) periodic(ctx context.Context) error {
  66. t0 := time.Now()
  67. slog.DebugContext(ctx, "Periodic start")
  68. t1 := time.Now()
  69. defer func() { slog.DebugContext(ctx, "Periodic done in", "t1", time.Since(t1), "t0t1", t1.Sub(t0)) }()
  70. s.sdb.updateLock.Lock()
  71. err := tidy(ctx, s.sdb.sql)
  72. s.sdb.updateLock.Unlock()
  73. if err != nil {
  74. return err
  75. }
  76. return wrap(s.sdb.forEachFolder(func(fdb *folderDB) error {
  77. fdb.updateLock.Lock()
  78. defer fdb.updateLock.Unlock()
  79. if err := garbageCollectOldDeletedLocked(ctx, fdb); err != nil {
  80. return wrap(err)
  81. }
  82. if err := garbageCollectBlocklistsAndBlocksLocked(ctx, fdb); err != nil {
  83. return wrap(err)
  84. }
  85. return tidy(ctx, fdb.sql)
  86. }))
  87. }
  88. func tidy(ctx context.Context, db *sqlx.DB) error {
  89. conn, err := db.Conn(ctx)
  90. if err != nil {
  91. return wrap(err)
  92. }
  93. defer conn.Close()
  94. _, _ = conn.ExecContext(ctx, `ANALYZE`)
  95. _, _ = conn.ExecContext(ctx, `PRAGMA optimize`)
  96. _, _ = conn.ExecContext(ctx, `PRAGMA incremental_vacuum`)
  97. _, _ = conn.ExecContext(ctx, `PRAGMA journal_size_limit = 8388608`)
  98. _, _ = conn.ExecContext(ctx, `PRAGMA wal_checkpoint(TRUNCATE)`)
  99. return nil
  100. }
  101. func garbageCollectOldDeletedLocked(ctx context.Context, fdb *folderDB) error {
  102. l := slog.With("fdb", fdb.baseDB)
  103. if fdb.deleteRetention <= 0 {
  104. slog.DebugContext(ctx, "Delete retention is infinite, skipping cleanup")
  105. return nil
  106. }
  107. // Remove deleted files that are marked as not needed (we have processed
  108. // them) and they were deleted more than MaxDeletedFileAge ago.
  109. l.DebugContext(ctx, "Forgetting deleted files", "retention", fdb.deleteRetention)
  110. res, err := fdb.stmt(`
  111. DELETE FROM files
  112. WHERE deleted AND modified < ? AND local_flags & {{.FlagLocalNeeded}} == 0
  113. `).Exec(time.Now().Add(-fdb.deleteRetention).UnixNano())
  114. if err != nil {
  115. return wrap(err)
  116. }
  117. if aff, err := res.RowsAffected(); err == nil {
  118. l.DebugContext(ctx, "Removed old deleted file records", "affected", aff)
  119. }
  120. return nil
  121. }
  122. func garbageCollectBlocklistsAndBlocksLocked(ctx context.Context, fdb *folderDB) error {
  123. // Remove all blocklists not referred to by any files and, by extension,
  124. // any blocks not referred to by a blocklist. This is an expensive
  125. // operation when run normally, especially if there are a lot of blocks
  126. // to collect.
  127. //
  128. // We make this orders of magnitude faster by disabling foreign keys for
  129. // the transaction and doing the cleanup manually. This requires using
  130. // an explicit connection and disabling foreign keys before starting the
  131. // transaction. We make sure to clean up on the way out.
  132. conn, err := fdb.sql.Connx(ctx)
  133. if err != nil {
  134. return wrap(err)
  135. }
  136. defer conn.Close()
  137. if _, err := conn.ExecContext(ctx, `PRAGMA foreign_keys = 0`); err != nil {
  138. return wrap(err)
  139. }
  140. defer func() { //nolint:contextcheck
  141. _, _ = conn.ExecContext(context.Background(), `PRAGMA foreign_keys = 1`)
  142. }()
  143. tx, err := conn.BeginTxx(ctx, nil)
  144. if err != nil {
  145. return wrap(err)
  146. }
  147. defer tx.Rollback() //nolint:errcheck
  148. if res, err := tx.ExecContext(ctx, `
  149. DELETE FROM blocklists
  150. WHERE NOT EXISTS (
  151. SELECT 1 FROM files WHERE files.blocklist_hash = blocklists.blocklist_hash
  152. )`); err != nil {
  153. return wrap(err, "delete blocklists")
  154. } else {
  155. slog.DebugContext(ctx, "Blocklist GC", "fdb", fdb.baseName, "result", slogutil.Expensive(func() any {
  156. rows, err := res.RowsAffected()
  157. if err != nil {
  158. return slogutil.Error(err)
  159. }
  160. return slog.Int64("rows", rows)
  161. }))
  162. }
  163. if res, err := tx.ExecContext(ctx, `
  164. DELETE FROM blocks
  165. WHERE NOT EXISTS (
  166. SELECT 1 FROM blocklists WHERE blocklists.blocklist_hash = blocks.blocklist_hash
  167. )`); err != nil {
  168. return wrap(err, "delete blocks")
  169. } else {
  170. slog.DebugContext(ctx, "Blocks GC", "fdb", fdb.baseName, "result", slogutil.Expensive(func() any {
  171. rows, err := res.RowsAffected()
  172. if err != nil {
  173. return slogutil.Error(err)
  174. }
  175. return slog.Int64("rows", rows)
  176. }))
  177. }
  178. return wrap(tx.Commit())
  179. }