database_test.go 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. // Copyright (C) 2018 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 main
  7. import (
  8. "context"
  9. "fmt"
  10. "testing"
  11. "time"
  12. "github.com/syncthing/syncthing/lib/protocol"
  13. )
  14. func TestDatabaseGetSet(t *testing.T) {
  15. db := newInMemoryStore(t.TempDir(), 0, nil)
  16. ctx, cancel := context.WithCancel(context.Background())
  17. go db.Serve(ctx)
  18. defer cancel()
  19. // Check missing record
  20. rec, err := db.get(&protocol.EmptyDeviceID)
  21. if err != nil {
  22. t.Error("not found should not be an error")
  23. }
  24. if len(rec.Addresses) != 0 {
  25. t.Error("addresses should be empty")
  26. }
  27. // Set up a clock
  28. now := time.Now()
  29. tc := &testClock{now}
  30. db.clock = tc
  31. // Put a record
  32. rec.Addresses = []DatabaseAddress{
  33. {Address: "tcp://1.2.3.4:5", Expires: tc.Now().Add(time.Minute).UnixNano()},
  34. }
  35. if err := db.put(&protocol.EmptyDeviceID, rec); err != nil {
  36. t.Fatal(err)
  37. }
  38. // Verify it
  39. rec, err = db.get(&protocol.EmptyDeviceID)
  40. if err != nil {
  41. t.Fatal(err)
  42. }
  43. if len(rec.Addresses) != 1 {
  44. t.Log(rec.Addresses)
  45. t.Fatal("should have one address")
  46. }
  47. if rec.Addresses[0].Address != "tcp://1.2.3.4:5" {
  48. t.Log(rec.Addresses)
  49. t.Error("incorrect address")
  50. }
  51. // Wind the clock one half expiry, and merge in a new address
  52. tc.wind(30 * time.Second)
  53. addrs := []DatabaseAddress{
  54. {Address: "tcp://6.7.8.9:0", Expires: tc.Now().Add(time.Minute).UnixNano()},
  55. }
  56. if err := db.merge(&protocol.EmptyDeviceID, addrs, tc.Now().UnixNano()); err != nil {
  57. t.Fatal(err)
  58. }
  59. // Verify it
  60. rec, err = db.get(&protocol.EmptyDeviceID)
  61. if err != nil {
  62. t.Fatal(err)
  63. }
  64. if len(rec.Addresses) != 2 {
  65. t.Log(rec.Addresses)
  66. t.Fatal("should have two addresses")
  67. }
  68. if rec.Addresses[0].Address != "tcp://1.2.3.4:5" {
  69. t.Log(rec.Addresses)
  70. t.Error("incorrect address[0]")
  71. }
  72. if rec.Addresses[1].Address != "tcp://6.7.8.9:0" {
  73. t.Log(rec.Addresses)
  74. t.Error("incorrect address[1]")
  75. }
  76. // Pass the first expiry time
  77. tc.wind(45 * time.Second)
  78. // Verify it
  79. rec, err = db.get(&protocol.EmptyDeviceID)
  80. if err != nil {
  81. t.Fatal(err)
  82. }
  83. if len(rec.Addresses) != 1 {
  84. t.Log(rec.Addresses)
  85. t.Fatal("should have one address")
  86. }
  87. if rec.Addresses[0].Address != "tcp://6.7.8.9:0" {
  88. t.Log(rec.Addresses)
  89. t.Error("incorrect address")
  90. }
  91. // Set an address
  92. addrs = []DatabaseAddress{
  93. {Address: "tcp://6.7.8.9:0", Expires: tc.Now().Add(time.Minute).UnixNano()},
  94. }
  95. if err := db.merge(&protocol.GlobalDeviceID, addrs, tc.Now().UnixNano()); err != nil {
  96. t.Fatal(err)
  97. }
  98. // Verify it
  99. rec, err = db.get(&protocol.GlobalDeviceID)
  100. if err != nil {
  101. t.Fatal(err)
  102. }
  103. if len(rec.Addresses) != 1 {
  104. t.Log(rec.Addresses)
  105. t.Fatal("should have one address")
  106. }
  107. }
  108. func TestFilter(t *testing.T) {
  109. // all cases are expired with t=10
  110. cases := []struct {
  111. a []DatabaseAddress
  112. b []DatabaseAddress
  113. }{
  114. {
  115. a: nil,
  116. b: nil,
  117. },
  118. {
  119. a: []DatabaseAddress{{Address: "a", Expires: 9}, {Address: "b", Expires: 9}, {Address: "c", Expires: 9}},
  120. b: []DatabaseAddress{},
  121. },
  122. {
  123. a: []DatabaseAddress{{Address: "a", Expires: 10}},
  124. b: []DatabaseAddress{{Address: "a", Expires: 10}},
  125. },
  126. {
  127. a: []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 10}, {Address: "c", Expires: 10}},
  128. b: []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 10}, {Address: "c", Expires: 10}},
  129. },
  130. {
  131. a: []DatabaseAddress{{Address: "a", Expires: 5}, {Address: "b", Expires: 15}, {Address: "c", Expires: 5}, {Address: "d", Expires: 15}, {Address: "e", Expires: 5}},
  132. b: []DatabaseAddress{{Address: "b", Expires: 15}, {Address: "d", Expires: 15}},
  133. },
  134. }
  135. for _, tc := range cases {
  136. res := expire(tc.a, time.Unix(0, 10))
  137. if fmt.Sprint(res) != fmt.Sprint(tc.b) {
  138. t.Errorf("Incorrect result %v, expected %v", res, tc.b)
  139. }
  140. }
  141. }
  142. func TestMerge(t *testing.T) {
  143. cases := []struct {
  144. a, b, res []DatabaseAddress
  145. }{
  146. {nil, nil, nil},
  147. {
  148. nil,
  149. []DatabaseAddress{{Address: "a", Expires: 10}},
  150. []DatabaseAddress{{Address: "a", Expires: 10}},
  151. },
  152. {
  153. nil,
  154. []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 10}, {Address: "c", Expires: 10}},
  155. []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 10}, {Address: "c", Expires: 10}},
  156. },
  157. {
  158. []DatabaseAddress{{Address: "a", Expires: 10}},
  159. []DatabaseAddress{{Address: "a", Expires: 15}},
  160. []DatabaseAddress{{Address: "a", Expires: 15}},
  161. },
  162. {
  163. []DatabaseAddress{{Address: "a", Expires: 10}},
  164. []DatabaseAddress{{Address: "b", Expires: 15}},
  165. []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}},
  166. },
  167. {
  168. []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}},
  169. []DatabaseAddress{{Address: "a", Expires: 15}, {Address: "b", Expires: 15}},
  170. []DatabaseAddress{{Address: "a", Expires: 15}, {Address: "b", Expires: 15}},
  171. },
  172. {
  173. []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}},
  174. []DatabaseAddress{{Address: "b", Expires: 15}, {Address: "c", Expires: 20}},
  175. []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}, {Address: "c", Expires: 20}},
  176. },
  177. {
  178. []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}},
  179. []DatabaseAddress{{Address: "b", Expires: 5}, {Address: "c", Expires: 20}},
  180. []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}, {Address: "c", Expires: 20}},
  181. },
  182. {
  183. []DatabaseAddress{{Address: "y", Expires: 10}, {Address: "z", Expires: 10}},
  184. []DatabaseAddress{{Address: "a", Expires: 5}, {Address: "b", Expires: 15}},
  185. []DatabaseAddress{{Address: "a", Expires: 5}, {Address: "b", Expires: 15}, {Address: "y", Expires: 10}, {Address: "z", Expires: 10}},
  186. },
  187. {
  188. []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}, {Address: "d", Expires: 10}},
  189. []DatabaseAddress{{Address: "b", Expires: 5}, {Address: "c", Expires: 20}},
  190. []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}, {Address: "c", Expires: 20}, {Address: "d", Expires: 10}},
  191. },
  192. }
  193. for _, tc := range cases {
  194. rec := merge(DatabaseRecord{Addresses: tc.a}, DatabaseRecord{Addresses: tc.b})
  195. if fmt.Sprint(rec.Addresses) != fmt.Sprint(tc.res) {
  196. t.Errorf("Incorrect result %v, expected %v", rec.Addresses, tc.res)
  197. }
  198. rec = merge(DatabaseRecord{Addresses: tc.b}, DatabaseRecord{Addresses: tc.a})
  199. if fmt.Sprint(rec.Addresses) != fmt.Sprint(tc.res) {
  200. t.Errorf("Incorrect result %v, expected %v", rec.Addresses, tc.res)
  201. }
  202. }
  203. }
  204. func BenchmarkMergeEqual(b *testing.B) {
  205. for i := 0; i < b.N; i++ {
  206. ar := []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}}
  207. br := []DatabaseAddress{{Address: "a", Expires: 15}, {Address: "b", Expires: 10}}
  208. res := merge(DatabaseRecord{Addresses: ar}, DatabaseRecord{Addresses: br})
  209. if len(res.Addresses) != 2 {
  210. b.Fatal("wrong length")
  211. }
  212. if res.Addresses[0].Address != "a" || res.Addresses[1].Address != "b" {
  213. b.Fatal("wrong address")
  214. }
  215. if res.Addresses[0].Expires != 15 || res.Addresses[1].Expires != 15 {
  216. b.Fatal("wrong expiry")
  217. }
  218. }
  219. b.ReportAllocs() // should be zero per operation
  220. }
  221. type testClock struct {
  222. now time.Time
  223. }
  224. func (t *testClock) wind(d time.Duration) {
  225. t.now = t.now.Add(d)
  226. }
  227. func (t *testClock) Now() time.Time {
  228. t.now = t.now.Add(time.Nanosecond)
  229. return t.now
  230. }