config_test.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481
  1. // Copyright (C) 2014 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 http://mozilla.org/MPL/2.0/.
  6. package config
  7. import (
  8. "bytes"
  9. "encoding/json"
  10. "fmt"
  11. "os"
  12. "reflect"
  13. "runtime"
  14. "testing"
  15. "github.com/syncthing/protocol"
  16. )
  17. var device1, device2, device3, device4 protocol.DeviceID
  18. func init() {
  19. device1, _ = protocol.DeviceIDFromString("AIR6LPZ7K4PTTUXQSMUUCPQ5YWOEDFIIQJUG7772YQXXR5YD6AWQ")
  20. device2, _ = protocol.DeviceIDFromString("GYRZZQB-IRNPV4Z-T7TC52W-EQYJ3TT-FDQW6MW-DFLMU42-SSSU6EM-FBK2VAY")
  21. device3, _ = protocol.DeviceIDFromString("LGFPDIT-7SKNNJL-VJZA4FC-7QNCRKA-CE753K7-2BW5QDK-2FOZ7FR-FEP57QJ")
  22. device4, _ = protocol.DeviceIDFromString("P56IOI7-MZJNU2Y-IQGDREY-DM2MGTI-MGL3BXN-PQ6W5BM-TBBZ4TJ-XZWICQ2")
  23. }
  24. func TestDefaultValues(t *testing.T) {
  25. expected := OptionsConfiguration{
  26. ListenAddress: []string{"0.0.0.0:22000"},
  27. GlobalAnnServers: []string{"udp4://announce.syncthing.net:22026", "udp6://announce-v6.syncthing.net:22026"},
  28. GlobalAnnEnabled: true,
  29. LocalAnnEnabled: true,
  30. LocalAnnPort: 21025,
  31. LocalAnnMCAddr: "[ff32::5222]:21026",
  32. MaxSendKbps: 0,
  33. MaxRecvKbps: 0,
  34. ReconnectIntervalS: 60,
  35. StartBrowser: true,
  36. UPnPEnabled: true,
  37. UPnPLease: 0,
  38. UPnPRenewal: 30,
  39. RestartOnWakeup: true,
  40. AutoUpgradeIntervalH: 12,
  41. KeepTemporariesH: 24,
  42. CacheIgnoredFiles: true,
  43. ProgressUpdateIntervalS: 5,
  44. SymlinksEnabled: true,
  45. LimitBandwidthInLan: false,
  46. }
  47. cfg := New(device1)
  48. if !reflect.DeepEqual(cfg.Options, expected) {
  49. t.Errorf("Default config differs;\n E: %#v\n A: %#v", expected, cfg.Options)
  50. }
  51. }
  52. func TestDeviceConfig(t *testing.T) {
  53. for i := OldestHandledVersion; i <= CurrentVersion; i++ {
  54. os.Remove("testdata/.stfolder")
  55. wr, err := Load(fmt.Sprintf("testdata/v%d.xml", i), device1)
  56. if err != nil {
  57. t.Fatal(err)
  58. }
  59. _, err = os.Stat("testdata/.stfolder")
  60. if i < 6 && err != nil {
  61. t.Fatal(err)
  62. } else if i >= 6 && err == nil {
  63. t.Fatal("Unexpected file")
  64. }
  65. cfg := wr.cfg
  66. expectedFolders := []FolderConfiguration{
  67. {
  68. ID: "test",
  69. Path: "testdata",
  70. Devices: []FolderDeviceConfiguration{{DeviceID: device1}, {DeviceID: device4}},
  71. ReadOnly: true,
  72. RescanIntervalS: 600,
  73. Copiers: 1,
  74. Pullers: 16,
  75. Hashers: 0,
  76. AutoNormalize: true,
  77. },
  78. }
  79. expectedDevices := []DeviceConfiguration{
  80. {
  81. DeviceID: device1,
  82. Name: "node one",
  83. Addresses: []string{"a"},
  84. Compression: protocol.CompressMetadata,
  85. },
  86. {
  87. DeviceID: device4,
  88. Name: "node two",
  89. Addresses: []string{"b"},
  90. Compression: protocol.CompressMetadata,
  91. },
  92. }
  93. expectedDeviceIDs := []protocol.DeviceID{device1, device4}
  94. if cfg.Version != CurrentVersion {
  95. t.Errorf("%d: Incorrect version %d != %d", i, cfg.Version, CurrentVersion)
  96. }
  97. if !reflect.DeepEqual(cfg.Folders, expectedFolders) {
  98. t.Errorf("%d: Incorrect Folders\n A: %#v\n E: %#v", i, cfg.Folders, expectedFolders)
  99. }
  100. if !reflect.DeepEqual(cfg.Devices, expectedDevices) {
  101. t.Errorf("%d: Incorrect Devices\n A: %#v\n E: %#v", i, cfg.Devices, expectedDevices)
  102. }
  103. if !reflect.DeepEqual(cfg.Folders[0].DeviceIDs(), expectedDeviceIDs) {
  104. t.Errorf("%d: Incorrect DeviceIDs\n A: %#v\n E: %#v", i, cfg.Folders[0].DeviceIDs(), expectedDeviceIDs)
  105. }
  106. }
  107. }
  108. func TestNoListenAddress(t *testing.T) {
  109. cfg, err := Load("testdata/nolistenaddress.xml", device1)
  110. if err != nil {
  111. t.Error(err)
  112. }
  113. expected := []string{""}
  114. actual := cfg.Options().ListenAddress
  115. if !reflect.DeepEqual(actual, expected) {
  116. t.Errorf("Unexpected ListenAddress %#v", actual)
  117. }
  118. }
  119. func TestOverriddenValues(t *testing.T) {
  120. expected := OptionsConfiguration{
  121. ListenAddress: []string{":23000"},
  122. GlobalAnnServers: []string{"udp4://syncthing.nym.se:22026"},
  123. GlobalAnnEnabled: false,
  124. LocalAnnEnabled: false,
  125. LocalAnnPort: 42123,
  126. LocalAnnMCAddr: "quux:3232",
  127. MaxSendKbps: 1234,
  128. MaxRecvKbps: 2341,
  129. ReconnectIntervalS: 6000,
  130. StartBrowser: false,
  131. UPnPEnabled: false,
  132. UPnPLease: 60,
  133. UPnPRenewal: 15,
  134. RestartOnWakeup: false,
  135. AutoUpgradeIntervalH: 24,
  136. KeepTemporariesH: 48,
  137. CacheIgnoredFiles: false,
  138. ProgressUpdateIntervalS: 10,
  139. SymlinksEnabled: false,
  140. LimitBandwidthInLan: true,
  141. }
  142. cfg, err := Load("testdata/overridenvalues.xml", device1)
  143. if err != nil {
  144. t.Error(err)
  145. }
  146. if !reflect.DeepEqual(cfg.Options(), expected) {
  147. t.Errorf("Overridden config differs;\n E: %#v\n A: %#v", expected, cfg.Options())
  148. }
  149. }
  150. func TestDeviceAddressesDynamic(t *testing.T) {
  151. name, _ := os.Hostname()
  152. expected := map[protocol.DeviceID]DeviceConfiguration{
  153. device1: {
  154. DeviceID: device1,
  155. Addresses: []string{"dynamic"},
  156. },
  157. device2: {
  158. DeviceID: device2,
  159. Addresses: []string{"dynamic"},
  160. },
  161. device3: {
  162. DeviceID: device3,
  163. Addresses: []string{"dynamic"},
  164. },
  165. device4: {
  166. DeviceID: device4,
  167. Name: name, // Set when auto created
  168. Addresses: []string{"dynamic"},
  169. Compression: protocol.CompressMetadata,
  170. },
  171. }
  172. cfg, err := Load("testdata/deviceaddressesdynamic.xml", device4)
  173. if err != nil {
  174. t.Error(err)
  175. }
  176. actual := cfg.Devices()
  177. if !reflect.DeepEqual(actual, expected) {
  178. t.Errorf("Devices differ;\n E: %#v\n A: %#v", expected, actual)
  179. }
  180. }
  181. func TestDeviceCompression(t *testing.T) {
  182. name, _ := os.Hostname()
  183. expected := map[protocol.DeviceID]DeviceConfiguration{
  184. device1: {
  185. DeviceID: device1,
  186. Addresses: []string{"dynamic"},
  187. Compression: protocol.CompressMetadata,
  188. },
  189. device2: {
  190. DeviceID: device2,
  191. Addresses: []string{"dynamic"},
  192. Compression: protocol.CompressMetadata,
  193. },
  194. device3: {
  195. DeviceID: device3,
  196. Addresses: []string{"dynamic"},
  197. Compression: protocol.CompressNever,
  198. },
  199. device4: {
  200. DeviceID: device4,
  201. Name: name, // Set when auto created
  202. Addresses: []string{"dynamic"},
  203. Compression: protocol.CompressMetadata,
  204. },
  205. }
  206. cfg, err := Load("testdata/devicecompression.xml", device4)
  207. if err != nil {
  208. t.Error(err)
  209. }
  210. actual := cfg.Devices()
  211. if !reflect.DeepEqual(actual, expected) {
  212. t.Errorf("Devices differ;\n E: %#v\n A: %#v", expected, actual)
  213. }
  214. }
  215. func TestDeviceAddressesStatic(t *testing.T) {
  216. name, _ := os.Hostname()
  217. expected := map[protocol.DeviceID]DeviceConfiguration{
  218. device1: {
  219. DeviceID: device1,
  220. Addresses: []string{"192.0.2.1", "192.0.2.2"},
  221. },
  222. device2: {
  223. DeviceID: device2,
  224. Addresses: []string{"192.0.2.3:6070", "[2001:db8::42]:4242"},
  225. },
  226. device3: {
  227. DeviceID: device3,
  228. Addresses: []string{"[2001:db8::44]:4444", "192.0.2.4:6090"},
  229. },
  230. device4: {
  231. DeviceID: device4,
  232. Name: name, // Set when auto created
  233. Addresses: []string{"dynamic"},
  234. Compression: protocol.CompressMetadata,
  235. },
  236. }
  237. cfg, err := Load("testdata/deviceaddressesstatic.xml", device4)
  238. if err != nil {
  239. t.Error(err)
  240. }
  241. actual := cfg.Devices()
  242. if !reflect.DeepEqual(actual, expected) {
  243. t.Errorf("Devices differ;\n E: %#v\n A: %#v", expected, actual)
  244. }
  245. }
  246. func TestVersioningConfig(t *testing.T) {
  247. cfg, err := Load("testdata/versioningconfig.xml", device4)
  248. if err != nil {
  249. t.Error(err)
  250. }
  251. vc := cfg.Folders()["test"].Versioning
  252. if vc.Type != "simple" {
  253. t.Errorf(`vc.Type %q != "simple"`, vc.Type)
  254. }
  255. if l := len(vc.Params); l != 2 {
  256. t.Errorf("len(vc.Params) %d != 2", l)
  257. }
  258. expected := map[string]string{
  259. "foo": "bar",
  260. "baz": "quux",
  261. }
  262. if !reflect.DeepEqual(vc.Params, expected) {
  263. t.Errorf("vc.Params differ;\n E: %#v\n A: %#v", expected, vc.Params)
  264. }
  265. }
  266. func TestIssue1262(t *testing.T) {
  267. cfg, err := Load("testdata/issue-1262.xml", device4)
  268. if err != nil {
  269. t.Error(err)
  270. }
  271. actual := cfg.Folders()["test"].Path
  272. expected := "e:"
  273. if runtime.GOOS == "windows" {
  274. expected = `e:\`
  275. }
  276. if actual != expected {
  277. t.Errorf("%q != %q", actual, expected)
  278. }
  279. }
  280. func TestNewSaveLoad(t *testing.T) {
  281. path := "testdata/temp.xml"
  282. os.Remove(path)
  283. exists := func(path string) bool {
  284. _, err := os.Stat(path)
  285. return err == nil
  286. }
  287. intCfg := New(device1)
  288. cfg := Wrap(path, intCfg)
  289. // To make the equality pass later
  290. cfg.cfg.XMLName.Local = "configuration"
  291. if exists(path) {
  292. t.Error(path, "exists")
  293. }
  294. err := cfg.Save()
  295. if err != nil {
  296. t.Error(err)
  297. }
  298. if !exists(path) {
  299. t.Error(path, "does not exist")
  300. }
  301. cfg2, err := Load(path, device1)
  302. if err != nil {
  303. t.Error(err)
  304. }
  305. if !reflect.DeepEqual(cfg.Raw(), cfg2.Raw()) {
  306. t.Errorf("Configs are not equal;\n E: %#v\n A: %#v", cfg.Raw(), cfg2.Raw())
  307. }
  308. os.Remove(path)
  309. }
  310. func TestPrepare(t *testing.T) {
  311. var cfg Configuration
  312. if cfg.Folders != nil || cfg.Devices != nil || cfg.Options.ListenAddress != nil {
  313. t.Error("Expected nil")
  314. }
  315. cfg.prepare(device1)
  316. if cfg.Folders == nil || cfg.Devices == nil || cfg.Options.ListenAddress == nil {
  317. t.Error("Unexpected nil")
  318. }
  319. }
  320. func TestRequiresRestart(t *testing.T) {
  321. wr, err := Load("testdata/v6.xml", device1)
  322. if err != nil {
  323. t.Fatal(err)
  324. }
  325. cfg := wr.cfg
  326. if ChangeRequiresRestart(cfg, cfg) {
  327. t.Error("No change does not require restart")
  328. }
  329. newCfg := cfg
  330. newCfg.Devices = append(newCfg.Devices, DeviceConfiguration{
  331. DeviceID: device3,
  332. })
  333. if ChangeRequiresRestart(cfg, newCfg) {
  334. t.Error("Adding a device does not require restart")
  335. }
  336. newCfg = cfg
  337. newCfg.Devices = newCfg.Devices[:len(newCfg.Devices)-1]
  338. if !ChangeRequiresRestart(cfg, newCfg) {
  339. t.Error("Removing a device requires restart")
  340. }
  341. newCfg = cfg
  342. newCfg.Folders = append(newCfg.Folders, FolderConfiguration{
  343. ID: "t1",
  344. Path: "t1",
  345. })
  346. if !ChangeRequiresRestart(cfg, newCfg) {
  347. t.Error("Adding a folder requires restart")
  348. }
  349. newCfg = cfg
  350. newCfg.Folders = newCfg.Folders[:len(newCfg.Folders)-1]
  351. if !ChangeRequiresRestart(cfg, newCfg) {
  352. t.Error("Removing a folder requires restart")
  353. }
  354. newCfg = cfg
  355. newFolders := make([]FolderConfiguration, len(cfg.Folders))
  356. copy(newFolders, cfg.Folders)
  357. newCfg.Folders = newFolders
  358. if ChangeRequiresRestart(cfg, newCfg) {
  359. t.Error("No changes done yet")
  360. }
  361. newCfg.Folders[0].Path = "different"
  362. if !ChangeRequiresRestart(cfg, newCfg) {
  363. t.Error("Changing a folder requires restart")
  364. }
  365. newCfg = cfg
  366. newDevices := make([]DeviceConfiguration, len(cfg.Devices))
  367. copy(newDevices, cfg.Devices)
  368. newCfg.Devices = newDevices
  369. if ChangeRequiresRestart(cfg, newCfg) {
  370. t.Error("No changes done yet")
  371. }
  372. newCfg.Devices[0].Name = "different"
  373. if ChangeRequiresRestart(cfg, newCfg) {
  374. t.Error("Changing a device does not require restart")
  375. }
  376. newCfg = cfg
  377. newCfg.Options.GlobalAnnEnabled = !cfg.Options.GlobalAnnEnabled
  378. if !ChangeRequiresRestart(cfg, newCfg) {
  379. t.Error("Changing general options requires restart")
  380. }
  381. newCfg = cfg
  382. newCfg.GUI.UseTLS = !cfg.GUI.UseTLS
  383. if !ChangeRequiresRestart(cfg, newCfg) {
  384. t.Error("Changing GUI options requires restart")
  385. }
  386. }
  387. func TestCopy(t *testing.T) {
  388. wrapper, err := Load("testdata/example.xml", device1)
  389. if err != nil {
  390. t.Fatal(err)
  391. }
  392. cfg := wrapper.Raw()
  393. bsOrig, err := json.MarshalIndent(cfg, "", " ")
  394. if err != nil {
  395. t.Fatal(err)
  396. }
  397. copy := cfg.Copy()
  398. cfg.Devices[0].Addresses[0] = "wrong"
  399. cfg.Folders[0].Devices[0].DeviceID = protocol.DeviceID{0, 1, 2, 3}
  400. cfg.Options.ListenAddress[0] = "wrong"
  401. cfg.GUI.APIKey = "wrong"
  402. bsChanged, err := json.MarshalIndent(cfg, "", " ")
  403. if err != nil {
  404. t.Fatal(err)
  405. }
  406. bsCopy, err := json.MarshalIndent(copy, "", " ")
  407. if err != nil {
  408. t.Fatal(err)
  409. }
  410. if bytes.Compare(bsOrig, bsChanged) == 0 {
  411. t.Error("Config should have changed")
  412. }
  413. if bytes.Compare(bsOrig, bsCopy) != 0 {
  414. //ioutil.WriteFile("a", bsOrig, 0644)
  415. //ioutil.WriteFile("b", bsCopy, 0644)
  416. t.Error("Copy should be unchanged")
  417. }
  418. }