mirror of
https://github.com/restic/restic.git
synced 2026-06-21 16:14:18 +00:00
8e11f5747d
decouple restic and ui/progress packages
758 lines
22 KiB
Go
758 lines
22 KiB
Go
package index_test
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"math/rand"
|
|
"runtime"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
"github.com/restic/restic/internal/checker"
|
|
"github.com/restic/restic/internal/data"
|
|
"github.com/restic/restic/internal/repository"
|
|
"github.com/restic/restic/internal/repository/crypto"
|
|
"github.com/restic/restic/internal/repository/index"
|
|
"github.com/restic/restic/internal/repository/pack"
|
|
"github.com/restic/restic/internal/restic"
|
|
rtest "github.com/restic/restic/internal/test"
|
|
"github.com/restic/restic/internal/ui/progress"
|
|
)
|
|
|
|
func TestMasterIndex(t *testing.T) {
|
|
bhInIdx1 := restic.NewRandomBlobHandle()
|
|
bhInIdx2 := restic.NewRandomBlobHandle()
|
|
bhInIdx12 := restic.BlobHandle{ID: restic.NewRandomID(), Type: restic.TreeBlob}
|
|
|
|
blob1 := &pack.PackedBlob{
|
|
Pack: restic.NewRandomID(),
|
|
Blob: pack.Blob{
|
|
BlobHandle: bhInIdx1,
|
|
Length: uint(crypto.CiphertextLength(10)),
|
|
Offset: 0,
|
|
},
|
|
}
|
|
|
|
blob2 := &pack.PackedBlob{
|
|
Pack: restic.NewRandomID(),
|
|
Blob: pack.Blob{
|
|
BlobHandle: bhInIdx2,
|
|
Length: uint(crypto.CiphertextLength(100)),
|
|
Offset: 10,
|
|
UncompressedLength: 200,
|
|
},
|
|
}
|
|
|
|
blob12a := &pack.PackedBlob{
|
|
Pack: restic.NewRandomID(),
|
|
Blob: pack.Blob{
|
|
BlobHandle: bhInIdx12,
|
|
Length: uint(crypto.CiphertextLength(123)),
|
|
Offset: 110,
|
|
UncompressedLength: 80,
|
|
},
|
|
}
|
|
|
|
blob12b := &pack.PackedBlob{
|
|
Pack: restic.NewRandomID(),
|
|
Blob: pack.Blob{
|
|
BlobHandle: bhInIdx12,
|
|
Length: uint(crypto.CiphertextLength(123)),
|
|
Offset: 50,
|
|
UncompressedLength: 80,
|
|
},
|
|
}
|
|
|
|
idx1 := index.NewIndex()
|
|
idx1.StorePack(blob1.PackID(), pack.Blobs{blob1.Blob})
|
|
idx1.StorePack(blob12a.PackID(), pack.Blobs{blob12a.Blob})
|
|
|
|
idx2 := index.NewIndex()
|
|
idx2.StorePack(blob2.PackID(), pack.Blobs{blob2.Blob})
|
|
idx2.StorePack(blob12b.PackID(), pack.Blobs{blob12b.Blob})
|
|
|
|
mIdx := index.NewMasterIndex()
|
|
mIdx.Insert(idx1)
|
|
mIdx.Insert(idx2)
|
|
|
|
// test idInIdx1
|
|
blobs := mIdx.Lookup(bhInIdx1)
|
|
rtest.Equals(t, []*pack.PackedBlob{blob1}, blobs)
|
|
|
|
size, found := mIdx.LookupSize(bhInIdx1)
|
|
rtest.Equals(t, true, found)
|
|
rtest.Equals(t, uint(10), size)
|
|
|
|
// test idInIdx2
|
|
blobs = mIdx.Lookup(bhInIdx2)
|
|
rtest.Equals(t, []*pack.PackedBlob{blob2}, blobs)
|
|
|
|
size, found = mIdx.LookupSize(bhInIdx2)
|
|
rtest.Equals(t, true, found)
|
|
rtest.Equals(t, uint(200), size)
|
|
|
|
// test idInIdx12
|
|
blobs = mIdx.Lookup(bhInIdx12)
|
|
rtest.Equals(t, 2, len(blobs))
|
|
|
|
containsPackedBlob := func(list []*pack.PackedBlob, want *pack.PackedBlob) bool {
|
|
for _, b := range list {
|
|
if b.PackID().Equal(want.PackID()) && b.Blob == want.Blob {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// test Lookup result for blob12a
|
|
rtest.Assert(t, containsPackedBlob(blobs, blob12a), "blob12a not found in result")
|
|
|
|
// test Lookup result for blob12b
|
|
rtest.Assert(t, containsPackedBlob(blobs, blob12b), "blob12b not found in result")
|
|
|
|
size, found = mIdx.LookupSize(bhInIdx12)
|
|
rtest.Equals(t, true, found)
|
|
rtest.Equals(t, uint(80), size)
|
|
|
|
// test not in index
|
|
blobs = mIdx.Lookup(restic.NewRandomBlobHandle())
|
|
rtest.Assert(t, blobs == nil, "Expected no blobs when fetching with a random id")
|
|
_, found = mIdx.LookupSize(restic.NewRandomBlobHandle())
|
|
rtest.Assert(t, !found, "Expected no blobs when fetching with a random id")
|
|
}
|
|
|
|
func TestMasterIndexAddPending(t *testing.T) {
|
|
mIdx := index.NewMasterIndex()
|
|
|
|
// Test AddPending: successfully add a new blob
|
|
bhPending := restic.NewRandomBlobHandle()
|
|
added := mIdx.AddPending(bhPending, 100)
|
|
rtest.Equals(t, true, added)
|
|
|
|
// Test AddPending: try to add the same blob again (should return false)
|
|
added = mIdx.AddPending(bhPending, 200)
|
|
rtest.Equals(t, false, added)
|
|
|
|
// Test AddPending: try to add a blob that's already in an index (should return false)
|
|
bhInIndex := restic.NewRandomBlobHandle()
|
|
idx := index.NewIndex()
|
|
idx.StorePack(restic.NewRandomID(), pack.Blobs{{
|
|
BlobHandle: bhInIndex,
|
|
Length: uint(crypto.CiphertextLength(50)),
|
|
Offset: 0,
|
|
UncompressedLength: 50,
|
|
}})
|
|
mIdx.Insert(idx)
|
|
|
|
added = mIdx.AddPending(bhInIndex, 100)
|
|
rtest.Equals(t, false, added)
|
|
|
|
// Test LookupSize: returns pending blob size when blob is pending
|
|
size, found := mIdx.LookupSize(bhPending)
|
|
rtest.Equals(t, true, found)
|
|
rtest.Equals(t, uint(100), size)
|
|
}
|
|
|
|
// noopSaver is a no-op implementation of SaverUnpacked for testing.
|
|
type noopSaver struct{}
|
|
|
|
func (n *noopSaver) Connections() uint {
|
|
return 2
|
|
}
|
|
|
|
func (n *noopSaver) SaveUnpacked(_ context.Context, _ restic.FileType, buf []byte) (restic.ID, error) {
|
|
return restic.Hash(buf), nil
|
|
}
|
|
|
|
func TestMasterIndexStorePackRemovesPending(t *testing.T) {
|
|
mIdx := index.NewMasterIndex()
|
|
|
|
// Add a blob as pending
|
|
bhPending := restic.NewRandomBlobHandle()
|
|
added := mIdx.AddPending(bhPending, 75)
|
|
rtest.Equals(t, true, added)
|
|
|
|
// Store the blob in a pack
|
|
packID := restic.NewRandomID()
|
|
blob := pack.Blob{
|
|
BlobHandle: bhPending,
|
|
Length: uint(crypto.CiphertextLength(75)),
|
|
Offset: 0,
|
|
UncompressedLength: 75,
|
|
}
|
|
saver := &noopSaver{}
|
|
err := mIdx.StorePack(context.Background(), packID, pack.Blobs{blob}, saver)
|
|
rtest.OK(t, err)
|
|
|
|
// Verify it is still found
|
|
size, found := mIdx.LookupSize(bhPending)
|
|
rtest.Equals(t, true, found)
|
|
rtest.Equals(t, uint(75), size)
|
|
|
|
// Verify the blob can be found via Lookup from the index
|
|
blobs := mIdx.Lookup(bhPending)
|
|
rtest.Assert(t, len(blobs) > 0, "blob should be found in index after StorePack")
|
|
rtest.Equals(t, packID, blobs[0].PackID())
|
|
rtest.Equals(t, bhPending, blobs[0].Handle())
|
|
|
|
// Test that adding the same blob as pending again fails (it's now in index)
|
|
added = mIdx.AddPending(bhPending, 100)
|
|
rtest.Equals(t, false, added)
|
|
}
|
|
|
|
func TestMasterMergeFinalIndexes(t *testing.T) {
|
|
bhInIdx1 := restic.NewRandomBlobHandle()
|
|
bhInIdx2 := restic.NewRandomBlobHandle()
|
|
|
|
blob1 := &pack.PackedBlob{
|
|
Pack: restic.NewRandomID(),
|
|
Blob: pack.Blob{
|
|
BlobHandle: bhInIdx1,
|
|
Length: 10,
|
|
Offset: 0,
|
|
},
|
|
}
|
|
|
|
blob2 := &pack.PackedBlob{
|
|
Pack: restic.NewRandomID(),
|
|
Blob: pack.Blob{
|
|
BlobHandle: bhInIdx2,
|
|
Length: 100,
|
|
Offset: 10,
|
|
UncompressedLength: 200,
|
|
},
|
|
}
|
|
|
|
idx1 := index.NewIndex()
|
|
idx1.StorePack(blob1.PackID(), pack.Blobs{blob1.Blob})
|
|
|
|
idx2 := index.NewIndex()
|
|
idx2.StorePack(blob2.PackID(), pack.Blobs{blob2.Blob})
|
|
|
|
mIdx := index.NewMasterIndex()
|
|
mIdx.Insert(idx1)
|
|
mIdx.Insert(idx2)
|
|
|
|
rtest.Equals(t, restic.NewIDSet(), mIdx.IDs())
|
|
|
|
finalIndexes, idxCount, ids := index.TestMergeIndex(t, mIdx)
|
|
rtest.Equals(t, []*index.Index{idx1, idx2}, finalIndexes)
|
|
rtest.Equals(t, 1, idxCount)
|
|
rtest.Equals(t, ids, mIdx.IDs())
|
|
|
|
blobCount := 0
|
|
for range mIdx.Values() {
|
|
blobCount++
|
|
}
|
|
rtest.Equals(t, 2, blobCount)
|
|
|
|
blobs := mIdx.Lookup(bhInIdx1)
|
|
rtest.Equals(t, []*pack.PackedBlob{blob1}, blobs)
|
|
|
|
blobs = mIdx.Lookup(bhInIdx2)
|
|
rtest.Equals(t, []*pack.PackedBlob{blob2}, blobs)
|
|
|
|
blobs = mIdx.Lookup(restic.NewRandomBlobHandle())
|
|
rtest.Assert(t, blobs == nil, "Expected no blobs when fetching with a random id")
|
|
|
|
// merge another index containing identical blobs
|
|
idx3 := index.NewIndex()
|
|
idx3.StorePack(blob1.PackID(), pack.Blobs{blob1.Blob})
|
|
idx3.StorePack(blob2.PackID(), pack.Blobs{blob2.Blob})
|
|
|
|
mIdx.Insert(idx3)
|
|
finalIndexes, idxCount, newIDs := index.TestMergeIndex(t, mIdx)
|
|
rtest.Equals(t, []*index.Index{idx3}, finalIndexes)
|
|
rtest.Equals(t, 1, idxCount)
|
|
ids.Merge(newIDs)
|
|
rtest.Equals(t, ids, mIdx.IDs())
|
|
|
|
// Index should have same entries as before!
|
|
blobs = mIdx.Lookup(bhInIdx1)
|
|
rtest.Equals(t, []*pack.PackedBlob{blob1}, blobs)
|
|
|
|
blobs = mIdx.Lookup(bhInIdx2)
|
|
rtest.Equals(t, []*pack.PackedBlob{blob2}, blobs)
|
|
|
|
blobCount = 0
|
|
for range mIdx.Values() {
|
|
blobCount++
|
|
}
|
|
rtest.Equals(t, 2, blobCount)
|
|
}
|
|
|
|
func createRandomMasterIndex(t testing.TB, rng *rand.Rand, num, size int) (*index.MasterIndex, restic.BlobHandle) {
|
|
mIdx := index.NewMasterIndex()
|
|
for i := 0; i < num-1; i++ {
|
|
idx, _ := createRandomIndex(rng, size)
|
|
mIdx.Insert(idx)
|
|
}
|
|
idx1, lookupBh := createRandomIndex(rng, size)
|
|
mIdx.Insert(idx1)
|
|
|
|
index.TestMergeIndex(t, mIdx)
|
|
|
|
return mIdx, lookupBh
|
|
}
|
|
|
|
func BenchmarkMasterIndexAlloc(b *testing.B) {
|
|
rng := rand.New(rand.NewSource(0))
|
|
b.ReportAllocs()
|
|
|
|
for b.Loop() {
|
|
createRandomMasterIndex(b, rng, 10000, 5)
|
|
}
|
|
}
|
|
|
|
func BenchmarkMasterIndexMerge(b *testing.B) {
|
|
rng := rand.New(rand.NewSource(0))
|
|
b.ReportAllocs()
|
|
|
|
for b.Loop() {
|
|
createRandomMasterIndex(b, rng, 1000, 1000)
|
|
}
|
|
}
|
|
|
|
func BenchmarkMasterIndexLookupSingleIndex(b *testing.B) {
|
|
mIdx, lookupBh := createRandomMasterIndex(b, rand.New(rand.NewSource(0)), 1, 200000)
|
|
|
|
for b.Loop() {
|
|
mIdx.Lookup(lookupBh)
|
|
}
|
|
}
|
|
|
|
func BenchmarkMasterIndexLookupMultipleIndex(b *testing.B) {
|
|
mIdx, lookupBh := createRandomMasterIndex(b, rand.New(rand.NewSource(0)), 100, 10000)
|
|
|
|
for b.Loop() {
|
|
mIdx.Lookup(lookupBh)
|
|
}
|
|
}
|
|
|
|
func BenchmarkMasterIndexLookupSingleIndexUnknown(b *testing.B) {
|
|
lookupBh := restic.NewRandomBlobHandle()
|
|
mIdx, _ := createRandomMasterIndex(b, rand.New(rand.NewSource(0)), 1, 200000)
|
|
|
|
for b.Loop() {
|
|
mIdx.Lookup(lookupBh)
|
|
}
|
|
}
|
|
|
|
func BenchmarkMasterIndexLookupMultipleIndexUnknown(b *testing.B) {
|
|
lookupBh := restic.NewRandomBlobHandle()
|
|
mIdx, _ := createRandomMasterIndex(b, rand.New(rand.NewSource(0)), 100, 10000)
|
|
|
|
for b.Loop() {
|
|
mIdx.Lookup(lookupBh)
|
|
}
|
|
}
|
|
|
|
func BenchmarkMasterIndexLookupParallel(b *testing.B) {
|
|
for _, numindices := range []int{25, 50, 100} {
|
|
var lookupBh restic.BlobHandle
|
|
|
|
b.StopTimer()
|
|
rng := rand.New(rand.NewSource(0))
|
|
mIdx, lookupBh := createRandomMasterIndex(b, rng, numindices, 10000)
|
|
b.StartTimer()
|
|
|
|
name := fmt.Sprintf("known,indices=%d", numindices)
|
|
b.Run(name, func(b *testing.B) {
|
|
b.RunParallel(func(pb *testing.PB) {
|
|
for pb.Next() {
|
|
mIdx.Lookup(lookupBh)
|
|
}
|
|
})
|
|
})
|
|
|
|
lookupBh = restic.NewRandomBlobHandle()
|
|
name = fmt.Sprintf("unknown,indices=%d", numindices)
|
|
b.Run(name, func(b *testing.B) {
|
|
b.RunParallel(func(pb *testing.PB) {
|
|
for pb.Next() {
|
|
mIdx.Lookup(lookupBh)
|
|
}
|
|
})
|
|
})
|
|
}
|
|
}
|
|
|
|
func BenchmarkMasterIndexLookupBlobSize(b *testing.B) {
|
|
rng := rand.New(rand.NewSource(0))
|
|
mIdx, lookupBh := createRandomMasterIndex(b, rand.New(rng), 5, 200000)
|
|
|
|
for b.Loop() {
|
|
mIdx.LookupSize(lookupBh)
|
|
}
|
|
}
|
|
|
|
func BenchmarkMasterIndexEach(b *testing.B) {
|
|
rng := rand.New(rand.NewSource(0))
|
|
mIdx, _ := createRandomMasterIndex(b, rand.New(rng), 5, 200000)
|
|
|
|
for b.Loop() {
|
|
entries := 0
|
|
for range mIdx.Values() {
|
|
entries++
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkMasterIndexGC(b *testing.B) {
|
|
mIdx, _ := createRandomMasterIndex(b, rand.New(rand.NewSource(0)), 100, 10000)
|
|
|
|
for b.Loop() {
|
|
runtime.GC()
|
|
}
|
|
runtime.KeepAlive(mIdx)
|
|
}
|
|
|
|
var (
|
|
snapshotTime = time.Unix(1470492820, 207401672)
|
|
depth = 3
|
|
)
|
|
|
|
func createFilledRepo(t testing.TB, snapshots int, version uint) (*repository.Repository, restic.Unpacked[restic.FileType]) {
|
|
repo, unpacked, _ := repository.TestRepositoryWithVersion(t, version)
|
|
|
|
for i := 0; i < snapshots; i++ {
|
|
data.TestCreateSnapshot(t, repo, snapshotTime.Add(time.Duration(i)*time.Second), depth)
|
|
}
|
|
return repo, unpacked
|
|
}
|
|
|
|
func TestIndexSave(t *testing.T) {
|
|
repository.TestAllVersions(t, testIndexSave)
|
|
}
|
|
|
|
func testIndexSave(t *testing.T, version uint) {
|
|
for _, test := range []struct {
|
|
name string
|
|
saver func(idx *index.MasterIndex, repo restic.Unpacked[restic.FileType]) error
|
|
}{
|
|
{"rewrite no-op", func(idx *index.MasterIndex, repo restic.Unpacked[restic.FileType]) error {
|
|
return idx.Rewrite(context.TODO(), repo, nil, nil, nil, index.MasterIndexRewriteOpts{})
|
|
}},
|
|
{"rewrite skip-all", func(idx *index.MasterIndex, repo restic.Unpacked[restic.FileType]) error {
|
|
return idx.Rewrite(context.TODO(), repo, nil, restic.NewIDSet(), nil, index.MasterIndexRewriteOpts{})
|
|
}},
|
|
{"SaveFallback", func(idx *index.MasterIndex, repo restic.Unpacked[restic.FileType]) error {
|
|
err := restic.ParallelRemove(context.TODO(), repo, idx.IDs(), restic.IndexFile, nil, restic.NoopCounter)
|
|
if err != nil {
|
|
return nil
|
|
}
|
|
return idx.SaveFallback(context.TODO(), repo, restic.NewIDSet(), restic.NoopCounter)
|
|
}},
|
|
} {
|
|
t.Run(test.name, func(t *testing.T) {
|
|
repo, unpacked := createFilledRepo(t, 3, version)
|
|
|
|
idx := index.NewMasterIndex()
|
|
rtest.OK(t, idx.Load(context.TODO(), repo, restic.NoopCounter, nil))
|
|
blobs := make(map[pack.PackedBlob]struct{})
|
|
for pb := range idx.Values() {
|
|
blobs[*pb] = struct{}{}
|
|
}
|
|
|
|
rtest.OK(t, test.saver(idx, unpacked))
|
|
idx = index.NewMasterIndex()
|
|
rtest.OK(t, idx.Load(context.TODO(), repo, restic.NoopCounter, nil))
|
|
|
|
for pb := range idx.Values() {
|
|
if _, ok := blobs[*pb]; ok {
|
|
delete(blobs, *pb)
|
|
} else {
|
|
t.Fatalf("unexpected blobs %v", pb)
|
|
}
|
|
}
|
|
rtest.Equals(t, 0, len(blobs), "saved index is missing blobs")
|
|
|
|
checker.TestCheckRepo(t, repo)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestIndexSavePartial(t *testing.T) {
|
|
repository.TestAllVersions(t, testIndexSavePartial)
|
|
}
|
|
|
|
func testIndexSavePartial(t *testing.T, version uint) {
|
|
repo, unpacked := createFilledRepo(t, 3, version)
|
|
|
|
// capture blob list before adding fourth snapshot
|
|
idx := index.NewMasterIndex()
|
|
rtest.OK(t, idx.Load(context.TODO(), repo, restic.NoopCounter, nil))
|
|
blobs := make(map[pack.PackedBlob]struct{})
|
|
for pb := range idx.Values() {
|
|
blobs[*pb] = struct{}{}
|
|
}
|
|
|
|
// add+remove new snapshot and track its pack files
|
|
packsBefore := listPacks(t, repo)
|
|
sn := data.TestCreateSnapshot(t, repo, snapshotTime.Add(time.Duration(4)*time.Second), depth)
|
|
rtest.OK(t, repo.RemoveUnpacked(context.TODO(), restic.WriteableSnapshotFile, *sn.ID()))
|
|
packsAfter := listPacks(t, repo)
|
|
newPacks := packsAfter.Sub(packsBefore)
|
|
|
|
// rewrite index and remove pack files of new snapshot
|
|
idx = index.NewMasterIndex()
|
|
rtest.OK(t, idx.Load(context.TODO(), repo, restic.NoopCounter, nil))
|
|
rtest.OK(t, idx.Rewrite(context.TODO(), unpacked, newPacks, nil, nil, index.MasterIndexRewriteOpts{}))
|
|
|
|
// check blobs
|
|
idx = index.NewMasterIndex()
|
|
rtest.OK(t, idx.Load(context.TODO(), repo, restic.NoopCounter, nil))
|
|
for pb := range idx.Values() {
|
|
if _, ok := blobs[*pb]; ok {
|
|
delete(blobs, *pb)
|
|
} else {
|
|
t.Fatalf("unexpected blobs %v", pb)
|
|
}
|
|
}
|
|
rtest.Equals(t, 0, len(blobs), "saved index is missing blobs")
|
|
|
|
// remove pack files to make check happy
|
|
rtest.OK(t, restic.ParallelRemove(context.TODO(), unpacked, newPacks, restic.PackFile, nil, restic.NoopCounter))
|
|
|
|
checker.TestCheckRepo(t, repo)
|
|
}
|
|
|
|
func loadIndexAndCollectBlobs(t *testing.T, repo restic.ListerLoaderUnpacked, master *index.MasterIndex, indexCount int) map[pack.PackedBlob]struct{} {
|
|
p := progress.NewCounter(0, 0, nil)
|
|
rtest.OK(t, master.Load(context.TODO(), repo, p, nil))
|
|
v, max := p.Get()
|
|
rtest.Equals(t, uint64(indexCount), v)
|
|
rtest.Equals(t, uint64(indexCount), max)
|
|
return collectBlobs(master)
|
|
}
|
|
|
|
func collectBlobs(master *index.MasterIndex) map[pack.PackedBlob]struct{} {
|
|
s := make(map[pack.PackedBlob]struct{})
|
|
for pb := range master.Values() {
|
|
s[*pb] = struct{}{}
|
|
}
|
|
return s
|
|
}
|
|
|
|
func TestMasterIndexIncrementalLoad(t *testing.T) {
|
|
repo, _ := createFilledRepo(t, 3, restic.StableRepoVersion)
|
|
|
|
// Normal full index load
|
|
master1 := index.NewMasterIndex()
|
|
blobs1 := loadIndexAndCollectBlobs(t, repo, master1, 3)
|
|
|
|
// Noop reload should not change the index content
|
|
blobs1NoopLoad := loadIndexAndCollectBlobs(t, repo, master1, 0)
|
|
if !cmp.Equal(blobs1, blobs1NoopLoad) {
|
|
t.Fatalf("index content mismatch after noop reload: %v", cmp.Diff(blobs1, blobs1NoopLoad))
|
|
}
|
|
|
|
// Add new snapshot, which also results in a new index file
|
|
data.TestCreateSnapshot(t, repo, snapshotTime.Add(time.Duration(4)*time.Second), depth)
|
|
|
|
// Incremental load should only load the new index
|
|
blobs1IncrementalLoad := loadIndexAndCollectBlobs(t, repo, master1, 1)
|
|
|
|
// Reload index from scratch and compare with incremental load
|
|
master2 := index.NewMasterIndex()
|
|
blobs2 := loadIndexAndCollectBlobs(t, repo, master2, 4)
|
|
if !cmp.Equal(blobs1IncrementalLoad, blobs2) {
|
|
t.Fatalf("index content mismatch compared to full reload: %v", cmp.Diff(blobs1IncrementalLoad, blobs2))
|
|
}
|
|
}
|
|
|
|
func listPacks(t testing.TB, repo restic.Lister) restic.IDSet {
|
|
s := restic.NewIDSet()
|
|
rtest.OK(t, repo.List(context.TODO(), restic.PackFile, func(id restic.ID, _ int64) error {
|
|
s.Insert(id)
|
|
return nil
|
|
}))
|
|
return s
|
|
}
|
|
|
|
func TestRewriteOversizedIndex(t *testing.T) {
|
|
repo, unpacked, _ := repository.TestRepositoryWithVersion(t, 2)
|
|
|
|
const fullIndexCount = 1000
|
|
|
|
// replace index size checks for testing
|
|
originalIndexFull := index.Full
|
|
originalIndexOversized := index.Oversized
|
|
defer func() {
|
|
index.Full = originalIndexFull
|
|
index.Oversized = originalIndexOversized
|
|
}()
|
|
index.Full = func(idx *index.Index) bool {
|
|
return idx.Len(restic.DataBlob) > fullIndexCount
|
|
}
|
|
index.Oversized = func(idx *index.Index) bool {
|
|
return idx.Len(restic.DataBlob) > 2*fullIndexCount
|
|
}
|
|
|
|
var blobs pack.Blobs
|
|
|
|
// build oversized index
|
|
idx := index.NewIndex()
|
|
numPacks := 5
|
|
for p := 0; p < numPacks; p++ {
|
|
packID := restic.NewRandomID()
|
|
packBlobs := make(pack.Blobs, 0, fullIndexCount)
|
|
|
|
for i := 0; i < fullIndexCount; i++ {
|
|
blob := pack.Blob{
|
|
BlobHandle: restic.BlobHandle{
|
|
Type: restic.DataBlob,
|
|
ID: restic.NewRandomID(),
|
|
},
|
|
Length: 100,
|
|
Offset: uint(i * 100),
|
|
}
|
|
packBlobs = append(packBlobs, blob)
|
|
blobs = append(blobs, blob)
|
|
}
|
|
idx.StorePack(packID, packBlobs)
|
|
}
|
|
idx.Finalize()
|
|
|
|
_, err := idx.SaveIndex(context.Background(), unpacked)
|
|
rtest.OK(t, err)
|
|
|
|
// construct master index for the oversized index
|
|
mi := index.NewMasterIndex()
|
|
rtest.OK(t, mi.Load(context.Background(), repo, restic.NoopCounter, nil))
|
|
|
|
// rewrite the index
|
|
rtest.OK(t, mi.Rewrite(context.Background(), unpacked, nil, nil, nil, index.MasterIndexRewriteOpts{}))
|
|
|
|
// load the rewritten indexes
|
|
mi2 := index.NewMasterIndex()
|
|
rtest.OK(t, mi2.Load(context.Background(), repo, restic.NoopCounter, nil))
|
|
|
|
// verify that blobs are still in the index
|
|
for _, blob := range blobs {
|
|
_, found := mi2.LookupSize(blob.BlobHandle)
|
|
rtest.Assert(t, found, "blob %v missing after rewrite", blob.ID)
|
|
}
|
|
|
|
// check that multiple indexes were created
|
|
ids := mi2.IDs()
|
|
rtest.Assert(t, len(ids) > 1, "oversized index was not split into multiple indexes")
|
|
}
|
|
|
|
func TestRewriteSplitPacks(t *testing.T) {
|
|
repo, unpacked, _ := repository.TestRepositoryWithVersion(t, restic.StableRepoVersion)
|
|
|
|
bh1 := restic.NewRandomBlobHandle()
|
|
bh2 := restic.NewRandomBlobHandle()
|
|
bhOther := restic.NewRandomBlobHandle()
|
|
|
|
blob1 := &pack.PackedBlob{
|
|
Pack: restic.NewRandomID(),
|
|
Blob: pack.Blob{
|
|
BlobHandle: bh1,
|
|
Length: uint(crypto.CiphertextLength(10)),
|
|
Offset: 0,
|
|
},
|
|
}
|
|
blob2 := &pack.PackedBlob{
|
|
Pack: blob1.PackID(),
|
|
Blob: pack.Blob{
|
|
BlobHandle: bh2,
|
|
Length: uint(crypto.CiphertextLength(100)),
|
|
Offset: 10,
|
|
UncompressedLength: 200,
|
|
},
|
|
}
|
|
// used to force index repacking
|
|
blobOther := &pack.PackedBlob{
|
|
Pack: restic.NewRandomID(),
|
|
Blob: pack.Blob{
|
|
BlobHandle: bhOther,
|
|
Length: uint(crypto.CiphertextLength(100)),
|
|
Offset: 10,
|
|
},
|
|
}
|
|
|
|
mi := index.NewMasterIndex()
|
|
rtest.OK(t, mi.StorePack(context.TODO(), blob1.PackID(), pack.Blobs{blob1.Blob}, unpacked))
|
|
rtest.OK(t, mi.StorePack(context.TODO(), blobOther.PackID(), pack.Blobs{blobOther.Blob}, unpacked))
|
|
rtest.OK(t, mi.Flush(context.TODO(), unpacked))
|
|
rtest.OK(t, mi.StorePack(context.TODO(), blob2.PackID(), pack.Blobs{blob2.Blob}, unpacked))
|
|
rtest.OK(t, mi.StorePack(context.TODO(), blobOther.PackID(), pack.Blobs{blobOther.Blob}, unpacked))
|
|
rtest.OK(t, mi.Flush(context.TODO(), unpacked))
|
|
|
|
rtest.OK(t, mi.Rewrite(context.TODO(), unpacked, restic.NewIDSet(blobOther.PackID()), nil, nil, index.MasterIndexRewriteOpts{}))
|
|
|
|
mi = index.NewMasterIndex()
|
|
rtest.OK(t, mi.Load(context.TODO(), repo, restic.NoopCounter, nil))
|
|
|
|
// test that all blobs are still in the index
|
|
for _, blob := range []*pack.PackedBlob{blob1, blob2} {
|
|
blobs := mi.Lookup(blob.Handle())
|
|
rtest.Equals(t, []*pack.PackedBlob{blob}, blobs)
|
|
}
|
|
|
|
blobs := mi.Lookup(blobOther.Handle())
|
|
rtest.Equals(t, nil, blobs)
|
|
}
|
|
|
|
// TestRewriteFullPacks checks that Rewrite drops a duplicate full index for the same
|
|
// pack while keeping the other index files and blob lookups intact. Creates 3 indexes:
|
|
// - indexA: contains packA
|
|
// - indexB: contains packB
|
|
// - indexC: contains packB
|
|
// After the rewrite, indexC must be dropped. The other indexes must be kept.
|
|
func TestRewriteFullPacks(t *testing.T) {
|
|
originalFull := index.Full
|
|
defer func() {
|
|
index.Full = originalFull
|
|
}()
|
|
index.Full = func(*index.Index) bool { return true }
|
|
|
|
repo, unpacked, _ := repository.TestRepositoryWithVersion(t, restic.StableRepoVersion)
|
|
|
|
packA := restic.NewRandomID()
|
|
packB := restic.NewRandomID()
|
|
|
|
blobA := &pack.PackedBlob{
|
|
Pack: packA,
|
|
Blob: pack.Blob{
|
|
BlobHandle: restic.NewRandomBlobHandle(),
|
|
Length: uint(crypto.CiphertextLength(10)),
|
|
Offset: 0,
|
|
},
|
|
}
|
|
blobB := &pack.PackedBlob{
|
|
Pack: packB,
|
|
Blob: pack.Blob{
|
|
BlobHandle: restic.NewRandomBlobHandle(),
|
|
Length: uint(crypto.CiphertextLength(50)),
|
|
Offset: 0,
|
|
},
|
|
}
|
|
|
|
mi := index.NewMasterIndex()
|
|
rtest.OK(t, mi.StorePack(context.TODO(), packA, pack.Blobs{blobA.Blob}, unpacked))
|
|
rtest.OK(t, mi.Flush(context.TODO(), unpacked))
|
|
rtest.OK(t, mi.StorePack(context.TODO(), packB, pack.Blobs{blobB.Blob}, unpacked))
|
|
rtest.OK(t, mi.Flush(context.TODO(), unpacked))
|
|
rtest.OK(t, mi.StorePack(context.TODO(), packB, pack.Blobs{blobB.Blob}, unpacked))
|
|
rtest.OK(t, mi.Flush(context.TODO(), unpacked))
|
|
|
|
indexIDs := mi.IDs()
|
|
rtest.Equals(t, 3, len(indexIDs))
|
|
|
|
rtest.OK(t, mi.Rewrite(context.TODO(), unpacked, nil, indexIDs, nil, index.MasterIndexRewriteOpts{}))
|
|
|
|
mi2 := index.NewMasterIndex()
|
|
rtest.OK(t, mi2.Load(context.TODO(), repo, restic.NoopCounter, nil))
|
|
|
|
afterRewrite := mi2.IDs()
|
|
rtest.Equals(t, 2, len(afterRewrite))
|
|
rtest.Equals(t, 2, len(afterRewrite.Intersect(indexIDs)))
|
|
|
|
rtest.Equals(t, []*pack.PackedBlob{blobA}, mi2.Lookup(blobA.Handle()))
|
|
rtest.Equals(t, []*pack.PackedBlob{blobB}, mi2.Lookup(blobB.Handle()))
|
|
}
|