restic/internal/testing.go

214 lines
4.6 KiB
Go
Raw Normal View History

2016-04-10 14:52:15 +00:00
package restic
import (
2017-06-04 09:16:55 +00:00
"context"
"encoding/json"
2016-04-10 14:52:15 +00:00
"fmt"
"io"
"math/rand"
"testing"
"time"
2016-09-01 20:17:37 +00:00
"restic/errors"
"github.com/restic/chunker"
2016-04-10 14:52:15 +00:00
)
// fakeFile returns a reader which yields deterministic pseudo-random data.
func fakeFile(t testing.TB, seed, size int64) io.Reader {
2016-08-31 18:29:54 +00:00
return io.LimitReader(NewRandReader(rand.New(rand.NewSource(seed))), size)
2016-04-10 14:52:15 +00:00
}
type fakeFileSystem struct {
t testing.TB
2016-08-31 18:29:54 +00:00
repo Repository
knownBlobs IDSet
duplication float32
buf []byte
chunker *chunker.Chunker
}
2016-04-10 14:52:15 +00:00
// saveFile reads from rd and saves the blobs in the repository. The list of
// IDs is returned.
2017-06-04 09:16:55 +00:00
func (fs *fakeFileSystem) saveFile(ctx context.Context, rd io.Reader) (blobs IDs) {
if fs.buf == nil {
fs.buf = make([]byte, chunker.MaxSize)
}
2016-04-10 14:52:15 +00:00
if fs.chunker == nil {
fs.chunker = chunker.New(rd, fs.repo.Config().ChunkerPolynomial)
} else {
fs.chunker.Reset(rd, fs.repo.Config().ChunkerPolynomial)
}
blobs = IDs{}
2016-04-10 14:52:15 +00:00
for {
chunk, err := fs.chunker.Next(fs.buf)
if errors.Cause(err) == io.EOF {
2016-04-10 14:52:15 +00:00
break
}
if err != nil {
fs.t.Fatalf("unable to save chunk in repo: %v", err)
2016-04-10 14:52:15 +00:00
}
2016-08-31 18:29:54 +00:00
id := Hash(chunk.Data)
2016-08-31 18:58:57 +00:00
if !fs.blobIsKnown(id, DataBlob) {
2017-06-04 09:16:55 +00:00
_, err := fs.repo.SaveBlob(ctx, DataBlob, chunk.Data, id)
if err != nil {
fs.t.Fatalf("error saving chunk: %v", err)
}
fs.knownBlobs.Insert(id)
2016-04-10 14:52:15 +00:00
}
2016-04-10 14:52:15 +00:00
blobs = append(blobs, id)
}
return blobs
}
2016-07-31 08:58:09 +00:00
const (
maxFileSize = 1500000
maxSeed = 32
2016-07-31 08:58:09 +00:00
maxNodes = 32
)
2016-04-10 15:25:32 +00:00
func (fs *fakeFileSystem) treeIsKnown(tree *Tree) (bool, []byte, ID) {
data, err := json.Marshal(tree)
if err != nil {
fs.t.Fatalf("json.Marshal(tree) returned error: %v", err)
2016-09-03 18:55:22 +00:00
return false, nil, ID{}
}
data = append(data, '\n')
2016-08-31 18:29:54 +00:00
id := Hash(data)
2016-09-03 18:55:22 +00:00
return fs.blobIsKnown(id, TreeBlob), data, id
}
func (fs *fakeFileSystem) blobIsKnown(id ID, t BlobType) bool {
if rand.Float32() < fs.duplication {
return false
}
if fs.knownBlobs.Has(id) {
return true
}
if fs.repo.Index().Has(id, t) {
return true
}
fs.knownBlobs.Insert(id)
return false
}
// saveTree saves a tree of fake files in the repo and returns the ID.
2017-06-04 09:16:55 +00:00
func (fs *fakeFileSystem) saveTree(ctx context.Context, seed int64, depth int) ID {
2016-04-10 14:52:15 +00:00
rnd := rand.NewSource(seed)
2016-07-31 08:58:09 +00:00
numNodes := int(rnd.Int63() % maxNodes)
2016-04-10 14:52:15 +00:00
var tree Tree
for i := 0; i < numNodes; i++ {
2016-07-31 08:58:09 +00:00
// randomly select the type of the node, either tree (p = 1/4) or file (p = 3/4).
if depth > 1 && rnd.Int63()%4 == 0 {
treeSeed := rnd.Int63() % maxSeed
2017-06-04 09:16:55 +00:00
id := fs.saveTree(ctx, treeSeed, depth-1)
2016-07-31 08:58:09 +00:00
node := &Node{
2016-09-01 19:20:03 +00:00
Name: fmt.Sprintf("dir-%v", treeSeed),
Type: "dir",
Mode: 0755,
Subtree: &id,
2016-07-31 08:58:09 +00:00
}
tree.Nodes = append(tree.Nodes, node)
continue
}
fileSeed := rnd.Int63() % maxSeed
fileSize := (maxFileSize / maxSeed) * fileSeed
2016-04-10 15:25:32 +00:00
node := &Node{
2016-09-01 19:20:03 +00:00
Name: fmt.Sprintf("file-%v", fileSeed),
Type: "file",
Mode: 0644,
Size: uint64(fileSize),
2016-04-10 15:25:32 +00:00
}
2016-04-10 14:52:15 +00:00
2017-06-04 09:16:55 +00:00
node.Content = fs.saveFile(ctx, fakeFile(fs.t, fileSeed, fileSize))
2016-04-10 14:52:15 +00:00
tree.Nodes = append(tree.Nodes, node)
}
2016-09-03 18:55:22 +00:00
known, buf, id := fs.treeIsKnown(&tree)
if known {
return id
}
2017-06-04 09:16:55 +00:00
_, err := fs.repo.SaveBlob(ctx, TreeBlob, buf, id)
2016-04-10 14:52:15 +00:00
if err != nil {
fs.t.Fatal(err)
2016-04-10 14:52:15 +00:00
}
return id
}
// TestCreateSnapshot creates a snapshot filled with fake data. The
2016-04-10 14:52:15 +00:00
// fake data is generated deterministically from the timestamp `at`, which is
2016-07-31 08:58:09 +00:00
// also used as the snapshot's timestamp. The tree's depth can be specified
// with the parameter depth. The parameter duplication is a probability that
// the same blob will saved again.
2016-08-31 18:29:54 +00:00
func TestCreateSnapshot(t testing.TB, repo Repository, at time.Time, depth int, duplication float32) *Snapshot {
2016-07-31 08:58:09 +00:00
seed := at.Unix()
t.Logf("create fake snapshot at %s with seed %d", at, seed)
2016-04-10 14:52:15 +00:00
fakedir := fmt.Sprintf("fakedir-at-%v", at.Format("2006-01-02 15:04:05"))
2017-02-10 18:37:33 +00:00
snapshot, err := NewSnapshot([]string{fakedir}, []string{"test"}, "foo")
2016-04-10 14:52:15 +00:00
if err != nil {
t.Fatal(err)
}
snapshot.Time = at
fs := fakeFileSystem{
t: t,
repo: repo,
2016-08-31 18:29:54 +00:00
knownBlobs: NewIDSet(),
duplication: duplication,
}
2017-06-04 09:16:55 +00:00
treeID := fs.saveTree(context.TODO(), seed, depth)
2016-04-10 14:52:15 +00:00
snapshot.Tree = &treeID
2017-06-04 09:16:55 +00:00
id, err := repo.SaveJSONUnpacked(context.TODO(), SnapshotFile, snapshot)
2016-04-10 14:52:15 +00:00
if err != nil {
t.Fatal(err)
}
snapshot.id = &id
2016-04-10 14:52:15 +00:00
t.Logf("saved snapshot %v", id.Str())
err = repo.Flush()
if err != nil {
t.Fatal(err)
}
2017-06-04 09:16:55 +00:00
err = repo.SaveIndex(context.TODO())
2016-04-10 14:52:15 +00:00
if err != nil {
t.Fatal(err)
}
return snapshot
2016-04-10 14:52:15 +00:00
}
2016-08-31 18:29:54 +00:00
// TestParseID parses s as a ID and panics if that fails.
func TestParseID(s string) ID {
id, err := ParseID(s)
if err != nil {
panic(fmt.Sprintf("unable to parse string %q as ID: %v", s, err))
}
2016-08-31 18:29:54 +00:00
return id
}