mirror of
https://github.com/syncthing/syncthing.git
synced 2026-04-27 18:00:07 -04:00
lib/db: Add test for GC run
This commit is contained in:
@@ -8,6 +8,7 @@ package db
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"context"
|
||||
"testing"
|
||||
|
||||
"github.com/syncthing/syncthing/lib/db/backend"
|
||||
@@ -633,3 +634,109 @@ func TestDropDuplicates(t *testing.T) {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestGCIndirect(t *testing.T) {
|
||||
// Verify that the gcIndirect run actually removes block lists.
|
||||
|
||||
db := NewLowlevel(backend.OpenMemory())
|
||||
defer db.Close()
|
||||
meta := newMetadataTracker()
|
||||
|
||||
// Add three files with different block lists
|
||||
|
||||
files := []protocol.FileInfo{
|
||||
{Name: "a", Blocks: genBlocks(100)},
|
||||
{Name: "b", Blocks: genBlocks(200)},
|
||||
{Name: "c", Blocks: genBlocks(300)},
|
||||
}
|
||||
|
||||
db.updateLocalFiles([]byte("folder"), files, meta)
|
||||
|
||||
// Run a GC pass
|
||||
|
||||
db.gcIndirect(context.Background())
|
||||
|
||||
// Verify that we have three different block lists
|
||||
|
||||
n, err := numBlockLists(db)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if n != len(files) {
|
||||
t.Fatal("expected each file to have a block list")
|
||||
}
|
||||
|
||||
// Change the block lists for each file
|
||||
|
||||
for i := range files {
|
||||
files[i].Version = files[i].Version.Update(42)
|
||||
files[i].Blocks = genBlocks(len(files[i].Blocks) + 1)
|
||||
}
|
||||
|
||||
db.updateLocalFiles([]byte("folder"), files, meta)
|
||||
|
||||
// Verify that we now have *six* different block lists
|
||||
|
||||
n, err = numBlockLists(db)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if n != 2*len(files) {
|
||||
t.Fatal("expected both old and new block lists to exist")
|
||||
}
|
||||
|
||||
// Run a GC pass
|
||||
|
||||
db.gcIndirect(context.Background())
|
||||
|
||||
// Verify that we now have just the three we need, again
|
||||
|
||||
n, err = numBlockLists(db)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if n != len(files) {
|
||||
t.Fatal("expected GC to collect all but the needed ones")
|
||||
}
|
||||
|
||||
// Double check the correctness by loading the block lists and comparing with what we stored
|
||||
|
||||
tr, err := db.newReadOnlyTransaction()
|
||||
if err != nil {
|
||||
t.Fatal()
|
||||
}
|
||||
defer tr.Release()
|
||||
for _, f := range files {
|
||||
fi, ok, err := tr.getFile([]byte("folder"), protocol.LocalDeviceID[:], []byte(f.Name))
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if !ok {
|
||||
t.Fatal("mysteriously missing")
|
||||
}
|
||||
if len(fi.Blocks) != len(f.Blocks) {
|
||||
t.Fatal("block list mismatch")
|
||||
}
|
||||
for i := range fi.Blocks {
|
||||
if !bytes.Equal(fi.Blocks[i].Hash, f.Blocks[i].Hash) {
|
||||
t.Fatal("hash mismatch")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func numBlockLists(db *Lowlevel) (int, error) {
|
||||
it, err := db.Backend.NewPrefixIterator([]byte{KeyTypeBlockList})
|
||||
if err != nil {
|
||||
return 0, err
|
||||
}
|
||||
defer it.Release()
|
||||
n := 0
|
||||
for it.Next() {
|
||||
n++
|
||||
}
|
||||
if err := it.Error(); err != nil {
|
||||
return 0, err
|
||||
}
|
||||
return n, nil
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user