testutils_test.go 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. // Copyright (C) 2016 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 model
  7. import (
  8. "io/ioutil"
  9. "os"
  10. "testing"
  11. "time"
  12. "github.com/syncthing/syncthing/lib/config"
  13. "github.com/syncthing/syncthing/lib/db"
  14. "github.com/syncthing/syncthing/lib/db/backend"
  15. "github.com/syncthing/syncthing/lib/events"
  16. "github.com/syncthing/syncthing/lib/fs"
  17. "github.com/syncthing/syncthing/lib/protocol"
  18. )
  19. var (
  20. myID, device1, device2 protocol.DeviceID
  21. defaultCfgWrapper config.Wrapper
  22. defaultFolderConfig config.FolderConfiguration
  23. defaultFs fs.Filesystem
  24. defaultCfg config.Configuration
  25. defaultAutoAcceptCfg config.Configuration
  26. )
  27. func init() {
  28. myID, _ = protocol.DeviceIDFromString("ZNWFSWE-RWRV2BD-45BLMCV-LTDE2UR-4LJDW6J-R5BPWEB-TXD27XJ-IZF5RA4")
  29. device1, _ = protocol.DeviceIDFromString("AIR6LPZ-7K4PTTV-UXQSMUU-CPQ5YWH-OEDFIIQ-JUG777G-2YQXXR5-YD6AWQR")
  30. device2, _ = protocol.DeviceIDFromString("GYRZZQB-IRNPV4Z-T7TC52W-EQYJ3TT-FDQW6MW-DFLMU42-SSSU6EM-FBK2VAY")
  31. defaultFs = fs.NewFilesystem(fs.FilesystemTypeBasic, "testdata")
  32. defaultFolderConfig = testFolderConfig("testdata")
  33. defaultCfgWrapper = createTmpWrapper(config.New(myID))
  34. _, _ = defaultCfgWrapper.SetDevice(config.NewDeviceConfiguration(device1, "device1"))
  35. _, _ = defaultCfgWrapper.SetFolder(defaultFolderConfig)
  36. opts := defaultCfgWrapper.Options()
  37. opts.KeepTemporariesH = 1
  38. _, _ = defaultCfgWrapper.SetOptions(opts)
  39. defaultCfg = defaultCfgWrapper.RawCopy()
  40. defaultAutoAcceptCfg = config.Configuration{
  41. Devices: []config.DeviceConfiguration{
  42. {
  43. DeviceID: myID, // self
  44. },
  45. {
  46. DeviceID: device1,
  47. AutoAcceptFolders: true,
  48. },
  49. {
  50. DeviceID: device2,
  51. AutoAcceptFolders: true,
  52. },
  53. },
  54. Options: config.OptionsConfiguration{
  55. DefaultFolderPath: ".",
  56. },
  57. }
  58. }
  59. func tmpDefaultWrapper() (config.Wrapper, config.FolderConfiguration) {
  60. w := createTmpWrapper(defaultCfgWrapper.RawCopy())
  61. fcfg := testFolderConfigTmp()
  62. _, _ = w.SetFolder(fcfg)
  63. return w, fcfg
  64. }
  65. func testFolderConfigTmp() config.FolderConfiguration {
  66. tmpDir := createTmpDir()
  67. return testFolderConfig(tmpDir)
  68. }
  69. func testFolderConfig(path string) config.FolderConfiguration {
  70. cfg := config.NewFolderConfiguration(myID, "default", "default", fs.FilesystemTypeBasic, path)
  71. cfg.FSWatcherEnabled = false
  72. cfg.Devices = append(cfg.Devices, config.FolderDeviceConfiguration{DeviceID: device1})
  73. return cfg
  74. }
  75. func setupModelWithConnection() (*model, *fakeConnection, config.FolderConfiguration) {
  76. w, fcfg := tmpDefaultWrapper()
  77. m, fc := setupModelWithConnectionFromWrapper(w)
  78. return m, fc, fcfg
  79. }
  80. func setupModelWithConnectionFromWrapper(w config.Wrapper) (*model, *fakeConnection) {
  81. m := setupModel(w)
  82. fc := addFakeConn(m, device1)
  83. fc.folder = "default"
  84. _ = m.ScanFolder("default")
  85. return m, fc
  86. }
  87. func setupModel(w config.Wrapper) *model {
  88. db := db.NewLowlevel(backend.OpenMemory())
  89. m := newModel(w, myID, "syncthing", "dev", db, nil)
  90. m.ServeBackground()
  91. m.ScanFolders()
  92. return m
  93. }
  94. func newModel(cfg config.Wrapper, id protocol.DeviceID, clientName, clientVersion string, ldb *db.Lowlevel, protectedFiles []string) *model {
  95. evLogger := events.NewLogger()
  96. m := NewModel(cfg, id, clientName, clientVersion, ldb, protectedFiles, evLogger).(*model)
  97. go evLogger.Serve()
  98. return m
  99. }
  100. func cleanupModel(m *model) {
  101. m.Stop()
  102. m.db.Close()
  103. m.evLogger.Stop()
  104. os.Remove(m.cfg.ConfigPath())
  105. }
  106. func cleanupModelAndRemoveDir(m *model, dir string) {
  107. cleanupModel(m)
  108. os.RemoveAll(dir)
  109. }
  110. func createTmpDir() string {
  111. tmpDir, err := ioutil.TempDir("", "syncthing_testFolder-")
  112. if err != nil {
  113. panic("Failed to create temporary testing dir")
  114. }
  115. return tmpDir
  116. }
  117. type alwaysChangedKey struct {
  118. fs fs.Filesystem
  119. name string
  120. }
  121. // alwaysChanges is an ignore.ChangeDetector that always returns true on Changed()
  122. type alwaysChanged struct {
  123. seen map[alwaysChangedKey]struct{}
  124. }
  125. func newAlwaysChanged() *alwaysChanged {
  126. return &alwaysChanged{
  127. seen: make(map[alwaysChangedKey]struct{}),
  128. }
  129. }
  130. func (c *alwaysChanged) Remember(fs fs.Filesystem, name string, _ time.Time) {
  131. c.seen[alwaysChangedKey{fs, name}] = struct{}{}
  132. }
  133. func (c *alwaysChanged) Reset() {
  134. c.seen = make(map[alwaysChangedKey]struct{})
  135. }
  136. func (c *alwaysChanged) Seen(fs fs.Filesystem, name string) bool {
  137. _, ok := c.seen[alwaysChangedKey{fs, name}]
  138. return ok
  139. }
  140. func (c *alwaysChanged) Changed() bool {
  141. return true
  142. }
  143. func localSize(t *testing.T, m Model, folder string) db.Counts {
  144. t.Helper()
  145. snap := dbSnapshot(t, m, folder)
  146. defer snap.Release()
  147. return snap.LocalSize()
  148. }
  149. func globalSize(t *testing.T, m Model, folder string) db.Counts {
  150. t.Helper()
  151. snap := dbSnapshot(t, m, folder)
  152. defer snap.Release()
  153. return snap.GlobalSize()
  154. }
  155. func receiveOnlyChangedSize(t *testing.T, m Model, folder string) db.Counts {
  156. t.Helper()
  157. snap := dbSnapshot(t, m, folder)
  158. defer snap.Release()
  159. return snap.ReceiveOnlyChangedSize()
  160. }
  161. func needSize(t *testing.T, m Model, folder string) db.Counts {
  162. t.Helper()
  163. snap := dbSnapshot(t, m, folder)
  164. defer snap.Release()
  165. return snap.NeedSize()
  166. }
  167. func dbSnapshot(t *testing.T, m Model, folder string) *db.Snapshot {
  168. t.Helper()
  169. snap, err := m.DBSnapshot(folder)
  170. if err != nil {
  171. t.Fatal(err)
  172. }
  173. return snap
  174. }