2
2
mirror of https://github.com/octoleo/restic.git synced 2025-01-25 16:18:34 +00:00
restic/internal/repository/repack_test.go

302 lines
7.0 KiB
Go
Raw Normal View History

2016-08-01 20:04:23 +02:00
package repository_test
import (
2017-06-05 23:56:59 +02:00
"context"
"math/rand"
"testing"
2020-11-02 12:55:34 +01:00
"time"
2017-07-23 14:21:03 +02:00
"github.com/restic/restic/internal/repository"
2017-07-24 17:42:25 +02:00
"github.com/restic/restic/internal/restic"
)
2016-08-01 20:24:15 +02:00
func randomSize(min, max int) int {
return rand.Intn(max-min) + min
}
func createRandomBlobs(t testing.TB, repo restic.Repository, blobs int, pData float32) {
2016-08-01 20:24:15 +02:00
for i := 0; i < blobs; i++ {
var (
2016-08-31 20:58:57 +02:00
tpe restic.BlobType
2016-08-01 20:24:15 +02:00
length int
)
if rand.Float32() < pData {
2016-08-31 23:07:50 +02:00
tpe = restic.DataBlob
length = randomSize(10*1024, 1024*1024) // 10KiB to 1MiB of data
2016-08-01 20:24:15 +02:00
} else {
2016-08-31 23:07:50 +02:00
tpe = restic.TreeBlob
length = randomSize(1*1024, 20*1024) // 1KiB to 20KiB
2016-08-01 20:24:15 +02:00
}
buf := make([]byte, length)
rand.Read(buf)
id, exists, err := repo.SaveBlob(context.TODO(), tpe, buf, restic.ID{}, false)
2016-08-01 20:24:15 +02:00
if err != nil {
t.Fatalf("SaveFrom() error %v", err)
}
if exists {
t.Errorf("duplicate blob %v/%v ignored", id, restic.DataBlob)
continue
}
2016-08-01 20:24:15 +02:00
if rand.Float32() < 0.2 {
if err = repo.Flush(context.Background()); err != nil {
2016-08-01 20:24:15 +02:00
t.Fatalf("repo.Flush() returned error %v", err)
}
}
}
if err := repo.Flush(context.Background()); err != nil {
2016-08-01 20:24:15 +02:00
t.Fatalf("repo.Flush() returned error %v", err)
}
}
func createRandomWrongBlob(t testing.TB, repo restic.Repository) {
length := randomSize(10*1024, 1024*1024) // 10KiB to 1MiB of data
buf := make([]byte, length)
rand.Read(buf)
id := restic.Hash(buf)
// invert first data byte
buf[0] ^= 0xff
_, _, err := repo.SaveBlob(context.TODO(), restic.DataBlob, buf, id, false)
if err != nil {
t.Fatalf("SaveFrom() error %v", err)
}
if err := repo.Flush(context.Background()); err != nil {
t.Fatalf("repo.Flush() returned error %v", err)
}
}
2016-08-01 20:24:15 +02:00
// selectBlobs splits the list of all blobs randomly into two lists. A blob
// will be contained in the firstone ith probability p.
func selectBlobs(t *testing.T, repo restic.Repository, p float32) (list1, list2 restic.BlobSet) {
2016-08-31 23:07:50 +02:00
list1 = restic.NewBlobSet()
list2 = restic.NewBlobSet()
2016-08-01 20:24:15 +02:00
2016-08-31 23:07:50 +02:00
blobs := restic.NewBlobSet()
err := repo.List(context.TODO(), restic.PackFile, func(id restic.ID, size int64) error {
entries, _, err := repo.ListPack(context.TODO(), id, size)
2016-08-01 20:24:15 +02:00
if err != nil {
t.Fatalf("error listing pack %v: %v", id, err)
}
for _, entry := range entries {
2016-08-31 23:07:50 +02:00
h := restic.BlobHandle{ID: entry.ID, Type: entry.Type}
if blobs.Has(h) {
t.Errorf("ignoring duplicate blob %v", h)
return nil
}
blobs.Insert(h)
2016-08-01 20:24:15 +02:00
if rand.Float32() <= p {
2016-08-31 23:07:50 +02:00
list1.Insert(restic.BlobHandle{ID: entry.ID, Type: entry.Type})
2016-08-01 20:24:15 +02:00
} else {
2016-08-31 23:07:50 +02:00
list2.Insert(restic.BlobHandle{ID: entry.ID, Type: entry.Type})
2016-08-01 20:24:15 +02:00
}
}
return nil
})
if err != nil {
t.Fatal(err)
2016-08-01 20:24:15 +02:00
}
return list1, list2
}
func listPacks(t *testing.T, repo restic.Repository) restic.IDSet {
2016-08-31 20:29:54 +02:00
list := restic.NewIDSet()
err := repo.List(context.TODO(), restic.PackFile, func(id restic.ID, size int64) error {
2016-08-01 20:24:15 +02:00
list.Insert(id)
return nil
})
if err != nil {
t.Fatal(err)
2016-08-01 20:24:15 +02:00
}
return list
}
func findPacksForBlobs(t *testing.T, repo restic.Repository, blobs restic.BlobSet) restic.IDSet {
2016-08-31 20:29:54 +02:00
packs := restic.NewIDSet()
2016-08-01 20:24:15 +02:00
idx := repo.Index()
for h := range blobs {
2020-11-05 22:18:00 +01:00
list := idx.Lookup(h)
if len(list) == 0 {
t.Fatal("Failed to find blob", h.ID.Str(), "with type", h.Type)
2016-08-01 20:24:15 +02:00
}
for _, pb := range list {
packs.Insert(pb.PackID)
}
2016-08-01 20:24:15 +02:00
}
return packs
}
func repack(t *testing.T, repo restic.Repository, packs restic.IDSet, blobs restic.BlobSet) {
repackedBlobs, err := repository.Repack(context.TODO(), repo, packs, blobs, nil)
if err != nil {
t.Fatal(err)
}
for id := range repackedBlobs {
err = repo.Backend().Remove(context.TODO(), restic.Handle{Type: restic.PackFile, Name: id.String()})
if err != nil {
t.Fatal(err)
}
}
}
func saveIndex(t *testing.T, repo restic.Repository) {
2017-06-05 23:56:59 +02:00
if err := repo.SaveIndex(context.TODO()); err != nil {
t.Fatalf("repo.SaveIndex() %v", err)
}
}
func rebuildIndex(t *testing.T, repo restic.Repository) {
2020-10-10 22:29:55 +02:00
err := repo.SetIndex(repository.NewMasterIndex())
if err != nil {
t.Fatal(err)
}
packs := make(map[restic.ID]int64)
err = repo.List(context.TODO(), restic.PackFile, func(id restic.ID, size int64) error {
packs[id] = size
return nil
})
if err != nil {
t.Fatal(err)
}
_, err = repo.(*repository.Repository).CreateIndexFromPacks(context.TODO(), packs, nil)
if err != nil {
t.Fatal(err)
}
err = repo.List(context.TODO(), restic.IndexFile, func(id restic.ID, size int64) error {
2017-06-05 23:56:59 +02:00
h := restic.Handle{
Type: restic.IndexFile,
Name: id.String(),
2017-06-05 23:56:59 +02:00
}
return repo.Backend().Remove(context.TODO(), h)
})
if err != nil {
t.Fatal(err)
}
2020-10-10 22:29:55 +02:00
_, err = (repo.Index()).(*repository.MasterIndex).
2020-10-18 09:24:34 +02:00
Save(context.TODO(), repo, restic.NewIDSet(), nil, nil)
2020-10-10 22:29:55 +02:00
if err != nil {
t.Fatal(err)
}
}
func reloadIndex(t *testing.T, repo restic.Repository) {
err := repo.SetIndex(repository.NewMasterIndex())
if err != nil {
t.Fatal(err)
}
2017-06-05 23:56:59 +02:00
if err := repo.LoadIndex(context.TODO()); err != nil {
t.Fatalf("error loading new index: %v", err)
}
}
func TestRepack(t *testing.T) {
repo, cleanup := repository.TestRepository(t)
defer cleanup()
2020-11-02 12:55:34 +01:00
seed := time.Now().UnixNano()
rand.Seed(seed)
t.Logf("rand seed is %v", seed)
createRandomBlobs(t, repo, 100, 0.7)
packsBefore := listPacks(t, repo)
// Running repack on empty ID sets should not do anything at all.
repack(t, repo, nil, nil)
packsAfter := listPacks(t, repo)
if !packsAfter.Equals(packsBefore) {
t.Fatalf("packs are not equal, Repack modified something. Before:\n %v\nAfter:\n %v",
packsBefore, packsAfter)
}
saveIndex(t, repo)
removeBlobs, keepBlobs := selectBlobs(t, repo, 0.2)
removePacks := findPacksForBlobs(t, repo, removeBlobs)
repack(t, repo, removePacks, keepBlobs)
rebuildIndex(t, repo)
reloadIndex(t, repo)
packsAfter = listPacks(t, repo)
for id := range removePacks {
if packsAfter.Has(id) {
t.Errorf("pack %v still present although it should have been repacked and removed", id.Str())
}
}
idx := repo.Index()
for h := range keepBlobs {
2020-11-05 22:18:00 +01:00
list := idx.Lookup(h)
if len(list) == 0 {
t.Errorf("unable to find blob %v in repo", h.ID.Str())
continue
}
if len(list) != 1 {
t.Errorf("expected one pack in the list, got: %v", list)
continue
}
pb := list[0]
if removePacks.Has(pb.PackID) {
t.Errorf("lookup returned pack ID %v that should've been removed", pb.PackID)
}
}
for h := range removeBlobs {
if _, found := repo.LookupBlobSize(h.ID, h.Type); found {
t.Errorf("blob %v still contained in the repo", h)
}
}
}
func TestRepackWrongBlob(t *testing.T) {
repo, cleanup := repository.TestRepository(t)
defer cleanup()
2020-11-02 12:55:34 +01:00
seed := time.Now().UnixNano()
rand.Seed(seed)
t.Logf("rand seed is %v", seed)
createRandomBlobs(t, repo, 5, 0.7)
createRandomWrongBlob(t, repo)
// just keep all blobs, but also rewrite every pack
_, keepBlobs := selectBlobs(t, repo, 0)
rewritePacks := findPacksForBlobs(t, repo, keepBlobs)
_, err := repository.Repack(context.TODO(), repo, rewritePacks, keepBlobs, nil)
if err == nil {
t.Fatal("expected repack to fail but got no error")
}
2020-11-02 12:53:45 +01:00
t.Logf("found expected error: %v", err)
}