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