2014-11-16 20:13:20 +00:00
|
|
|
// Copyright (C) 2014 The Syncthing Authors.
|
2014-10-06 20:57:33 +00:00
|
|
|
//
|
2015-03-07 20:36:35 +00:00
|
|
|
// 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,
|
2017-02-09 06:52:18 +00:00
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
2014-10-06 20:57:33 +00:00
|
|
|
|
2015-01-12 13:50:30 +00:00
|
|
|
package db
|
2014-10-06 20:57:33 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
|
2015-09-22 17:38:46 +00:00
|
|
|
"github.com/syncthing/syncthing/lib/protocol"
|
2016-01-03 18:08:19 +00:00
|
|
|
"github.com/syndtr/goleveldb/leveldb/util"
|
2014-10-06 20:57:33 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func genBlocks(n int) []protocol.BlockInfo {
|
|
|
|
b := make([]protocol.BlockInfo, n)
|
|
|
|
for i := range b {
|
|
|
|
h := make([]byte, 32)
|
|
|
|
for j := range h {
|
|
|
|
h[j] = byte(i + j)
|
|
|
|
}
|
2015-01-18 01:12:06 +00:00
|
|
|
b[i].Size = int32(i)
|
2014-10-06 20:57:33 +00:00
|
|
|
b[i].Hash = h
|
|
|
|
}
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
var f1, f2, f3 protocol.FileInfo
|
2015-09-04 10:01:00 +00:00
|
|
|
var folders = []string{"folder1", "folder2"}
|
2014-10-06 20:57:33 +00:00
|
|
|
|
|
|
|
func init() {
|
|
|
|
blocks := genBlocks(30)
|
|
|
|
|
|
|
|
f1 = protocol.FileInfo{
|
|
|
|
Name: "f1",
|
|
|
|
Blocks: blocks[:10],
|
|
|
|
}
|
|
|
|
|
|
|
|
f2 = protocol.FileInfo{
|
|
|
|
Name: "f2",
|
|
|
|
Blocks: blocks[10:20],
|
|
|
|
}
|
|
|
|
|
|
|
|
f3 = protocol.FileInfo{
|
|
|
|
Name: "f3",
|
|
|
|
Blocks: blocks[20:],
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-31 11:31:25 +00:00
|
|
|
func setup() (*Instance, *BlockFinder) {
|
2014-10-06 20:57:33 +00:00
|
|
|
// Setup
|
|
|
|
|
2015-10-31 11:31:25 +00:00
|
|
|
db := OpenMemory()
|
2015-09-04 10:01:00 +00:00
|
|
|
return db, NewBlockFinder(db)
|
2014-10-06 20:57:33 +00:00
|
|
|
}
|
|
|
|
|
2015-10-31 11:31:25 +00:00
|
|
|
func dbEmpty(db *Instance) bool {
|
2016-01-03 18:08:19 +00:00
|
|
|
iter := db.NewIterator(util.BytesPrefix([]byte{KeyTypeBlock}), nil)
|
2014-10-06 20:57:33 +00:00
|
|
|
defer iter.Release()
|
2016-12-18 18:57:41 +00:00
|
|
|
return !iter.Next()
|
2014-10-06 20:57:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestBlockMapAddUpdateWipe(t *testing.T) {
|
|
|
|
db, f := setup()
|
|
|
|
|
|
|
|
if !dbEmpty(db) {
|
|
|
|
t.Fatal("db not empty")
|
|
|
|
}
|
|
|
|
|
2016-01-03 18:08:19 +00:00
|
|
|
m := NewBlockMap(db, db.folderIdx.ID([]byte("folder1")))
|
2014-10-06 20:57:33 +00:00
|
|
|
|
2016-07-04 10:40:29 +00:00
|
|
|
f3.Type = protocol.FileInfoTypeDirectory
|
2014-10-06 20:57:33 +00:00
|
|
|
|
|
|
|
err := m.Add([]protocol.FileInfo{f1, f2, f3})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2015-09-04 10:01:00 +00:00
|
|
|
f.Iterate(folders, f1.Blocks[0].Hash, func(folder, file string, index int32) bool {
|
2014-10-06 20:57:33 +00:00
|
|
|
if folder != "folder1" || file != "f1" || index != 0 {
|
|
|
|
t.Fatal("Mismatch")
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
2015-09-04 10:01:00 +00:00
|
|
|
f.Iterate(folders, f2.Blocks[0].Hash, func(folder, file string, index int32) bool {
|
2014-10-06 20:57:33 +00:00
|
|
|
if folder != "folder1" || file != "f2" || index != 0 {
|
|
|
|
t.Fatal("Mismatch")
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
2015-09-04 10:01:00 +00:00
|
|
|
f.Iterate(folders, f3.Blocks[0].Hash, func(folder, file string, index int32) bool {
|
2014-10-06 20:57:33 +00:00
|
|
|
t.Fatal("Unexpected block")
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
2016-07-04 10:40:29 +00:00
|
|
|
f1.Deleted = true
|
2018-06-24 07:50:18 +00:00
|
|
|
f2.LocalFlags = protocol.FlagLocalMustRescan // one of the invalid markers
|
2014-10-06 20:57:33 +00:00
|
|
|
|
|
|
|
// Should remove
|
|
|
|
err = m.Update([]protocol.FileInfo{f1, f2, f3})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2015-09-04 10:01:00 +00:00
|
|
|
f.Iterate(folders, f1.Blocks[0].Hash, func(folder, file string, index int32) bool {
|
2014-10-06 20:57:33 +00:00
|
|
|
t.Fatal("Unexpected block")
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
|
2015-09-04 10:01:00 +00:00
|
|
|
f.Iterate(folders, f2.Blocks[0].Hash, func(folder, file string, index int32) bool {
|
2014-10-06 20:57:33 +00:00
|
|
|
t.Fatal("Unexpected block")
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
|
2015-09-04 10:01:00 +00:00
|
|
|
f.Iterate(folders, f3.Blocks[0].Hash, func(folder, file string, index int32) bool {
|
2014-10-06 20:57:33 +00:00
|
|
|
if folder != "folder1" || file != "f3" || index != 0 {
|
|
|
|
t.Fatal("Mismatch")
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
|
|
|
err = m.Drop()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !dbEmpty(db) {
|
|
|
|
t.Fatal("db not empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should not add
|
|
|
|
err = m.Add([]protocol.FileInfo{f1, f2})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !dbEmpty(db) {
|
|
|
|
t.Fatal("db not empty")
|
|
|
|
}
|
|
|
|
|
2016-07-04 10:40:29 +00:00
|
|
|
f1.Deleted = false
|
2018-06-24 07:50:18 +00:00
|
|
|
f1.LocalFlags = 0
|
2016-07-04 10:40:29 +00:00
|
|
|
f2.Deleted = false
|
2018-06-24 07:50:18 +00:00
|
|
|
f2.LocalFlags = 0
|
2016-07-04 10:40:29 +00:00
|
|
|
f3.Deleted = false
|
2018-06-24 07:50:18 +00:00
|
|
|
f3.LocalFlags = 0
|
2014-10-06 20:57:33 +00:00
|
|
|
}
|
|
|
|
|
2014-10-24 21:23:19 +00:00
|
|
|
func TestBlockFinderLookup(t *testing.T) {
|
2014-10-06 20:57:33 +00:00
|
|
|
db, f := setup()
|
|
|
|
|
2016-01-03 18:08:19 +00:00
|
|
|
m1 := NewBlockMap(db, db.folderIdx.ID([]byte("folder1")))
|
|
|
|
m2 := NewBlockMap(db, db.folderIdx.ID([]byte("folder2")))
|
2014-10-06 20:57:33 +00:00
|
|
|
|
|
|
|
err := m1.Add([]protocol.FileInfo{f1})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
err = m2.Add([]protocol.FileInfo{f1})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
counter := 0
|
2015-09-04 10:01:00 +00:00
|
|
|
f.Iterate(folders, f1.Blocks[0].Hash, func(folder, file string, index int32) bool {
|
2014-10-06 20:57:33 +00:00
|
|
|
counter++
|
|
|
|
switch counter {
|
|
|
|
case 1:
|
|
|
|
if folder != "folder1" || file != "f1" || index != 0 {
|
|
|
|
t.Fatal("Mismatch")
|
|
|
|
}
|
|
|
|
case 2:
|
|
|
|
if folder != "folder2" || file != "f1" || index != 0 {
|
|
|
|
t.Fatal("Mismatch")
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
t.Fatal("Unexpected block")
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
})
|
2015-09-04 10:01:00 +00:00
|
|
|
|
2014-10-06 20:57:33 +00:00
|
|
|
if counter != 2 {
|
|
|
|
t.Fatal("Incorrect count", counter)
|
|
|
|
}
|
|
|
|
|
2016-07-04 10:40:29 +00:00
|
|
|
f1.Deleted = true
|
2014-10-06 20:57:33 +00:00
|
|
|
|
|
|
|
err = m1.Update([]protocol.FileInfo{f1})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
counter = 0
|
2015-09-04 10:01:00 +00:00
|
|
|
f.Iterate(folders, f1.Blocks[0].Hash, func(folder, file string, index int32) bool {
|
2014-10-06 20:57:33 +00:00
|
|
|
counter++
|
|
|
|
switch counter {
|
|
|
|
case 1:
|
|
|
|
if folder != "folder2" || file != "f1" || index != 0 {
|
|
|
|
t.Fatal("Mismatch")
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
t.Fatal("Unexpected block")
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
})
|
2015-09-04 10:01:00 +00:00
|
|
|
|
2014-10-06 20:57:33 +00:00
|
|
|
if counter != 1 {
|
|
|
|
t.Fatal("Incorrect count")
|
|
|
|
}
|
2014-10-24 21:23:19 +00:00
|
|
|
|
2016-07-04 10:40:29 +00:00
|
|
|
f1.Deleted = false
|
2014-10-06 20:57:33 +00:00
|
|
|
}
|