Merge pull request #1049 from restic/fix-backend-tests-delayed-remove

backend tests: Add configurable delay for delayed remove
This commit is contained in:
Alexander Neumann 2017-06-18 18:31:38 +02:00
commit de92ce7a88
4 changed files with 30 additions and 13 deletions

View file

@ -19,6 +19,9 @@ func newB2TestSuite(t testing.TB) *test.Suite {
// do not use excessive data // do not use excessive data
MinimalData: true, MinimalData: true,
// wait for at most 10 seconds for removed files to disappear
WaitForDelayedRemoval: 10 * time.Second,
// NewConfig returns a config for a new temporary backend that will be used in tests. // NewConfig returns a config for a new temporary backend that will be used in tests.
NewConfig: func() (interface{}, error) { NewConfig: func() (interface{}, error) {
b2cfg, err := b2.ParseConfig(os.Getenv("RESTIC_TEST_B2_REPOSITORY")) b2cfg, err := b2.ParseConfig(os.Getenv("RESTIC_TEST_B2_REPOSITORY"))

View file

@ -20,6 +20,9 @@ func newSwiftTestSuite(t testing.TB) *test.Suite {
// do not use excessive data // do not use excessive data
MinimalData: true, MinimalData: true,
// wait for removals for at least 10s
WaitForDelayedRemoval: 10 * time.Second,
// NewConfig returns a config for a new temporary backend that will be used in tests. // NewConfig returns a config for a new temporary backend that will be used in tests.
NewConfig: func() (interface{}, error) { NewConfig: func() (interface{}, error) {
swiftcfg, err := swift.ParseConfig(os.Getenv("RESTIC_TEST_SWIFT")) swiftcfg, err := swift.ParseConfig(os.Getenv("RESTIC_TEST_SWIFT"))

View file

@ -6,10 +6,12 @@ import (
"restic/test" "restic/test"
"strings" "strings"
"testing" "testing"
"time"
) )
// Suite implements a test suite for restic backends. // Suite implements a test suite for restic backends.
type Suite struct { type Suite struct {
// Config should be used to configure the backend.
Config interface{} Config interface{}
// NewConfig returns a config for a new temporary backend that will be used in tests. // NewConfig returns a config for a new temporary backend that will be used in tests.
@ -26,6 +28,11 @@ type Suite struct {
// MinimalData instructs the tests to not use excessive data. // MinimalData instructs the tests to not use excessive data.
MinimalData bool MinimalData bool
// WaitForDelayedRemoval is set to a non-zero value to instruct the test
// suite to wait for this amount of time until a file that was removed
// really disappeared.
WaitForDelayedRemoval time.Duration
} }
// RunTests executes all defined tests as subtests of t. // RunTests executes all defined tests as subtests of t.

View file

@ -435,17 +435,18 @@ func testLoad(b restic.Backend, h restic.Handle, length int, offset int64) error
return err return err
} }
func delayedRemove(b restic.Backend, h restic.Handle) error { func delayedRemove(t testing.TB, be restic.Backend, h restic.Handle, maxwait time.Duration) error {
// Some backend (swift, I'm looking at you) may implement delayed // Some backend (swift, I'm looking at you) may implement delayed
// removal of data. Let's wait a bit if this happens. // removal of data. Let's wait a bit if this happens.
err := b.Remove(context.TODO(), h) err := be.Remove(context.TODO(), h)
if err != nil { if err != nil {
return err return err
} }
found, err := b.Test(context.TODO(), h) start := time.Now()
for i := 0; found && i < 20; i++ { attempt := 0
found, err = b.Test(context.TODO(), h) for time.Since(start) <= maxwait {
found, err := be.Test(context.TODO(), h)
if err != nil { if err != nil {
return err return err
} }
@ -454,20 +455,23 @@ func delayedRemove(b restic.Backend, h restic.Handle) error {
break break
} }
time.Sleep(100 * time.Millisecond) time.Sleep(500 * time.Millisecond)
attempt++
} }
return err
return nil
} }
func delayedList(t testing.TB, b restic.Backend, tpe restic.FileType, max int) restic.IDs { func delayedList(t testing.TB, b restic.Backend, tpe restic.FileType, max int, maxwait time.Duration) restic.IDs {
list := restic.NewIDSet() list := restic.NewIDSet()
start := time.Now()
for i := 0; i < max; i++ { for i := 0; i < max; i++ {
for s := range b.List(context.TODO(), tpe) { for s := range b.List(context.TODO(), tpe) {
id := restic.TestParseID(s) id := restic.TestParseID(s)
list.Insert(id) list.Insert(id)
} }
if len(list) < max { if len(list) < max && time.Since(start) < maxwait {
time.Sleep(100 * time.Millisecond) time.Sleep(500 * time.Millisecond)
} }
} }
@ -548,7 +552,7 @@ func (s *Suite) TestBackend(t *testing.T) {
test.Assert(t, err != nil, "expected error for %v, got %v", h, err) test.Assert(t, err != nil, "expected error for %v, got %v", h, err)
// remove and recreate // remove and recreate
err = delayedRemove(b, h) err = delayedRemove(t, b, h, s.WaitForDelayedRemoval)
test.OK(t, err) test.OK(t, err)
// test that the blob is gone // test that the blob is gone
@ -569,7 +573,7 @@ func (s *Suite) TestBackend(t *testing.T) {
IDs = append(IDs, id) IDs = append(IDs, id)
} }
list := delayedList(t, b, tpe, len(IDs)) list := delayedList(t, b, tpe, len(IDs), s.WaitForDelayedRemoval)
if len(IDs) != len(list) { if len(IDs) != len(list) {
t.Fatalf("wrong number of IDs returned: want %d, got %d", len(IDs), len(list)) t.Fatalf("wrong number of IDs returned: want %d, got %d", len(IDs), len(list))
} }
@ -593,7 +597,7 @@ func (s *Suite) TestBackend(t *testing.T) {
test.OK(t, err) test.OK(t, err)
test.Assert(t, found, fmt.Sprintf("id %q not found", id)) test.Assert(t, found, fmt.Sprintf("id %q not found", id))
test.OK(t, delayedRemove(b, h)) test.OK(t, delayedRemove(t, b, h, s.WaitForDelayedRemoval))
found, err = b.Test(context.TODO(), h) found, err = b.Test(context.TODO(), h)
test.OK(t, err) test.OK(t, err)