2014-07-12 22:45:33 +00:00
|
|
|
// Copyright (C) 2014 Jakob Borg and Contributors (see the CONTRIBUTORS file).
|
|
|
|
// All rights reserved. Use of this source code is governed by an MIT-style
|
|
|
|
// license that can be found in the LICENSE file.
|
2014-06-01 20:50:14 +00:00
|
|
|
|
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-08-01 14:35:37 +00:00
|
|
|
"github.com/syncthing/syncthing/config"
|
|
|
|
"github.com/syncthing/syncthing/protocol"
|
2014-07-06 12:46:48 +00:00
|
|
|
"github.com/syndtr/goleveldb/leveldb"
|
|
|
|
"github.com/syndtr/goleveldb/leveldb/storage"
|
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-07-12 21:06:48 +00:00
|
|
|
var testDataExpected = map[string]protocol.FileInfo{
|
|
|
|
"foo": protocol.FileInfo{
|
2014-03-02 22:58:14 +00:00
|
|
|
Name: "foo",
|
|
|
|
Flags: 0,
|
|
|
|
Modified: 0,
|
2014-07-12 21:06:48 +00:00
|
|
|
Blocks: []protocol.BlockInfo{{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-07-12 21:06:48 +00:00
|
|
|
"empty": protocol.FileInfo{
|
2014-03-02 22:58:14 +00:00
|
|
|
Name: "empty",
|
|
|
|
Flags: 0,
|
|
|
|
Modified: 0,
|
2014-07-12 21:06:48 +00:00
|
|
|
Blocks: []protocol.BlockInfo{{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-07-12 21:06:48 +00:00
|
|
|
"bar": protocol.FileInfo{
|
2014-03-02 22:58:14 +00:00
|
|
|
Name: "bar",
|
|
|
|
Flags: 0,
|
|
|
|
Modified: 0,
|
2014-07-12 21:06:48 +00:00
|
|
|
Blocks: []protocol.BlockInfo{{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-07-06 12:46:48 +00:00
|
|
|
db, _ := leveldb.Open(storage.NewMemStorage(), nil)
|
2014-08-14 22:15:26 +00:00
|
|
|
m := NewModel("/tmp", &config.Configuration{}, "node", "syncthing", "dev", db)
|
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,
|
2014-07-12 21:06:48 +00:00
|
|
|
Blocks: []protocol.BlockInfo{{0, 100, []byte("some hash bytes")}},
|
2014-03-02 22:58:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return files
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkIndex10000(b *testing.B) {
|
2014-07-06 12:46:48 +00:00
|
|
|
db, _ := leveldb.Open(storage.NewMemStorage(), nil)
|
2014-08-14 22:15:26 +00:00
|
|
|
m := NewModel("/tmp", nil, "node", "syncthing", "dev", db)
|
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-07-06 12:46:48 +00:00
|
|
|
db, _ := leveldb.Open(storage.NewMemStorage(), nil)
|
2014-08-14 22:15:26 +00:00
|
|
|
m := NewModel("/tmp", nil, "node", "syncthing", "dev", db)
|
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-07-06 12:46:48 +00:00
|
|
|
db, _ := leveldb.Open(storage.NewMemStorage(), nil)
|
2014-08-14 22:15:26 +00:00
|
|
|
m := NewModel("/tmp", nil, "node", "syncthing", "dev", db)
|
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-07-06 12:46:48 +00:00
|
|
|
db, _ := leveldb.Open(storage.NewMemStorage(), nil)
|
2014-08-14 22:15:26 +00:00
|
|
|
m := NewModel("/tmp", nil, "node", "syncthing", "dev", db)
|
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-07-06 12:46:48 +00:00
|
|
|
db, _ := leveldb.Open(storage.NewMemStorage(), nil)
|
2014-08-14 22:15:26 +00:00
|
|
|
m := NewModel("/tmp", nil, "node", "syncthing", "dev", db)
|
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
|
|
|
}
|
|
|
|
|
2014-07-17 12:48:02 +00:00
|
|
|
func (f FakeConnection) Name() string {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2014-03-02 22:58:14 +00:00
|
|
|
func (f FakeConnection) Option(string) string {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2014-07-17 12:48:02 +00:00
|
|
|
func (FakeConnection) Index(string, []protocol.FileInfo) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (FakeConnection) IndexUpdate(string, []protocol.FileInfo) error {
|
|
|
|
return nil
|
|
|
|
}
|
2014-03-02 22:58:14 +00:00
|
|
|
|
|
|
|
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-07-06 12:46:48 +00:00
|
|
|
db, _ := leveldb.Open(storage.NewMemStorage(), nil)
|
2014-08-14 22:15:26 +00:00
|
|
|
m := NewModel("/tmp", nil, "node", "syncthing", "dev", db)
|
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,
|
2014-07-12 21:06:48 +00:00
|
|
|
Blocks: []protocol.BlockInfo{{0, 100, []byte("some hash bytes")}},
|
2014-03-02 22:58:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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) {
|
2014-07-21 08:48:22 +00:00
|
|
|
isValid := func(protocol.NodeID) bool {
|
|
|
|
return true
|
|
|
|
}
|
2014-03-28 13:36:57 +00:00
|
|
|
m := make(activityMap)
|
2014-07-21 08:48:22 +00:00
|
|
|
if node := m.leastBusyNode([]protocol.NodeID{node1}, isValid); node != node1 {
|
2014-03-28 13:36:57 +00:00
|
|
|
t.Errorf("Incorrect least busy node %q", node)
|
|
|
|
}
|
2014-07-21 08:48:22 +00:00
|
|
|
if node := m.leastBusyNode([]protocol.NodeID{node2}, isValid); node != node2 {
|
2014-03-28 13:36:57 +00:00
|
|
|
t.Errorf("Incorrect least busy node %q", node)
|
|
|
|
}
|
2014-07-21 08:48:22 +00:00
|
|
|
if node := m.leastBusyNode([]protocol.NodeID{node1, node2}, isValid); node != node1 {
|
2014-03-28 13:36:57 +00:00
|
|
|
t.Errorf("Incorrect least busy node %q", node)
|
|
|
|
}
|
2014-07-21 08:48:22 +00:00
|
|
|
if node := m.leastBusyNode([]protocol.NodeID{node1, node2}, isValid); node != node2 {
|
2014-03-28 13:36:57 +00:00
|
|
|
t.Errorf("Incorrect least busy node %q", node)
|
|
|
|
}
|
|
|
|
}
|
2014-08-14 22:15:26 +00:00
|
|
|
|
|
|
|
func TestNodeRename(t *testing.T) {
|
|
|
|
ccm := protocol.ClusterConfigMessage{
|
|
|
|
ClientName: "syncthing",
|
|
|
|
ClientVersion: "v0.9.4",
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg, _ := config.Load(nil, node1)
|
|
|
|
cfg.Nodes = []config.NodeConfiguration{
|
|
|
|
{
|
|
|
|
NodeID: node1,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
db, _ := leveldb.Open(storage.NewMemStorage(), nil)
|
|
|
|
m := NewModel("/tmp", &cfg, "node", "syncthing", "dev", db)
|
|
|
|
if cfg.Nodes[0].Name != "" {
|
|
|
|
t.Errorf("Node already has a name")
|
|
|
|
}
|
|
|
|
|
|
|
|
m.ClusterConfig(node1, ccm)
|
|
|
|
if cfg.Nodes[0].Name != "" {
|
|
|
|
t.Errorf("Node already has a name")
|
|
|
|
}
|
|
|
|
|
|
|
|
ccm.Options = []protocol.Option{
|
|
|
|
{
|
|
|
|
Key: "name",
|
|
|
|
Value: "tester",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
m.ClusterConfig(node1, ccm)
|
|
|
|
if cfg.Nodes[0].Name != "tester" {
|
|
|
|
t.Errorf("Node did not get a name")
|
|
|
|
}
|
|
|
|
|
|
|
|
ccm.Options[0].Value = "tester2"
|
|
|
|
m.ClusterConfig(node1, ccm)
|
|
|
|
if cfg.Nodes[0].Name != "tester" {
|
|
|
|
t.Errorf("Node name got overwritten")
|
|
|
|
}
|
|
|
|
}
|