2023-05-06 08:37:17 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"io"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/restic/restic/internal/errors"
|
|
|
|
"github.com/restic/restic/internal/index"
|
|
|
|
"github.com/restic/restic/internal/restic"
|
|
|
|
rtest "github.com/restic/restic/internal/test"
|
|
|
|
)
|
|
|
|
|
|
|
|
func testRunRebuildIndex(t testing.TB, gopts GlobalOptions) {
|
2023-05-07 20:06:39 +00:00
|
|
|
rtest.OK(t, withRestoreGlobalOptions(func() error {
|
|
|
|
globalOptions.stdout = io.Discard
|
|
|
|
return runRebuildIndex(context.TODO(), RepairIndexOptions{}, gopts)
|
|
|
|
}))
|
2023-05-06 08:37:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func testRebuildIndex(t *testing.T, backendTestHook backendWrapper) {
|
|
|
|
env, cleanup := withTestEnvironment(t)
|
|
|
|
defer cleanup()
|
|
|
|
|
|
|
|
datafile := filepath.Join("..", "..", "internal", "checker", "testdata", "duplicate-packs-in-index-test-repo.tar.gz")
|
|
|
|
rtest.SetupTarTestFixture(t, env.base, datafile)
|
|
|
|
|
|
|
|
out, err := testRunCheckOutput(env.gopts, false)
|
|
|
|
if !strings.Contains(out, "contained in several indexes") {
|
|
|
|
t.Fatalf("did not find checker hint for packs in several indexes")
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("expected no error from checker for test repository, got %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !strings.Contains(out, "restic repair index") {
|
|
|
|
t.Fatalf("did not find hint for repair index command")
|
|
|
|
}
|
|
|
|
|
|
|
|
env.gopts.backendTestHook = backendTestHook
|
|
|
|
testRunRebuildIndex(t, env.gopts)
|
|
|
|
|
|
|
|
env.gopts.backendTestHook = nil
|
|
|
|
out, err = testRunCheckOutput(env.gopts, false)
|
|
|
|
if len(out) != 0 {
|
|
|
|
t.Fatalf("expected no output from the checker, got: %v", out)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("expected no error from checker after repair index, got: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRebuildIndex(t *testing.T) {
|
|
|
|
testRebuildIndex(t, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRebuildIndexAlwaysFull(t *testing.T) {
|
|
|
|
indexFull := index.IndexFull
|
|
|
|
defer func() {
|
|
|
|
index.IndexFull = indexFull
|
|
|
|
}()
|
|
|
|
index.IndexFull = func(*index.Index, bool) bool { return true }
|
|
|
|
testRebuildIndex(t, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
// indexErrorBackend modifies the first index after reading.
|
|
|
|
type indexErrorBackend struct {
|
|
|
|
restic.Backend
|
|
|
|
lock sync.Mutex
|
|
|
|
hasErred bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *indexErrorBackend) Load(ctx context.Context, h restic.Handle, length int, offset int64, consumer func(rd io.Reader) error) error {
|
|
|
|
return b.Backend.Load(ctx, h, length, offset, func(rd io.Reader) error {
|
|
|
|
// protect hasErred
|
|
|
|
b.lock.Lock()
|
|
|
|
defer b.lock.Unlock()
|
|
|
|
if !b.hasErred && h.Type == restic.IndexFile {
|
|
|
|
b.hasErred = true
|
|
|
|
return consumer(errorReadCloser{rd})
|
|
|
|
}
|
|
|
|
return consumer(rd)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
type errorReadCloser struct {
|
|
|
|
io.Reader
|
|
|
|
}
|
|
|
|
|
|
|
|
func (erd errorReadCloser) Read(p []byte) (int, error) {
|
|
|
|
n, err := erd.Reader.Read(p)
|
|
|
|
if n > 0 {
|
|
|
|
p[0] ^= 1
|
|
|
|
}
|
|
|
|
return n, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRebuildIndexDamage(t *testing.T) {
|
|
|
|
testRebuildIndex(t, func(r restic.Backend) (restic.Backend, error) {
|
|
|
|
return &indexErrorBackend{
|
|
|
|
Backend: r,
|
|
|
|
}, nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
type appendOnlyBackend struct {
|
|
|
|
restic.Backend
|
|
|
|
}
|
|
|
|
|
|
|
|
// called via repo.Backend().Remove()
|
|
|
|
func (b *appendOnlyBackend) Remove(_ context.Context, h restic.Handle) error {
|
|
|
|
return errors.Errorf("Failed to remove %v", h)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRebuildIndexFailsOnAppendOnly(t *testing.T) {
|
|
|
|
env, cleanup := withTestEnvironment(t)
|
|
|
|
defer cleanup()
|
|
|
|
|
|
|
|
datafile := filepath.Join("..", "..", "internal", "checker", "testdata", "duplicate-packs-in-index-test-repo.tar.gz")
|
|
|
|
rtest.SetupTarTestFixture(t, env.base, datafile)
|
|
|
|
|
2023-05-07 20:06:39 +00:00
|
|
|
err := withRestoreGlobalOptions(func() error {
|
|
|
|
globalOptions.stdout = io.Discard
|
|
|
|
|
|
|
|
env.gopts.backendTestHook = func(r restic.Backend) (restic.Backend, error) {
|
|
|
|
return &appendOnlyBackend{r}, nil
|
|
|
|
}
|
|
|
|
return runRebuildIndex(context.TODO(), RepairIndexOptions{}, env.gopts)
|
|
|
|
})
|
2023-05-06 08:37:17 +00:00
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
t.Error("expected rebuildIndex to fail")
|
|
|
|
}
|
|
|
|
t.Log(err)
|
|
|
|
}
|