model_test.go 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. // Copyright (C) 2014 Jakob Borg and other contributors. All rights reserved.
  2. // Use of this source code is governed by an MIT-style license that can be
  3. // found in the LICENSE file.
  4. package model
  5. import (
  6. "bytes"
  7. "fmt"
  8. "os"
  9. "testing"
  10. "time"
  11. "github.com/calmh/syncthing/cid"
  12. "github.com/calmh/syncthing/config"
  13. "github.com/calmh/syncthing/protocol"
  14. "github.com/calmh/syncthing/scanner"
  15. )
  16. var node1, node2 protocol.NodeID
  17. func init() {
  18. node1, _ = protocol.NodeIDFromString("AIR6LPZ-7K4PTTV-UXQSMUU-CPQ5YWH-OEDFIIQ-JUG777G-2YQXXR5-YD6AWQR")
  19. node2, _ = protocol.NodeIDFromString("GYRZZQB-IRNPV4Z-T7TC52W-EQYJ3TT-FDQW6MW-DFLMU42-SSSU6EM-FBK2VAY")
  20. }
  21. var testDataExpected = map[string]scanner.File{
  22. "foo": scanner.File{
  23. Name: "foo",
  24. Flags: 0,
  25. Modified: 0,
  26. Size: 7,
  27. Blocks: []scanner.Block{{Offset: 0x0, Size: 0x7, Hash: []uint8{0xae, 0xc0, 0x70, 0x64, 0x5f, 0xe5, 0x3e, 0xe3, 0xb3, 0x76, 0x30, 0x59, 0x37, 0x61, 0x34, 0xf0, 0x58, 0xcc, 0x33, 0x72, 0x47, 0xc9, 0x78, 0xad, 0xd1, 0x78, 0xb6, 0xcc, 0xdf, 0xb0, 0x1, 0x9f}}},
  28. },
  29. "empty": scanner.File{
  30. Name: "empty",
  31. Flags: 0,
  32. Modified: 0,
  33. Size: 0,
  34. Blocks: []scanner.Block{{Offset: 0x0, Size: 0x0, Hash: []uint8{0xe3, 0xb0, 0xc4, 0x42, 0x98, 0xfc, 0x1c, 0x14, 0x9a, 0xfb, 0xf4, 0xc8, 0x99, 0x6f, 0xb9, 0x24, 0x27, 0xae, 0x41, 0xe4, 0x64, 0x9b, 0x93, 0x4c, 0xa4, 0x95, 0x99, 0x1b, 0x78, 0x52, 0xb8, 0x55}}},
  35. },
  36. "bar": scanner.File{
  37. Name: "bar",
  38. Flags: 0,
  39. Modified: 0,
  40. Size: 10,
  41. Blocks: []scanner.Block{{Offset: 0x0, Size: 0xa, Hash: []uint8{0x2f, 0x72, 0xcc, 0x11, 0xa6, 0xfc, 0xd0, 0x27, 0x1e, 0xce, 0xf8, 0xc6, 0x10, 0x56, 0xee, 0x1e, 0xb1, 0x24, 0x3b, 0xe3, 0x80, 0x5b, 0xf9, 0xa9, 0xdf, 0x98, 0xf9, 0x2f, 0x76, 0x36, 0xb0, 0x5c}}},
  42. },
  43. }
  44. func init() {
  45. // Fix expected test data to match reality
  46. for n, f := range testDataExpected {
  47. fi, _ := os.Stat("testdata/" + n)
  48. f.Flags = uint32(fi.Mode())
  49. f.Modified = fi.ModTime().Unix()
  50. testDataExpected[n] = f
  51. }
  52. }
  53. func TestRequest(t *testing.T) {
  54. m := NewModel("/tmp", &config.Configuration{}, "syncthing", "dev")
  55. m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
  56. m.ScanRepo("default")
  57. bs, err := m.Request(node1, "default", "foo", 0, 6)
  58. if err != nil {
  59. t.Fatal(err)
  60. }
  61. if bytes.Compare(bs, []byte("foobar")) != 0 {
  62. t.Errorf("Incorrect data from request: %q", string(bs))
  63. }
  64. bs, err = m.Request(node1, "default", "../walk.go", 0, 6)
  65. if err == nil {
  66. t.Error("Unexpected nil error on insecure file read")
  67. }
  68. if bs != nil {
  69. t.Errorf("Unexpected non nil data on insecure file read: %q", string(bs))
  70. }
  71. }
  72. func genFiles(n int) []protocol.FileInfo {
  73. files := make([]protocol.FileInfo, n)
  74. t := time.Now().Unix()
  75. for i := 0; i < n; i++ {
  76. files[i] = protocol.FileInfo{
  77. Name: fmt.Sprintf("file%d", i),
  78. Modified: t,
  79. Blocks: []protocol.BlockInfo{{100, []byte("some hash bytes")}},
  80. }
  81. }
  82. return files
  83. }
  84. func BenchmarkIndex10000(b *testing.B) {
  85. m := NewModel("/tmp", nil, "syncthing", "dev")
  86. m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
  87. m.ScanRepo("default")
  88. files := genFiles(10000)
  89. b.ResetTimer()
  90. for i := 0; i < b.N; i++ {
  91. m.Index(node1, "default", files)
  92. }
  93. }
  94. func BenchmarkIndex00100(b *testing.B) {
  95. m := NewModel("/tmp", nil, "syncthing", "dev")
  96. m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
  97. m.ScanRepo("default")
  98. files := genFiles(100)
  99. b.ResetTimer()
  100. for i := 0; i < b.N; i++ {
  101. m.Index(node1, "default", files)
  102. }
  103. }
  104. func BenchmarkIndexUpdate10000f10000(b *testing.B) {
  105. m := NewModel("/tmp", nil, "syncthing", "dev")
  106. m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
  107. m.ScanRepo("default")
  108. files := genFiles(10000)
  109. m.Index(node1, "default", files)
  110. b.ResetTimer()
  111. for i := 0; i < b.N; i++ {
  112. m.IndexUpdate(node1, "default", files)
  113. }
  114. }
  115. func BenchmarkIndexUpdate10000f00100(b *testing.B) {
  116. m := NewModel("/tmp", nil, "syncthing", "dev")
  117. m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
  118. m.ScanRepo("default")
  119. files := genFiles(10000)
  120. m.Index(node1, "default", files)
  121. ufiles := genFiles(100)
  122. b.ResetTimer()
  123. for i := 0; i < b.N; i++ {
  124. m.IndexUpdate(node1, "default", ufiles)
  125. }
  126. }
  127. func BenchmarkIndexUpdate10000f00001(b *testing.B) {
  128. m := NewModel("/tmp", nil, "syncthing", "dev")
  129. m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
  130. m.ScanRepo("default")
  131. files := genFiles(10000)
  132. m.Index(node1, "default", files)
  133. ufiles := genFiles(1)
  134. b.ResetTimer()
  135. for i := 0; i < b.N; i++ {
  136. m.IndexUpdate(node1, "default", ufiles)
  137. }
  138. }
  139. type FakeConnection struct {
  140. id protocol.NodeID
  141. requestData []byte
  142. }
  143. func (FakeConnection) Close() error {
  144. return nil
  145. }
  146. func (f FakeConnection) ID() protocol.NodeID {
  147. return f.id
  148. }
  149. func (f FakeConnection) Option(string) string {
  150. return ""
  151. }
  152. func (FakeConnection) Index(string, []protocol.FileInfo) {}
  153. func (f FakeConnection) Request(repo, name string, offset int64, size int) ([]byte, error) {
  154. return f.requestData, nil
  155. }
  156. func (FakeConnection) ClusterConfig(protocol.ClusterConfigMessage) {}
  157. func (FakeConnection) Ping() bool {
  158. return true
  159. }
  160. func (FakeConnection) Statistics() protocol.Statistics {
  161. return protocol.Statistics{}
  162. }
  163. func BenchmarkRequest(b *testing.B) {
  164. m := NewModel("/tmp", nil, "syncthing", "dev")
  165. m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
  166. m.ScanRepo("default")
  167. const n = 1000
  168. files := make([]protocol.FileInfo, n)
  169. t := time.Now().Unix()
  170. for i := 0; i < n; i++ {
  171. files[i] = protocol.FileInfo{
  172. Name: fmt.Sprintf("file%d", i),
  173. Modified: t,
  174. Blocks: []protocol.BlockInfo{{100, []byte("some hash bytes")}},
  175. }
  176. }
  177. fc := FakeConnection{
  178. id: node1,
  179. requestData: []byte("some data to return"),
  180. }
  181. m.AddConnection(fc, fc)
  182. m.Index(node1, "default", files)
  183. b.ResetTimer()
  184. for i := 0; i < b.N; i++ {
  185. data, err := m.requestGlobal(node1, "default", files[i%n].Name, 0, 32, nil)
  186. if err != nil {
  187. b.Error(err)
  188. }
  189. if data == nil {
  190. b.Error("nil data")
  191. }
  192. }
  193. }
  194. func TestActivityMap(t *testing.T) {
  195. cm := cid.NewMap()
  196. fooID := cm.Get(node1)
  197. if fooID == 0 {
  198. t.Fatal("ID cannot be zero")
  199. }
  200. barID := cm.Get(node2)
  201. if barID == 0 {
  202. t.Fatal("ID cannot be zero")
  203. }
  204. m := make(activityMap)
  205. if node := m.leastBusyNode(1<<fooID, cm); node != node1 {
  206. t.Errorf("Incorrect least busy node %q", node)
  207. }
  208. if node := m.leastBusyNode(1<<barID, cm); node != node2 {
  209. t.Errorf("Incorrect least busy node %q", node)
  210. }
  211. if node := m.leastBusyNode(1<<fooID|1<<barID, cm); node != node1 {
  212. t.Errorf("Incorrect least busy node %q", node)
  213. }
  214. if node := m.leastBusyNode(1<<fooID|1<<barID, cm); node != node2 {
  215. t.Errorf("Incorrect least busy node %q", node)
  216. }
  217. }