2014-06-01 20:50:14 +00:00
|
|
|
// Copyright (C) 2014 Jakob Borg and other contributors. All rights reserved.
|
|
|
|
// Use of this source code is governed by an MIT-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2014-05-15 03:26:55 +00:00
|
|
|
package model
|
2014-03-02 22:58:14 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2014-03-28 13:36:57 +00:00
|
|
|
"github.com/calmh/syncthing/cid"
|
2014-05-15 03:47:11 +00:00
|
|
|
"github.com/calmh/syncthing/config"
|
2014-03-02 22:58:14 +00:00
|
|
|
"github.com/calmh/syncthing/protocol"
|
2014-03-08 22:02:01 +00:00
|
|
|
"github.com/calmh/syncthing/scanner"
|
2014-03-02 22:58:14 +00:00
|
|
|
)
|
|
|
|
|
2014-06-29 23:42:03 +00:00
|
|
|
var node1, node2 protocol.NodeID
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
node1, _ = protocol.NodeIDFromString("AIR6LPZ-7K4PTTV-UXQSMUU-CPQ5YWH-OEDFIIQ-JUG777G-2YQXXR5-YD6AWQR")
|
|
|
|
node2, _ = protocol.NodeIDFromString("GYRZZQB-IRNPV4Z-T7TC52W-EQYJ3TT-FDQW6MW-DFLMU42-SSSU6EM-FBK2VAY")
|
|
|
|
}
|
|
|
|
|
2014-03-08 22:02:01 +00:00
|
|
|
var testDataExpected = map[string]scanner.File{
|
|
|
|
"foo": scanner.File{
|
2014-03-02 22:58:14 +00:00
|
|
|
Name: "foo",
|
|
|
|
Flags: 0,
|
|
|
|
Modified: 0,
|
|
|
|
Size: 7,
|
2014-03-08 22:02:01 +00:00
|
|
|
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}}},
|
2014-03-02 22:58:14 +00:00
|
|
|
},
|
2014-03-08 22:02:01 +00:00
|
|
|
"empty": scanner.File{
|
2014-03-02 22:58:14 +00:00
|
|
|
Name: "empty",
|
|
|
|
Flags: 0,
|
|
|
|
Modified: 0,
|
|
|
|
Size: 0,
|
2014-03-08 22:02:01 +00:00
|
|
|
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}}},
|
2014-03-02 22:58:14 +00:00
|
|
|
},
|
2014-03-08 22:02:01 +00:00
|
|
|
"bar": scanner.File{
|
2014-03-02 22:58:14 +00:00
|
|
|
Name: "bar",
|
|
|
|
Flags: 0,
|
|
|
|
Modified: 0,
|
|
|
|
Size: 10,
|
2014-03-08 22:02:01 +00:00
|
|
|
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}}},
|
2014-03-02 22:58:14 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
// Fix expected test data to match reality
|
|
|
|
for n, f := range testDataExpected {
|
|
|
|
fi, _ := os.Stat("testdata/" + n)
|
|
|
|
f.Flags = uint32(fi.Mode())
|
|
|
|
f.Modified = fi.ModTime().Unix()
|
|
|
|
testDataExpected[n] = f
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRequest(t *testing.T) {
|
2014-05-15 03:47:11 +00:00
|
|
|
m := NewModel("/tmp", &config.Configuration{}, "syncthing", "dev")
|
2014-05-23 13:54:45 +00:00
|
|
|
m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
|
2014-03-30 19:59:40 +00:00
|
|
|
m.ScanRepo("default")
|
2014-03-02 22:58:14 +00:00
|
|
|
|
2014-06-29 23:42:03 +00:00
|
|
|
bs, err := m.Request(node1, "default", "foo", 0, 6)
|
2014-03-02 22:58:14 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if bytes.Compare(bs, []byte("foobar")) != 0 {
|
|
|
|
t.Errorf("Incorrect data from request: %q", string(bs))
|
|
|
|
}
|
|
|
|
|
2014-06-29 23:42:03 +00:00
|
|
|
bs, err = m.Request(node1, "default", "../walk.go", 0, 6)
|
2014-03-02 22:58:14 +00:00
|
|
|
if err == nil {
|
|
|
|
t.Error("Unexpected nil error on insecure file read")
|
|
|
|
}
|
|
|
|
if bs != nil {
|
|
|
|
t.Errorf("Unexpected non nil data on insecure file read: %q", string(bs))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func genFiles(n int) []protocol.FileInfo {
|
|
|
|
files := make([]protocol.FileInfo, n)
|
|
|
|
t := time.Now().Unix()
|
|
|
|
for i := 0; i < n; i++ {
|
|
|
|
files[i] = protocol.FileInfo{
|
|
|
|
Name: fmt.Sprintf("file%d", i),
|
|
|
|
Modified: t,
|
|
|
|
Blocks: []protocol.BlockInfo{{100, []byte("some hash bytes")}},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return files
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkIndex10000(b *testing.B) {
|
2014-05-15 03:47:11 +00:00
|
|
|
m := NewModel("/tmp", nil, "syncthing", "dev")
|
2014-05-23 13:54:45 +00:00
|
|
|
m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
|
2014-03-30 19:59:40 +00:00
|
|
|
m.ScanRepo("default")
|
2014-03-02 22:58:14 +00:00
|
|
|
files := genFiles(10000)
|
|
|
|
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
2014-06-29 23:42:03 +00:00
|
|
|
m.Index(node1, "default", files)
|
2014-03-02 22:58:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkIndex00100(b *testing.B) {
|
2014-05-15 03:47:11 +00:00
|
|
|
m := NewModel("/tmp", nil, "syncthing", "dev")
|
2014-05-23 13:54:45 +00:00
|
|
|
m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
|
2014-03-30 19:59:40 +00:00
|
|
|
m.ScanRepo("default")
|
2014-03-02 22:58:14 +00:00
|
|
|
files := genFiles(100)
|
|
|
|
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
2014-06-29 23:42:03 +00:00
|
|
|
m.Index(node1, "default", files)
|
2014-03-02 22:58:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkIndexUpdate10000f10000(b *testing.B) {
|
2014-05-15 03:47:11 +00:00
|
|
|
m := NewModel("/tmp", nil, "syncthing", "dev")
|
2014-05-23 13:54:45 +00:00
|
|
|
m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
|
2014-03-30 19:59:40 +00:00
|
|
|
m.ScanRepo("default")
|
2014-03-02 22:58:14 +00:00
|
|
|
files := genFiles(10000)
|
2014-06-29 23:42:03 +00:00
|
|
|
m.Index(node1, "default", files)
|
2014-03-02 22:58:14 +00:00
|
|
|
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
2014-06-29 23:42:03 +00:00
|
|
|
m.IndexUpdate(node1, "default", files)
|
2014-03-02 22:58:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkIndexUpdate10000f00100(b *testing.B) {
|
2014-05-15 03:47:11 +00:00
|
|
|
m := NewModel("/tmp", nil, "syncthing", "dev")
|
2014-05-23 13:54:45 +00:00
|
|
|
m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
|
2014-03-30 19:59:40 +00:00
|
|
|
m.ScanRepo("default")
|
2014-03-02 22:58:14 +00:00
|
|
|
files := genFiles(10000)
|
2014-06-29 23:42:03 +00:00
|
|
|
m.Index(node1, "default", files)
|
2014-03-02 22:58:14 +00:00
|
|
|
|
|
|
|
ufiles := genFiles(100)
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
2014-06-29 23:42:03 +00:00
|
|
|
m.IndexUpdate(node1, "default", ufiles)
|
2014-03-02 22:58:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkIndexUpdate10000f00001(b *testing.B) {
|
2014-05-15 03:47:11 +00:00
|
|
|
m := NewModel("/tmp", nil, "syncthing", "dev")
|
2014-05-23 13:54:45 +00:00
|
|
|
m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
|
2014-03-30 19:59:40 +00:00
|
|
|
m.ScanRepo("default")
|
2014-03-02 22:58:14 +00:00
|
|
|
files := genFiles(10000)
|
2014-06-29 23:42:03 +00:00
|
|
|
m.Index(node1, "default", files)
|
2014-03-02 22:58:14 +00:00
|
|
|
|
|
|
|
ufiles := genFiles(1)
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
2014-06-29 23:42:03 +00:00
|
|
|
m.IndexUpdate(node1, "default", ufiles)
|
2014-03-02 22:58:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type FakeConnection struct {
|
2014-06-29 23:42:03 +00:00
|
|
|
id protocol.NodeID
|
2014-03-02 22:58:14 +00:00
|
|
|
requestData []byte
|
|
|
|
}
|
|
|
|
|
|
|
|
func (FakeConnection) Close() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-06-29 23:42:03 +00:00
|
|
|
func (f FakeConnection) ID() protocol.NodeID {
|
|
|
|
return f.id
|
2014-03-02 22:58:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (f FakeConnection) Option(string) string {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func (FakeConnection) Index(string, []protocol.FileInfo) {}
|
|
|
|
|
|
|
|
func (f FakeConnection) Request(repo, name string, offset int64, size int) ([]byte, error) {
|
|
|
|
return f.requestData, nil
|
|
|
|
}
|
|
|
|
|
2014-04-13 13:28:26 +00:00
|
|
|
func (FakeConnection) ClusterConfig(protocol.ClusterConfigMessage) {}
|
|
|
|
|
2014-03-02 22:58:14 +00:00
|
|
|
func (FakeConnection) Ping() bool {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (FakeConnection) Statistics() protocol.Statistics {
|
|
|
|
return protocol.Statistics{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkRequest(b *testing.B) {
|
2014-05-15 03:47:11 +00:00
|
|
|
m := NewModel("/tmp", nil, "syncthing", "dev")
|
2014-05-23 13:54:45 +00:00
|
|
|
m.AddRepo(config.RepositoryConfiguration{ID: "default", Directory: "testdata"})
|
2014-03-30 19:59:40 +00:00
|
|
|
m.ScanRepo("default")
|
2014-03-02 22:58:14 +00:00
|
|
|
|
|
|
|
const n = 1000
|
|
|
|
files := make([]protocol.FileInfo, n)
|
|
|
|
t := time.Now().Unix()
|
|
|
|
for i := 0; i < n; i++ {
|
|
|
|
files[i] = protocol.FileInfo{
|
|
|
|
Name: fmt.Sprintf("file%d", i),
|
|
|
|
Modified: t,
|
|
|
|
Blocks: []protocol.BlockInfo{{100, []byte("some hash bytes")}},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fc := FakeConnection{
|
2014-06-29 23:42:03 +00:00
|
|
|
id: node1,
|
2014-03-02 22:58:14 +00:00
|
|
|
requestData: []byte("some data to return"),
|
|
|
|
}
|
|
|
|
m.AddConnection(fc, fc)
|
2014-06-29 23:42:03 +00:00
|
|
|
m.Index(node1, "default", files)
|
2014-03-02 22:58:14 +00:00
|
|
|
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
2014-06-29 23:42:03 +00:00
|
|
|
data, err := m.requestGlobal(node1, "default", files[i%n].Name, 0, 32, nil)
|
2014-03-02 22:58:14 +00:00
|
|
|
if err != nil {
|
|
|
|
b.Error(err)
|
|
|
|
}
|
|
|
|
if data == nil {
|
|
|
|
b.Error("nil data")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-03-28 13:36:57 +00:00
|
|
|
|
|
|
|
func TestActivityMap(t *testing.T) {
|
|
|
|
cm := cid.NewMap()
|
2014-06-29 23:42:03 +00:00
|
|
|
fooID := cm.Get(node1)
|
2014-03-28 13:36:57 +00:00
|
|
|
if fooID == 0 {
|
|
|
|
t.Fatal("ID cannot be zero")
|
|
|
|
}
|
2014-06-29 23:42:03 +00:00
|
|
|
barID := cm.Get(node2)
|
2014-03-28 13:36:57 +00:00
|
|
|
if barID == 0 {
|
|
|
|
t.Fatal("ID cannot be zero")
|
|
|
|
}
|
|
|
|
|
|
|
|
m := make(activityMap)
|
2014-06-29 23:42:03 +00:00
|
|
|
if node := m.leastBusyNode(1<<fooID, cm); node != node1 {
|
2014-03-28 13:36:57 +00:00
|
|
|
t.Errorf("Incorrect least busy node %q", node)
|
|
|
|
}
|
2014-06-29 23:42:03 +00:00
|
|
|
if node := m.leastBusyNode(1<<barID, cm); node != node2 {
|
2014-03-28 13:36:57 +00:00
|
|
|
t.Errorf("Incorrect least busy node %q", node)
|
|
|
|
}
|
2014-06-29 23:42:03 +00:00
|
|
|
if node := m.leastBusyNode(1<<fooID|1<<barID, cm); node != node1 {
|
2014-03-28 13:36:57 +00:00
|
|
|
t.Errorf("Incorrect least busy node %q", node)
|
|
|
|
}
|
2014-06-29 23:42:03 +00:00
|
|
|
if node := m.leastBusyNode(1<<fooID|1<<barID, cm); node != node2 {
|
2014-03-28 13:36:57 +00:00
|
|
|
t.Errorf("Incorrect least busy node %q", node)
|
|
|
|
}
|
|
|
|
}
|