mirror of
https://github.com/octoleo/restic.git
synced 2024-11-30 00:33:57 +00:00
9c55e8d69c
More optimizations to avoid calling Index.Lookup()
207 lines
4.5 KiB
Go
207 lines
4.5 KiB
Go
package archiver
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"io"
|
|
"math/rand"
|
|
"testing"
|
|
|
|
"github.com/restic/restic/internal/checker"
|
|
"github.com/restic/restic/internal/repository"
|
|
"github.com/restic/restic/internal/restic"
|
|
)
|
|
|
|
func loadBlob(t *testing.T, repo restic.Repository, id restic.ID, buf []byte) int {
|
|
n, err := repo.LoadBlob(context.TODO(), restic.DataBlob, id, buf)
|
|
if err != nil {
|
|
t.Fatalf("LoadBlob(%v) returned error %v", id, err)
|
|
}
|
|
|
|
return n
|
|
}
|
|
|
|
func checkSavedFile(t *testing.T, repo restic.Repository, treeID restic.ID, name string, rd io.Reader) {
|
|
tree, err := repo.LoadTree(context.TODO(), treeID)
|
|
if err != nil {
|
|
t.Fatalf("LoadTree() returned error %v", err)
|
|
}
|
|
|
|
if len(tree.Nodes) != 1 {
|
|
t.Fatalf("wrong number of nodes for tree, want %v, got %v", 1, len(tree.Nodes))
|
|
}
|
|
|
|
node := tree.Nodes[0]
|
|
if node.Name != "fakefile" {
|
|
t.Fatalf("wrong filename, want %v, got %v", "fakefile", node.Name)
|
|
}
|
|
|
|
if len(node.Content) == 0 {
|
|
t.Fatalf("node.Content has length 0")
|
|
}
|
|
|
|
// check blobs
|
|
for i, id := range node.Content {
|
|
size, found := repo.LookupBlobSize(id, restic.DataBlob)
|
|
if !found {
|
|
t.Fatal("Failed to find blob", id.Str())
|
|
}
|
|
|
|
buf := restic.NewBlobBuffer(int(size))
|
|
n := loadBlob(t, repo, id, buf)
|
|
if n != len(buf) {
|
|
t.Errorf("wrong number of bytes read, want %d, got %d", len(buf), n)
|
|
}
|
|
|
|
buf2 := make([]byte, int(size))
|
|
_, err := io.ReadFull(rd, buf2)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if !bytes.Equal(buf, buf2) {
|
|
t.Fatalf("blob %d (%v) is wrong", i, id.Str())
|
|
}
|
|
}
|
|
}
|
|
|
|
// fakeFile returns a reader which yields deterministic pseudo-random data.
|
|
func fakeFile(t testing.TB, seed, size int64) io.Reader {
|
|
return io.LimitReader(restic.NewRandReader(rand.New(rand.NewSource(seed))), size)
|
|
}
|
|
|
|
func TestArchiveReader(t *testing.T) {
|
|
repo, cleanup := repository.TestRepository(t)
|
|
defer cleanup()
|
|
|
|
seed := rand.Int63()
|
|
size := int64(rand.Intn(50*1024*1024) + 50*1024*1024)
|
|
t.Logf("seed is 0x%016x, size is %v", seed, size)
|
|
|
|
f := fakeFile(t, seed, size)
|
|
|
|
r := &Reader{
|
|
Repository: repo,
|
|
Hostname: "localhost",
|
|
Tags: []string{"test"},
|
|
}
|
|
|
|
sn, id, err := r.Archive(context.TODO(), "fakefile", f, nil)
|
|
if err != nil {
|
|
t.Fatalf("ArchiveReader() returned error %v", err)
|
|
}
|
|
|
|
if id.IsNull() {
|
|
t.Fatalf("ArchiveReader() returned null ID")
|
|
}
|
|
|
|
t.Logf("snapshot saved as %v, tree is %v", id.Str(), sn.Tree.Str())
|
|
|
|
checkSavedFile(t, repo, *sn.Tree, "fakefile", fakeFile(t, seed, size))
|
|
|
|
checker.TestCheckRepo(t, repo)
|
|
}
|
|
|
|
func TestArchiveReaderNull(t *testing.T) {
|
|
repo, cleanup := repository.TestRepository(t)
|
|
defer cleanup()
|
|
|
|
r := &Reader{
|
|
Repository: repo,
|
|
Hostname: "localhost",
|
|
Tags: []string{"test"},
|
|
}
|
|
|
|
sn, id, err := r.Archive(context.TODO(), "fakefile", bytes.NewReader(nil), nil)
|
|
if err != nil {
|
|
t.Fatalf("ArchiveReader() returned error %v", err)
|
|
}
|
|
|
|
if id.IsNull() {
|
|
t.Fatalf("ArchiveReader() returned null ID")
|
|
}
|
|
|
|
t.Logf("snapshot saved as %v, tree is %v", id.Str(), sn.Tree.Str())
|
|
|
|
checker.TestCheckRepo(t, repo)
|
|
}
|
|
|
|
type errReader string
|
|
|
|
func (e errReader) Read([]byte) (int, error) {
|
|
return 0, errors.New(string(e))
|
|
}
|
|
|
|
func countSnapshots(t testing.TB, repo restic.Repository) int {
|
|
snapshots := 0
|
|
err := repo.List(context.TODO(), restic.SnapshotFile, func(id restic.ID, size int64) error {
|
|
snapshots++
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return snapshots
|
|
}
|
|
|
|
func TestArchiveReaderError(t *testing.T) {
|
|
repo, cleanup := repository.TestRepository(t)
|
|
defer cleanup()
|
|
|
|
r := &Reader{
|
|
Repository: repo,
|
|
Hostname: "localhost",
|
|
Tags: []string{"test"},
|
|
}
|
|
|
|
sn, id, err := r.Archive(context.TODO(), "fakefile", errReader("error returned by reading stdin"), nil)
|
|
if err == nil {
|
|
t.Errorf("expected error not returned")
|
|
}
|
|
|
|
if sn != nil {
|
|
t.Errorf("Snapshot should be nil, but isn't")
|
|
}
|
|
|
|
if !id.IsNull() {
|
|
t.Errorf("id should be null, but %v returned", id.Str())
|
|
}
|
|
|
|
n := countSnapshots(t, repo)
|
|
if n > 0 {
|
|
t.Errorf("expected zero snapshots, but got %d", n)
|
|
}
|
|
|
|
checker.TestCheckRepo(t, repo)
|
|
}
|
|
|
|
func BenchmarkArchiveReader(t *testing.B) {
|
|
repo, cleanup := repository.TestRepository(t)
|
|
defer cleanup()
|
|
|
|
const size = 50 * 1024 * 1024
|
|
|
|
buf := make([]byte, size)
|
|
_, err := io.ReadFull(fakeFile(t, 23, size), buf)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
r := &Reader{
|
|
Repository: repo,
|
|
Hostname: "localhost",
|
|
Tags: []string{"test"},
|
|
}
|
|
|
|
t.SetBytes(size)
|
|
t.ResetTimer()
|
|
|
|
for i := 0; i < t.N; i++ {
|
|
_, _, err := r.Archive(context.TODO(), "fakefile", bytes.NewReader(buf), nil)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
}
|