mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-04 23:55:22 +00:00
264 lines
7.3 KiB
Go
264 lines
7.3 KiB
Go
// Copyright (C) 2018 The Syncthing Authors.
|
|
//
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
package main
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/syncthing/syncthing/lib/protocol"
|
|
)
|
|
|
|
func TestDatabaseGetSet(t *testing.T) {
|
|
db := newInMemoryStore(t.TempDir(), 0, nil)
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
go db.Serve(ctx)
|
|
defer cancel()
|
|
|
|
// Check missing record
|
|
|
|
rec, err := db.get(&protocol.EmptyDeviceID)
|
|
if err != nil {
|
|
t.Error("not found should not be an error")
|
|
}
|
|
if len(rec.Addresses) != 0 {
|
|
t.Error("addresses should be empty")
|
|
}
|
|
|
|
// Set up a clock
|
|
|
|
now := time.Now()
|
|
tc := &testClock{now}
|
|
db.clock = tc
|
|
|
|
// Put a record
|
|
|
|
rec.Addresses = []DatabaseAddress{
|
|
{Address: "tcp://1.2.3.4:5", Expires: tc.Now().Add(time.Minute).UnixNano()},
|
|
}
|
|
if err := db.put(&protocol.EmptyDeviceID, rec); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// Verify it
|
|
|
|
rec, err = db.get(&protocol.EmptyDeviceID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if len(rec.Addresses) != 1 {
|
|
t.Log(rec.Addresses)
|
|
t.Fatal("should have one address")
|
|
}
|
|
if rec.Addresses[0].Address != "tcp://1.2.3.4:5" {
|
|
t.Log(rec.Addresses)
|
|
t.Error("incorrect address")
|
|
}
|
|
|
|
// Wind the clock one half expiry, and merge in a new address
|
|
|
|
tc.wind(30 * time.Second)
|
|
|
|
addrs := []DatabaseAddress{
|
|
{Address: "tcp://6.7.8.9:0", Expires: tc.Now().Add(time.Minute).UnixNano()},
|
|
}
|
|
if err := db.merge(&protocol.EmptyDeviceID, addrs, tc.Now().UnixNano()); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// Verify it
|
|
|
|
rec, err = db.get(&protocol.EmptyDeviceID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if len(rec.Addresses) != 2 {
|
|
t.Log(rec.Addresses)
|
|
t.Fatal("should have two addresses")
|
|
}
|
|
if rec.Addresses[0].Address != "tcp://1.2.3.4:5" {
|
|
t.Log(rec.Addresses)
|
|
t.Error("incorrect address[0]")
|
|
}
|
|
if rec.Addresses[1].Address != "tcp://6.7.8.9:0" {
|
|
t.Log(rec.Addresses)
|
|
t.Error("incorrect address[1]")
|
|
}
|
|
|
|
// Pass the first expiry time
|
|
|
|
tc.wind(45 * time.Second)
|
|
|
|
// Verify it
|
|
|
|
rec, err = db.get(&protocol.EmptyDeviceID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if len(rec.Addresses) != 1 {
|
|
t.Log(rec.Addresses)
|
|
t.Fatal("should have one address")
|
|
}
|
|
if rec.Addresses[0].Address != "tcp://6.7.8.9:0" {
|
|
t.Log(rec.Addresses)
|
|
t.Error("incorrect address")
|
|
}
|
|
|
|
// Set an address
|
|
|
|
addrs = []DatabaseAddress{
|
|
{Address: "tcp://6.7.8.9:0", Expires: tc.Now().Add(time.Minute).UnixNano()},
|
|
}
|
|
if err := db.merge(&protocol.GlobalDeviceID, addrs, tc.Now().UnixNano()); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// Verify it
|
|
|
|
rec, err = db.get(&protocol.GlobalDeviceID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if len(rec.Addresses) != 1 {
|
|
t.Log(rec.Addresses)
|
|
t.Fatal("should have one address")
|
|
}
|
|
}
|
|
|
|
func TestFilter(t *testing.T) {
|
|
// all cases are expired with t=10
|
|
cases := []struct {
|
|
a []DatabaseAddress
|
|
b []DatabaseAddress
|
|
}{
|
|
{
|
|
a: nil,
|
|
b: nil,
|
|
},
|
|
{
|
|
a: []DatabaseAddress{{Address: "a", Expires: 9}, {Address: "b", Expires: 9}, {Address: "c", Expires: 9}},
|
|
b: []DatabaseAddress{},
|
|
},
|
|
{
|
|
a: []DatabaseAddress{{Address: "a", Expires: 10}},
|
|
b: []DatabaseAddress{{Address: "a", Expires: 10}},
|
|
},
|
|
{
|
|
a: []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 10}, {Address: "c", Expires: 10}},
|
|
b: []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 10}, {Address: "c", Expires: 10}},
|
|
},
|
|
{
|
|
a: []DatabaseAddress{{Address: "a", Expires: 5}, {Address: "b", Expires: 15}, {Address: "c", Expires: 5}, {Address: "d", Expires: 15}, {Address: "e", Expires: 5}},
|
|
b: []DatabaseAddress{{Address: "b", Expires: 15}, {Address: "d", Expires: 15}},
|
|
},
|
|
}
|
|
|
|
for _, tc := range cases {
|
|
res := expire(tc.a, time.Unix(0, 10))
|
|
if fmt.Sprint(res) != fmt.Sprint(tc.b) {
|
|
t.Errorf("Incorrect result %v, expected %v", res, tc.b)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestMerge(t *testing.T) {
|
|
cases := []struct {
|
|
a, b, res []DatabaseAddress
|
|
}{
|
|
{nil, nil, nil},
|
|
{
|
|
nil,
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}},
|
|
},
|
|
{
|
|
nil,
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 10}, {Address: "c", Expires: 10}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 10}, {Address: "c", Expires: 10}},
|
|
},
|
|
{
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 15}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 15}},
|
|
},
|
|
{
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}},
|
|
[]DatabaseAddress{{Address: "b", Expires: 15}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}},
|
|
},
|
|
{
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 15}, {Address: "b", Expires: 15}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 15}, {Address: "b", Expires: 15}},
|
|
},
|
|
{
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}},
|
|
[]DatabaseAddress{{Address: "b", Expires: 15}, {Address: "c", Expires: 20}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}, {Address: "c", Expires: 20}},
|
|
},
|
|
{
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}},
|
|
[]DatabaseAddress{{Address: "b", Expires: 5}, {Address: "c", Expires: 20}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}, {Address: "c", Expires: 20}},
|
|
},
|
|
{
|
|
[]DatabaseAddress{{Address: "y", Expires: 10}, {Address: "z", Expires: 10}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 5}, {Address: "b", Expires: 15}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 5}, {Address: "b", Expires: 15}, {Address: "y", Expires: 10}, {Address: "z", Expires: 10}},
|
|
},
|
|
{
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}, {Address: "d", Expires: 10}},
|
|
[]DatabaseAddress{{Address: "b", Expires: 5}, {Address: "c", Expires: 20}},
|
|
[]DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}, {Address: "c", Expires: 20}, {Address: "d", Expires: 10}},
|
|
},
|
|
}
|
|
|
|
for _, tc := range cases {
|
|
rec := merge(DatabaseRecord{Addresses: tc.a}, DatabaseRecord{Addresses: tc.b})
|
|
if fmt.Sprint(rec.Addresses) != fmt.Sprint(tc.res) {
|
|
t.Errorf("Incorrect result %v, expected %v", rec.Addresses, tc.res)
|
|
}
|
|
rec = merge(DatabaseRecord{Addresses: tc.b}, DatabaseRecord{Addresses: tc.a})
|
|
if fmt.Sprint(rec.Addresses) != fmt.Sprint(tc.res) {
|
|
t.Errorf("Incorrect result %v, expected %v", rec.Addresses, tc.res)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkMergeEqual(b *testing.B) {
|
|
for i := 0; i < b.N; i++ {
|
|
ar := []DatabaseAddress{{Address: "a", Expires: 10}, {Address: "b", Expires: 15}}
|
|
br := []DatabaseAddress{{Address: "a", Expires: 15}, {Address: "b", Expires: 10}}
|
|
res := merge(DatabaseRecord{Addresses: ar}, DatabaseRecord{Addresses: br})
|
|
if len(res.Addresses) != 2 {
|
|
b.Fatal("wrong length")
|
|
}
|
|
if res.Addresses[0].Address != "a" || res.Addresses[1].Address != "b" {
|
|
b.Fatal("wrong address")
|
|
}
|
|
if res.Addresses[0].Expires != 15 || res.Addresses[1].Expires != 15 {
|
|
b.Fatal("wrong expiry")
|
|
}
|
|
}
|
|
b.ReportAllocs() // should be zero per operation
|
|
}
|
|
|
|
type testClock struct {
|
|
now time.Time
|
|
}
|
|
|
|
func (t *testClock) wind(d time.Duration) {
|
|
t.now = t.now.Add(d)
|
|
}
|
|
|
|
func (t *testClock) Now() time.Time {
|
|
t.now = t.now.Add(time.Nanosecond)
|
|
return t.now
|
|
}
|