forked from TrueCloudLab/restic
retry key loading on hash mismatch
This commit is contained in:
parent
78d2312ee9
commit
822422ef03
3 changed files with 57 additions and 1 deletions
|
@ -6,6 +6,8 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
|
"github.com/restic/restic/internal/debug"
|
||||||
|
"github.com/restic/restic/internal/errors"
|
||||||
"github.com/restic/restic/internal/restic"
|
"github.com/restic/restic/internal/restic"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -13,6 +15,7 @@ import (
|
||||||
// buffer, which is truncated. If the buffer is not large enough or nil, a new
|
// buffer, which is truncated. If the buffer is not large enough or nil, a new
|
||||||
// one is allocated.
|
// one is allocated.
|
||||||
func LoadAll(ctx context.Context, buf []byte, be restic.Backend, h restic.Handle) ([]byte, error) {
|
func LoadAll(ctx context.Context, buf []byte, be restic.Backend, h restic.Handle) ([]byte, error) {
|
||||||
|
retriedInvalidData := false
|
||||||
err := be.Load(ctx, h, 0, 0, func(rd io.Reader) error {
|
err := be.Load(ctx, h, 0, 0, func(rd io.Reader) error {
|
||||||
// make sure this is idempotent, in case an error occurs this function may be called multiple times!
|
// make sure this is idempotent, in case an error occurs this function may be called multiple times!
|
||||||
wr := bytes.NewBuffer(buf[:0])
|
wr := bytes.NewBuffer(buf[:0])
|
||||||
|
@ -21,6 +24,18 @@ func LoadAll(ctx context.Context, buf []byte, be restic.Backend, h restic.Handle
|
||||||
return cerr
|
return cerr
|
||||||
}
|
}
|
||||||
buf = wr.Bytes()
|
buf = wr.Bytes()
|
||||||
|
|
||||||
|
// retry loading damaged data only once. If a file fails to download correctly
|
||||||
|
// the second time, then it is likely corrupted at the backend. Return the data
|
||||||
|
// to the caller in that case to let it decide what to do with the data.
|
||||||
|
if !retriedInvalidData && h.Type != restic.ConfigFile {
|
||||||
|
id, err := restic.ParseID(h.Name)
|
||||||
|
if err == nil && !restic.Hash(buf).Equal(id) {
|
||||||
|
debug.Log("retry loading broken blob %v", h)
|
||||||
|
retriedInvalidData = true
|
||||||
|
return errors.Errorf("loadAll(%v): invalid data returned", h)
|
||||||
|
}
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,7 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
"math/rand"
|
"math/rand"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
@ -56,6 +57,47 @@ func save(t testing.TB, be restic.Backend, buf []byte) restic.Handle {
|
||||||
return h
|
return h
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type quickRetryBackend struct {
|
||||||
|
restic.Backend
|
||||||
|
}
|
||||||
|
|
||||||
|
func (be *quickRetryBackend) Load(ctx context.Context, h restic.Handle, length int, offset int64, fn func(rd io.Reader) error) error {
|
||||||
|
err := be.Backend.Load(ctx, h, length, offset, fn)
|
||||||
|
if err != nil {
|
||||||
|
// retry
|
||||||
|
err = be.Backend.Load(ctx, h, length, offset, fn)
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLoadAllBroken(t *testing.T) {
|
||||||
|
b := mock.NewBackend()
|
||||||
|
|
||||||
|
data := rtest.Random(23, rand.Intn(MiB)+500*KiB)
|
||||||
|
id := restic.Hash(data)
|
||||||
|
// damage buffer
|
||||||
|
data[0] ^= 0xff
|
||||||
|
|
||||||
|
b.OpenReaderFn = func(ctx context.Context, h restic.Handle, length int, offset int64) (io.ReadCloser, error) {
|
||||||
|
return ioutil.NopCloser(bytes.NewReader(data)), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// must fail on first try
|
||||||
|
_, err := backend.LoadAll(context.TODO(), nil, b, restic.Handle{Type: restic.PackFile, Name: id.String()})
|
||||||
|
if err == nil {
|
||||||
|
t.Fatalf("missing expected error")
|
||||||
|
}
|
||||||
|
|
||||||
|
// must return the broken data after a retry
|
||||||
|
be := &quickRetryBackend{Backend: b}
|
||||||
|
buf, err := backend.LoadAll(context.TODO(), nil, be, restic.Handle{Type: restic.PackFile, Name: id.String()})
|
||||||
|
rtest.OK(t, err)
|
||||||
|
|
||||||
|
if !bytes.Equal(buf, data) {
|
||||||
|
t.Fatalf("wrong data returned")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestLoadAllAppend(t *testing.T) {
|
func TestLoadAllAppend(t *testing.T) {
|
||||||
b := mem.New()
|
b := mem.New()
|
||||||
|
|
||||||
|
|
1
internal/cache/backend_test.go
vendored
1
internal/cache/backend_test.go
vendored
|
@ -48,7 +48,6 @@ func remove(t testing.TB, be restic.Backend, h restic.Handle) {
|
||||||
func randomData(n int) (restic.Handle, []byte) {
|
func randomData(n int) (restic.Handle, []byte) {
|
||||||
data := test.Random(rand.Int(), n)
|
data := test.Random(rand.Int(), n)
|
||||||
id := restic.Hash(data)
|
id := restic.Hash(data)
|
||||||
copy(id[:], data)
|
|
||||||
h := restic.Handle{
|
h := restic.Handle{
|
||||||
Type: restic.IndexFile,
|
Type: restic.IndexFile,
|
||||||
Name: id.String(),
|
Name: id.String(),
|
||||||
|
|
Loading…
Reference in a new issue