| 
									
										
										
										
											2022-10-15 16:33:15 +02:00
										 |  |  | package retry | 
					
						
							| 
									
										
										
										
											2017-10-17 21:46:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2024-05-12 12:34:54 +02:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2017-10-17 21:46:38 +02:00
										 |  |  | 	"testing" | 
					
						
							| 
									
										
										
										
											2020-03-21 19:05:36 +00:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2017-10-17 21:46:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-21 19:05:36 +00:00
										 |  |  | 	"github.com/cenkalti/backoff/v4" | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	"github.com/restic/restic/internal/backend" | 
					
						
							| 
									
										
										
										
											2022-06-12 14:38:19 +02:00
										 |  |  | 	"github.com/restic/restic/internal/backend/mock" | 
					
						
							| 
									
										
										
										
											2017-10-17 21:46:38 +02:00
										 |  |  | 	"github.com/restic/restic/internal/errors" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/restic" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/test" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestBackendSaveRetry(t *testing.T) { | 
					
						
							|  |  |  | 	buf := bytes.NewBuffer(nil) | 
					
						
							|  |  |  | 	errcount := 0 | 
					
						
							|  |  |  | 	be := &mock.Backend{ | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 		SaveFn: func(ctx context.Context, h backend.Handle, rd backend.RewindReader) error { | 
					
						
							| 
									
										
										
										
											2017-10-17 21:46:38 +02:00
										 |  |  | 			if errcount == 0 { | 
					
						
							|  |  |  | 				errcount++ | 
					
						
							| 
									
										
										
										
											2022-12-02 19:36:43 +01:00
										 |  |  | 				_, err := io.CopyN(io.Discard, rd, 120) | 
					
						
							| 
									
										
										
										
											2017-10-17 21:46:38 +02:00
										 |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				return errors.New("injected error") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			_, err := io.Copy(buf, rd) | 
					
						
							|  |  |  | 			return err | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-15 23:13:44 +02:00
										 |  |  | 	TestFastRetries(t) | 
					
						
							| 
									
										
										
										
											2022-10-15 16:33:15 +02:00
										 |  |  | 	retryBackend := New(be, 10, nil, nil) | 
					
						
							| 
									
										
										
										
											2017-10-17 21:46:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	data := test.Random(23, 5*1024*1024+11241) | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	err := retryBackend.Save(context.TODO(), backend.Handle{}, backend.NewByteReader(data, be.Hasher())) | 
					
						
							| 
									
										
										
										
											2017-10-17 21:46:38 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if len(data) != buf.Len() { | 
					
						
							|  |  |  | 		t.Errorf("wrong number of bytes written: want %d, got %d", len(data), buf.Len()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !bytes.Equal(data, buf.Bytes()) { | 
					
						
							|  |  |  | 		t.Errorf("wrong data written to backend") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-01-24 10:25:40 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-26 21:20:52 +02:00
										 |  |  | func TestBackendSaveRetryAtomic(t *testing.T) { | 
					
						
							|  |  |  | 	errcount := 0 | 
					
						
							|  |  |  | 	calledRemove := false | 
					
						
							|  |  |  | 	be := &mock.Backend{ | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 		SaveFn: func(ctx context.Context, h backend.Handle, rd backend.RewindReader) error { | 
					
						
							| 
									
										
										
										
											2022-08-26 21:20:52 +02:00
										 |  |  | 			if errcount == 0 { | 
					
						
							|  |  |  | 				errcount++ | 
					
						
							|  |  |  | 				return errors.New("injected error") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 		RemoveFn: func(ctx context.Context, h backend.Handle) error { | 
					
						
							| 
									
										
										
										
											2022-08-26 21:20:52 +02:00
										 |  |  | 			calledRemove = true | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		HasAtomicReplaceFn: func() bool { return true }, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-15 23:13:44 +02:00
										 |  |  | 	TestFastRetries(t) | 
					
						
							| 
									
										
										
										
											2022-10-15 16:33:15 +02:00
										 |  |  | 	retryBackend := New(be, 10, nil, nil) | 
					
						
							| 
									
										
										
										
											2022-08-26 21:20:52 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	data := test.Random(23, 5*1024*1024+11241) | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	err := retryBackend.Save(context.TODO(), backend.Handle{}, backend.NewByteReader(data, be.Hasher())) | 
					
						
							| 
									
										
										
										
											2022-08-26 21:20:52 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if calledRemove { | 
					
						
							|  |  |  | 		t.Fatal("remove must not be called") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-24 10:25:40 -05:00
										 |  |  | func TestBackendListRetry(t *testing.T) { | 
					
						
							|  |  |  | 	const ( | 
					
						
							|  |  |  | 		ID1 = "id1" | 
					
						
							|  |  |  | 		ID2 = "id2" | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	retry := 0 | 
					
						
							|  |  |  | 	be := &mock.Backend{ | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 		ListFn: func(ctx context.Context, t backend.FileType, fn func(backend.FileInfo) error) error { | 
					
						
							| 
									
										
										
										
											2018-01-24 10:25:40 -05:00
										 |  |  | 			// fail during first retry, succeed during second | 
					
						
							|  |  |  | 			retry++ | 
					
						
							|  |  |  | 			if retry == 1 { | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 				_ = fn(backend.FileInfo{Name: ID1}) | 
					
						
							| 
									
										
										
										
											2018-01-24 10:25:40 -05:00
										 |  |  | 				return errors.New("test list error") | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 			_ = fn(backend.FileInfo{Name: ID1}) | 
					
						
							|  |  |  | 			_ = fn(backend.FileInfo{Name: ID2}) | 
					
						
							| 
									
										
										
										
											2018-01-24 10:25:40 -05:00
										 |  |  | 			return nil | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-15 23:13:44 +02:00
										 |  |  | 	TestFastRetries(t) | 
					
						
							| 
									
										
										
										
											2022-10-15 16:33:15 +02:00
										 |  |  | 	retryBackend := New(be, 10, nil, nil) | 
					
						
							| 
									
										
										
										
											2018-01-24 10:25:40 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var listed []string | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	err := retryBackend.List(context.TODO(), backend.PackFile, func(fi backend.FileInfo) error { | 
					
						
							| 
									
										
										
										
											2018-01-24 10:25:40 -05:00
										 |  |  | 		listed = append(listed, fi.Name) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	test.OK(t, err)                            // assert overall success | 
					
						
							|  |  |  | 	test.Equals(t, 2, retry)                   // assert retried once | 
					
						
							|  |  |  | 	test.Equals(t, []string{ID1, ID2}, listed) // assert no duplicate files | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:23:42 +01:00
										 |  |  | func TestBackendListRetryErrorFn(t *testing.T) { | 
					
						
							|  |  |  | 	var names = []string{"id1", "id2", "foo", "bar"} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	be := &mock.Backend{ | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 		ListFn: func(ctx context.Context, tpe backend.FileType, fn func(backend.FileInfo) error) error { | 
					
						
							| 
									
										
										
										
											2018-02-24 13:23:42 +01:00
										 |  |  | 			t.Logf("List called for %v", tpe) | 
					
						
							|  |  |  | 			for _, name := range names { | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 				err := fn(backend.FileInfo{Name: name}) | 
					
						
							| 
									
										
										
										
											2018-02-24 13:23:42 +01:00
										 |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-15 23:13:44 +02:00
										 |  |  | 	TestFastRetries(t) | 
					
						
							| 
									
										
										
										
											2022-10-15 16:33:15 +02:00
										 |  |  | 	retryBackend := New(be, 10, nil, nil) | 
					
						
							| 
									
										
										
										
											2018-02-24 13:23:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var ErrTest = errors.New("test error") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var listed []string | 
					
						
							|  |  |  | 	run := 0 | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	err := retryBackend.List(context.TODO(), backend.PackFile, func(fi backend.FileInfo) error { | 
					
						
							| 
									
										
										
										
											2018-02-24 13:23:42 +01:00
										 |  |  | 		t.Logf("fn called for %v", fi.Name) | 
					
						
							|  |  |  | 		run++ | 
					
						
							|  |  |  | 		// return an error for the third item in the list | 
					
						
							|  |  |  | 		if run == 3 { | 
					
						
							|  |  |  | 			t.Log("returning an error") | 
					
						
							|  |  |  | 			return ErrTest | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		listed = append(listed, fi.Name) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err != ErrTest { | 
					
						
							|  |  |  | 		t.Fatalf("wrong error returned, want %v, got %v", ErrTest, err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// processing should stop after the error was returned, so run should be 3 | 
					
						
							|  |  |  | 	if run != 3 { | 
					
						
							|  |  |  | 		t.Fatalf("function was called %d times, wanted %v", run, 3) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	test.Equals(t, []string{"id1", "id2"}, listed) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestBackendListRetryErrorBackend(t *testing.T) { | 
					
						
							|  |  |  | 	var names = []string{"id1", "id2", "foo", "bar"} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var ErrBackendTest = errors.New("test error") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	retries := 0 | 
					
						
							|  |  |  | 	be := &mock.Backend{ | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 		ListFn: func(ctx context.Context, tpe backend.FileType, fn func(backend.FileInfo) error) error { | 
					
						
							| 
									
										
										
										
											2018-02-24 13:23:42 +01:00
										 |  |  | 			t.Logf("List called for %v, retries %v", tpe, retries) | 
					
						
							|  |  |  | 			retries++ | 
					
						
							|  |  |  | 			for i, name := range names { | 
					
						
							|  |  |  | 				if i == 2 { | 
					
						
							|  |  |  | 					return ErrBackendTest | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 				err := fn(backend.FileInfo{Name: name}) | 
					
						
							| 
									
										
										
										
											2018-02-24 13:23:42 +01:00
										 |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-15 23:13:44 +02:00
										 |  |  | 	TestFastRetries(t) | 
					
						
							| 
									
										
										
										
											2024-04-29 21:12:21 +02:00
										 |  |  | 	const maxElapsedTime = 10 * time.Millisecond | 
					
						
							|  |  |  | 	now := time.Now() | 
					
						
							|  |  |  | 	retryBackend := New(be, maxElapsedTime, nil, nil) | 
					
						
							| 
									
										
										
										
											2018-02-24 13:23:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var listed []string | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	err := retryBackend.List(context.TODO(), backend.PackFile, func(fi backend.FileInfo) error { | 
					
						
							| 
									
										
										
										
											2018-02-24 13:23:42 +01:00
										 |  |  | 		t.Logf("fn called for %v", fi.Name) | 
					
						
							|  |  |  | 		listed = append(listed, fi.Name) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err != ErrBackendTest { | 
					
						
							|  |  |  | 		t.Fatalf("wrong error returned, want %v, got %v", ErrBackendTest, err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-29 21:12:21 +02:00
										 |  |  | 	duration := time.Since(now) | 
					
						
							|  |  |  | 	if duration > 100*time.Millisecond { | 
					
						
							|  |  |  | 		t.Fatalf("list retries took %v, expected at most 10ms", duration) | 
					
						
							| 
									
										
										
										
											2018-02-24 13:23:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	test.Equals(t, names[:2], listed) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | // failingReader returns an error after reading limit number of bytes | 
					
						
							|  |  |  | type failingReader struct { | 
					
						
							|  |  |  | 	data  []byte | 
					
						
							|  |  |  | 	pos   int | 
					
						
							|  |  |  | 	limit int | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (r failingReader) Read(p []byte) (n int, err error) { | 
					
						
							|  |  |  | 	i := 0 | 
					
						
							|  |  |  | 	for ; i < len(p) && i+r.pos < r.limit; i++ { | 
					
						
							|  |  |  | 		p[i] = r.data[r.pos+i] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	r.pos += i | 
					
						
							|  |  |  | 	if r.pos >= r.limit { | 
					
						
							|  |  |  | 		return i, errors.Errorf("reader reached limit of %d", r.limit) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return i, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | func (r failingReader) Close() error { | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // closingReader adapts io.Reader to io.ReadCloser interface | 
					
						
							|  |  |  | type closingReader struct { | 
					
						
							|  |  |  | 	rd io.Reader | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (r closingReader) Read(p []byte) (n int, err error) { | 
					
						
							|  |  |  | 	return r.rd.Read(p) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | func (r closingReader) Close() error { | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestBackendLoadRetry(t *testing.T) { | 
					
						
							|  |  |  | 	data := test.Random(23, 1024) | 
					
						
							|  |  |  | 	limit := 100 | 
					
						
							|  |  |  | 	attempt := 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	be := mock.NewBackend() | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	be.OpenReaderFn = func(ctx context.Context, h backend.Handle, length int, offset int64) (io.ReadCloser, error) { | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 		// returns failing reader on first invocation, good reader on subsequent invocations | 
					
						
							|  |  |  | 		attempt++ | 
					
						
							|  |  |  | 		if attempt > 1 { | 
					
						
							|  |  |  | 			return closingReader{rd: bytes.NewReader(data)}, nil | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return failingReader{data: data, limit: limit}, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-15 23:13:44 +02:00
										 |  |  | 	TestFastRetries(t) | 
					
						
							| 
									
										
										
										
											2022-10-15 16:33:15 +02:00
										 |  |  | 	retryBackend := New(be, 10, nil, nil) | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var buf []byte | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	err := retryBackend.Load(context.TODO(), backend.Handle{}, 0, 0, func(rd io.Reader) (err error) { | 
					
						
							| 
									
										
										
										
											2022-12-02 19:36:43 +01:00
										 |  |  | 		buf, err = io.ReadAll(rd) | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 		return err | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	test.OK(t, err) | 
					
						
							|  |  |  | 	test.Equals(t, data, buf) | 
					
						
							|  |  |  | 	test.Equals(t, 2, attempt) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-12-28 20:47:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-18 12:51:39 -05:00
										 |  |  | func TestBackendLoadNotExists(t *testing.T) { | 
					
						
							|  |  |  | 	// load should not retry if the error matches IsNotExist | 
					
						
							|  |  |  | 	notFound := errors.New("not found") | 
					
						
							|  |  |  | 	attempt := 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	be := mock.NewBackend() | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	be.OpenReaderFn = func(ctx context.Context, h backend.Handle, length int, offset int64) (io.ReadCloser, error) { | 
					
						
							| 
									
										
										
										
											2023-10-18 12:51:39 -05:00
										 |  |  | 		attempt++ | 
					
						
							|  |  |  | 		if attempt > 1 { | 
					
						
							|  |  |  | 			t.Fail() | 
					
						
							|  |  |  | 			return nil, errors.New("must not retry") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return nil, notFound | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-05-11 20:25:04 +02:00
										 |  |  | 	be.IsPermanentErrorFn = func(err error) bool { | 
					
						
							| 
									
										
										
										
											2023-10-18 12:51:39 -05:00
										 |  |  | 		return errors.Is(err, notFound) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	TestFastRetries(t) | 
					
						
							|  |  |  | 	retryBackend := New(be, 10, nil, nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	err := retryBackend.Load(context.TODO(), backend.Handle{}, 0, 0, func(rd io.Reader) (err error) { | 
					
						
							| 
									
										
										
										
											2023-10-18 12:51:39 -05:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2024-05-11 20:25:04 +02:00
										 |  |  | 	test.Assert(t, be.IsPermanentErrorFn(err), "unexpected error %v", err) | 
					
						
							| 
									
										
										
										
											2023-10-18 12:51:39 -05:00
										 |  |  | 	test.Equals(t, 1, attempt) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-12 12:34:54 +02:00
										 |  |  | func TestBackendLoadCircuitBreaker(t *testing.T) { | 
					
						
							|  |  |  | 	// retry should not retry if the error matches IsPermanentError | 
					
						
							|  |  |  | 	notFound := errors.New("not found") | 
					
						
							|  |  |  | 	otherError := errors.New("something") | 
					
						
							|  |  |  | 	attempt := 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	be := mock.NewBackend() | 
					
						
							|  |  |  | 	be.IsPermanentErrorFn = func(err error) bool { | 
					
						
							|  |  |  | 		return errors.Is(err, notFound) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	be.OpenReaderFn = func(ctx context.Context, h backend.Handle, length int, offset int64) (io.ReadCloser, error) { | 
					
						
							|  |  |  | 		attempt++ | 
					
						
							|  |  |  | 		return nil, otherError | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	nilRd := func(rd io.Reader) (err error) { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	TestFastRetries(t) | 
					
						
							|  |  |  | 	retryBackend := New(be, 2, nil, nil) | 
					
						
							|  |  |  | 	// trip the circuit breaker for file "other" | 
					
						
							|  |  |  | 	err := retryBackend.Load(context.TODO(), backend.Handle{Name: "other"}, 0, 0, nilRd) | 
					
						
							|  |  |  | 	test.Equals(t, otherError, err, "unexpected error") | 
					
						
							| 
									
										
										
										
											2024-04-29 21:12:21 +02:00
										 |  |  | 	test.Equals(t, 2, attempt) | 
					
						
							| 
									
										
										
										
											2024-05-12 12:34:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	attempt = 0 | 
					
						
							|  |  |  | 	err = retryBackend.Load(context.TODO(), backend.Handle{Name: "other"}, 0, 0, nilRd) | 
					
						
							|  |  |  | 	test.Assert(t, strings.Contains(err.Error(), "circuit breaker open for file"), "expected circuit breaker error, got %v") | 
					
						
							|  |  |  | 	test.Equals(t, 0, attempt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// don't trip for permanent errors | 
					
						
							|  |  |  | 	be.OpenReaderFn = func(ctx context.Context, h backend.Handle, length int, offset int64) (io.ReadCloser, error) { | 
					
						
							|  |  |  | 		attempt++ | 
					
						
							|  |  |  | 		return nil, notFound | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	err = retryBackend.Load(context.TODO(), backend.Handle{Name: "notfound"}, 0, 0, nilRd) | 
					
						
							|  |  |  | 	test.Equals(t, notFound, err, "expected circuit breaker to only affect other file, got %v") | 
					
						
							|  |  |  | 	err = retryBackend.Load(context.TODO(), backend.Handle{Name: "notfound"}, 0, 0, nilRd) | 
					
						
							|  |  |  | 	test.Equals(t, notFound, err, "persistent error must not trigger circuit breaker, got %v") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// wait for circuit breaker to expire | 
					
						
							|  |  |  | 	time.Sleep(5 * time.Millisecond) | 
					
						
							|  |  |  | 	old := failedLoadExpiry | 
					
						
							|  |  |  | 	defer func() { | 
					
						
							|  |  |  | 		failedLoadExpiry = old | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 	failedLoadExpiry = 3 * time.Millisecond | 
					
						
							|  |  |  | 	err = retryBackend.Load(context.TODO(), backend.Handle{Name: "other"}, 0, 0, nilRd) | 
					
						
							|  |  |  | 	test.Equals(t, notFound, err, "expected circuit breaker to reset, got %v") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-03 11:42:48 +01:00
										 |  |  | func TestBackendStatNotExists(t *testing.T) { | 
					
						
							|  |  |  | 	// stat should not retry if the error matches IsNotExist | 
					
						
							|  |  |  | 	notFound := errors.New("not found") | 
					
						
							|  |  |  | 	attempt := 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	be := mock.NewBackend() | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	be.StatFn = func(ctx context.Context, h backend.Handle) (backend.FileInfo, error) { | 
					
						
							| 
									
										
										
										
											2022-12-03 11:42:48 +01:00
										 |  |  | 		attempt++ | 
					
						
							|  |  |  | 		if attempt > 1 { | 
					
						
							|  |  |  | 			t.Fail() | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 			return backend.FileInfo{}, errors.New("must not retry") | 
					
						
							| 
									
										
										
										
											2022-12-03 11:42:48 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 		return backend.FileInfo{}, notFound | 
					
						
							| 
									
										
										
										
											2022-12-03 11:42:48 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	be.IsNotExistFn = func(err error) bool { | 
					
						
							|  |  |  | 		return errors.Is(err, notFound) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	TestFastRetries(t) | 
					
						
							|  |  |  | 	retryBackend := New(be, 10, nil, nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	_, err := retryBackend.Stat(context.TODO(), backend.Handle{}) | 
					
						
							| 
									
										
										
										
											2022-12-03 11:42:48 +01:00
										 |  |  | 	test.Assert(t, be.IsNotExistFn(err), "unexpected error %v", err) | 
					
						
							|  |  |  | 	test.Equals(t, 1, attempt) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-11 20:25:04 +02:00
										 |  |  | func TestBackendRetryPermanent(t *testing.T) { | 
					
						
							|  |  |  | 	// retry should not retry if the error matches IsPermanentError | 
					
						
							|  |  |  | 	notFound := errors.New("not found") | 
					
						
							|  |  |  | 	attempt := 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	be := mock.NewBackend() | 
					
						
							|  |  |  | 	be.IsPermanentErrorFn = func(err error) bool { | 
					
						
							|  |  |  | 		return errors.Is(err, notFound) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	TestFastRetries(t) | 
					
						
							|  |  |  | 	retryBackend := New(be, 2, nil, nil) | 
					
						
							|  |  |  | 	err := retryBackend.retry(context.TODO(), "test", func() error { | 
					
						
							|  |  |  | 		attempt++ | 
					
						
							|  |  |  | 		return notFound | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	test.Assert(t, be.IsPermanentErrorFn(err), "unexpected error %v", err) | 
					
						
							|  |  |  | 	test.Equals(t, 1, attempt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	attempt = 0 | 
					
						
							|  |  |  | 	err = retryBackend.retry(context.TODO(), "test", func() error { | 
					
						
							|  |  |  | 		attempt++ | 
					
						
							|  |  |  | 		return errors.New("something") | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	test.Assert(t, !be.IsPermanentErrorFn(err), "error unexpectedly considered permanent %v", err) | 
					
						
							| 
									
										
										
										
											2024-04-29 21:12:21 +02:00
										 |  |  | 	test.Equals(t, 2, attempt) | 
					
						
							| 
									
										
										
										
											2024-05-11 20:25:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-28 20:47:11 +01:00
										 |  |  | func assertIsCanceled(t *testing.T, err error) { | 
					
						
							|  |  |  | 	test.Assert(t, err == context.Canceled, "got unexpected err %v", err) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestBackendCanceledContext(t *testing.T) { | 
					
						
							|  |  |  | 	// unimplemented mock backend functions return an error by default | 
					
						
							|  |  |  | 	// check that we received the expected context canceled error instead | 
					
						
							| 
									
										
										
										
											2022-10-15 23:13:44 +02:00
										 |  |  | 	TestFastRetries(t) | 
					
						
							| 
									
										
										
										
											2022-10-15 16:33:15 +02:00
										 |  |  | 	retryBackend := New(mock.NewBackend(), 2, nil, nil) | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	h := backend.Handle{Type: backend.PackFile, Name: restic.NewRandomID().String()} | 
					
						
							| 
									
										
										
										
											2020-12-28 20:47:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// create an already canceled context | 
					
						
							|  |  |  | 	ctx, cancel := context.WithCancel(context.Background()) | 
					
						
							|  |  |  | 	cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-03 11:28:10 +01:00
										 |  |  | 	_, err := retryBackend.Stat(ctx, h) | 
					
						
							| 
									
										
										
										
											2020-12-28 20:47:11 +01:00
										 |  |  | 	assertIsCanceled(t, err) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	err = retryBackend.Save(ctx, h, backend.NewByteReader([]byte{}, nil)) | 
					
						
							| 
									
										
										
										
											2020-12-28 20:47:11 +01:00
										 |  |  | 	assertIsCanceled(t, err) | 
					
						
							|  |  |  | 	err = retryBackend.Remove(ctx, h) | 
					
						
							|  |  |  | 	assertIsCanceled(t, err) | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	err = retryBackend.Load(ctx, backend.Handle{}, 0, 0, func(rd io.Reader) (err error) { | 
					
						
							| 
									
										
										
										
											2020-12-28 20:47:11 +01:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	assertIsCanceled(t, err) | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	err = retryBackend.List(ctx, backend.PackFile, func(backend.FileInfo) error { | 
					
						
							| 
									
										
										
										
											2020-12-28 20:47:11 +01:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	assertIsCanceled(t, err) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// don't test "Delete" as it is not used by normal code | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-03-21 19:05:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | func TestNotifyWithSuccessIsNotCalled(t *testing.T) { | 
					
						
							|  |  |  | 	operation := func() error { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	notify := func(error, time.Duration) { | 
					
						
							|  |  |  | 		t.Fatal("Notify should not have been called") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-21 19:39:01 +00:00
										 |  |  | 	success := func(retries int) { | 
					
						
							| 
									
										
										
										
											2020-03-21 19:05:36 +00:00
										 |  |  | 		t.Fatal("Success should not have been called") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err := retryNotifyErrorWithSuccess(operation, &backoff.ZeroBackOff{}, notify, success) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal("retry should not have returned an error") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestNotifyWithSuccessIsCalled(t *testing.T) { | 
					
						
							|  |  |  | 	operationCalled := 0 | 
					
						
							|  |  |  | 	operation := func() error { | 
					
						
							|  |  |  | 		operationCalled++ | 
					
						
							|  |  |  | 		if operationCalled <= 2 { | 
					
						
							|  |  |  | 			return errors.New("expected error in test") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	notifyCalled := 0 | 
					
						
							|  |  |  | 	notify := func(error, time.Duration) { | 
					
						
							|  |  |  | 		notifyCalled++ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	successCalled := 0 | 
					
						
							| 
									
										
										
										
											2020-03-21 19:39:01 +00:00
										 |  |  | 	success := func(retries int) { | 
					
						
							| 
									
										
										
										
											2020-03-21 19:05:36 +00:00
										 |  |  | 		successCalled++ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err := retryNotifyErrorWithSuccess(operation, &backoff.ZeroBackOff{}, notify, success) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal("retry should not have returned an error") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if notifyCalled != 2 { | 
					
						
							|  |  |  | 		t.Fatalf("Notify should have been called 2 times, but was called %d times instead", notifyCalled) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if successCalled != 1 { | 
					
						
							|  |  |  | 		t.Fatalf("Success should have been called only once, but was called %d times instead", successCalled) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-04-29 21:07:17 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | func TestNotifyWithSuccessFinalError(t *testing.T) { | 
					
						
							|  |  |  | 	operation := func() error { | 
					
						
							|  |  |  | 		return errors.New("expected error in test") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	notifyCalled := 0 | 
					
						
							|  |  |  | 	notify := func(error, time.Duration) { | 
					
						
							|  |  |  | 		notifyCalled++ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	successCalled := 0 | 
					
						
							|  |  |  | 	success := func(retries int) { | 
					
						
							|  |  |  | 		successCalled++ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err := retryNotifyErrorWithSuccess(operation, backoff.WithMaxRetries(&backoff.ZeroBackOff{}, 5), notify, success) | 
					
						
							|  |  |  | 	test.Assert(t, err.Error() == "expected error in test", "wrong error message %v", err) | 
					
						
							|  |  |  | 	test.Equals(t, 6, notifyCalled, "notify should have been called 6 times") | 
					
						
							|  |  |  | 	test.Equals(t, 0, successCalled, "success should not have been called") | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-04-29 21:16:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | type testClock struct { | 
					
						
							|  |  |  | 	Time time.Time | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (c *testClock) Now() time.Time { | 
					
						
							|  |  |  | 	return c.Time | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestRetryAtLeastOnce(t *testing.T) { | 
					
						
							|  |  |  | 	expBackOff := backoff.NewExponentialBackOff() | 
					
						
							|  |  |  | 	expBackOff.InitialInterval = 500 * time.Millisecond | 
					
						
							|  |  |  | 	expBackOff.RandomizationFactor = 0 | 
					
						
							|  |  |  | 	expBackOff.MaxElapsedTime = 5 * time.Second | 
					
						
							|  |  |  | 	expBackOff.Multiplier = 2 // guarantee numerical stability | 
					
						
							|  |  |  | 	clock := &testClock{Time: time.Now()} | 
					
						
							|  |  |  | 	expBackOff.Clock = clock | 
					
						
							|  |  |  | 	expBackOff.Reset() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	retry := withRetryAtLeastOnce(expBackOff) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// expire backoff | 
					
						
							|  |  |  | 	clock.Time = clock.Time.Add(10 * time.Second) | 
					
						
							|  |  |  | 	delay := retry.NextBackOff() | 
					
						
							|  |  |  | 	test.Equals(t, expBackOff.InitialInterval, delay, "must retry at least once") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	delay = retry.NextBackOff() | 
					
						
							|  |  |  | 	test.Equals(t, expBackOff.Stop, delay, "must not retry more than once") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// test reset behavior | 
					
						
							|  |  |  | 	retry.Reset() | 
					
						
							|  |  |  | 	test.Equals(t, uint64(0), retry.numTries, "numTries should be reset to 0") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Verify that after reset, NextBackOff returns the initial interval again | 
					
						
							|  |  |  | 	delay = retry.NextBackOff() | 
					
						
							|  |  |  | 	test.Equals(t, expBackOff.InitialInterval, delay, "retries must work after reset") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	delay = retry.NextBackOff() | 
					
						
							|  |  |  | 	test.Equals(t, expBackOff.InitialInterval*time.Duration(expBackOff.Multiplier), delay, "retries must work after reset") | 
					
						
							|  |  |  | } |