| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | package repository | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2017-06-04 11:16:55 +02:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2022-04-29 23:16:16 +02:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2024-03-28 23:14:32 +01:00
										 |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | 	"testing" | 
					
						
							| 
									
										
										
										
											2016-07-31 17:45:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	"github.com/restic/restic/internal/backend" | 
					
						
							| 
									
										
										
										
											2017-07-23 14:21:03 +02:00
										 |  |  | 	"github.com/restic/restic/internal/backend/local" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/backend/mem" | 
					
						
							| 
									
										
										
										
											2023-01-14 16:06:35 +01:00
										 |  |  | 	"github.com/restic/restic/internal/backend/retry" | 
					
						
							| 
									
										
										
										
											2017-07-23 14:21:03 +02:00
										 |  |  | 	"github.com/restic/restic/internal/crypto" | 
					
						
							| 
									
										
										
										
											2017-07-24 17:42:25 +02:00
										 |  |  | 	"github.com/restic/restic/internal/restic" | 
					
						
							| 
									
										
										
										
											2017-07-23 14:21:03 +02:00
										 |  |  | 	"github.com/restic/restic/internal/test" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-31 17:45:22 +02:00
										 |  |  | 	"github.com/restic/chunker" | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-25 18:00:42 +01:00
										 |  |  | type logger interface { | 
					
						
							|  |  |  | 	Logf(format string, args ...interface{}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-28 23:14:32 +01:00
										 |  |  | var paramsOnce sync.Once | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 13:42:04 +02:00
										 |  |  | // TestUseLowSecurityKDFParameters configures low-security KDF parameters for testing. | 
					
						
							| 
									
										
										
										
											2017-03-25 18:00:42 +01:00
										 |  |  | func TestUseLowSecurityKDFParameters(t logger) { | 
					
						
							| 
									
										
										
										
											2016-08-21 13:42:04 +02:00
										 |  |  | 	t.Logf("using low-security KDF parameters for test") | 
					
						
							| 
									
										
										
										
											2024-03-28 23:14:32 +01:00
										 |  |  | 	paramsOnce.Do(func() { | 
					
						
							|  |  |  | 		params = &crypto.Params{ | 
					
						
							|  |  |  | 			N: 128, | 
					
						
							|  |  |  | 			R: 1, | 
					
						
							|  |  |  | 			P: 1, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2016-08-21 13:42:04 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | // TestBackend returns a fully configured in-memory backend. | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | func TestBackend(_ testing.TB) backend.Backend { | 
					
						
							| 
									
										
										
										
											2022-12-11 10:41:22 +01:00
										 |  |  | 	return mem.New() | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-28 23:14:32 +01:00
										 |  |  | const testChunkerPol = chunker.Pol(0x3DA3358B4DC173) | 
					
						
							| 
									
										
										
										
											2016-07-31 17:45:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | // TestRepositoryWithBackend returns a repository initialized with a test | 
					
						
							| 
									
										
										
										
											2016-07-31 17:45:22 +02:00
										 |  |  | // password. If be is nil, an in-memory backend is used. A constant polynomial | 
					
						
							| 
									
										
										
										
											2016-08-21 13:42:04 +02:00
										 |  |  | // is used for the chunker and low-security test parameters. | 
					
						
							| 
									
										
										
										
											2024-02-03 17:47:36 +01:00
										 |  |  | func TestRepositoryWithBackend(t testing.TB, be backend.Backend, version uint, opts Options) restic.Repository { | 
					
						
							| 
									
										
										
										
											2020-02-17 13:07:55 +01:00
										 |  |  | 	t.Helper() | 
					
						
							| 
									
										
										
										
											2016-08-21 13:42:04 +02:00
										 |  |  | 	TestUseLowSecurityKDFParameters(t) | 
					
						
							| 
									
										
										
										
											2018-03-11 21:42:39 +01:00
										 |  |  | 	restic.TestDisableCheckPolynomial(t) | 
					
						
							| 
									
										
										
										
											2016-08-21 13:42:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | 	if be == nil { | 
					
						
							| 
									
										
										
										
											2022-12-11 10:41:22 +01:00
										 |  |  | 		be = TestBackend(t) | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-03 17:47:36 +01:00
										 |  |  | 	repo, err := New(be, opts) | 
					
						
							| 
									
										
										
										
											2022-07-02 23:30:26 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("TestRepository(): new repo failed: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-14 11:49:33 +02:00
										 |  |  | 	if version == 0 { | 
					
						
							|  |  |  | 		version = restic.StableRepoVersion | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	pol := testChunkerPol | 
					
						
							|  |  |  | 	err = repo.Init(context.TODO(), version, test.TestPassword, &pol) | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2016-07-31 17:45:22 +02:00
										 |  |  | 		t.Fatalf("TestRepository(): initialize repo failed: %v", err) | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-11 10:41:22 +01:00
										 |  |  | 	return repo | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // TestRepository returns a repository initialized with a test password on an | 
					
						
							|  |  |  | // in-memory backend. When the environment variable RESTIC_TEST_REPO is set to | 
					
						
							|  |  |  | // a non-existing directory, a local backend is created there and this is used | 
					
						
							| 
									
										
										
										
											2016-07-31 17:45:22 +02:00
										 |  |  | // instead. The directory is not removed, but left there for inspection. | 
					
						
							| 
									
										
										
										
											2022-12-11 10:41:22 +01:00
										 |  |  | func TestRepository(t testing.TB) restic.Repository { | 
					
						
							| 
									
										
										
										
											2022-04-29 23:16:16 +02:00
										 |  |  | 	t.Helper() | 
					
						
							|  |  |  | 	return TestRepositoryWithVersion(t, 0) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-11 10:41:22 +01:00
										 |  |  | func TestRepositoryWithVersion(t testing.TB, version uint) restic.Repository { | 
					
						
							| 
									
										
										
										
											2020-02-17 13:07:55 +01:00
										 |  |  | 	t.Helper() | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | 	dir := os.Getenv("RESTIC_TEST_REPO") | 
					
						
							| 
									
										
										
										
											2024-02-03 17:47:36 +01:00
										 |  |  | 	opts := Options{} | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | 	if dir != "" { | 
					
						
							|  |  |  | 		_, err := os.Stat(dir) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2020-09-19 22:01:32 +02:00
										 |  |  | 			be, err := local.Create(context.TODO(), local.Config{Path: dir}) | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				t.Fatalf("error creating local backend at %v: %v", dir, err) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2024-02-03 17:47:36 +01:00
										 |  |  | 			return TestRepositoryWithBackend(t, be, version, opts) | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if err == nil { | 
					
						
							|  |  |  | 			t.Logf("directory at %v already exists, using mem backend", dir) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-03 17:47:36 +01:00
										 |  |  | 	return TestRepositoryWithBackend(t, nil, version, opts) | 
					
						
							| 
									
										
										
										
											2016-04-10 16:52:15 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-04 14:29:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-24 21:45:24 +01:00
										 |  |  | func TestFromFixture(t testing.TB, repoFixture string) (restic.Repository, func()) { | 
					
						
							|  |  |  | 	repodir, cleanup := test.Env(t, repoFixture) | 
					
						
							|  |  |  | 	repo := TestOpenLocal(t, repodir) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return repo, cleanup | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-04 14:29:04 +02:00
										 |  |  | // TestOpenLocal opens a local repository. | 
					
						
							| 
									
										
										
										
											2024-02-24 21:45:24 +01:00
										 |  |  | func TestOpenLocal(t testing.TB, dir string) restic.Repository { | 
					
						
							| 
									
										
										
										
											2023-10-01 11:40:12 +02:00
										 |  |  | 	var be backend.Backend | 
					
						
							| 
									
										
										
										
											2021-08-07 19:50:00 +02:00
										 |  |  | 	be, err := local.Open(context.TODO(), local.Config{Path: dir, Connections: 2}) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:29:04 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-14 16:06:35 +01:00
										 |  |  | 	be = retry.New(be, 3, nil, nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-24 21:45:24 +01:00
										 |  |  | 	return TestOpenBackend(t, be) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestOpenBackend(t testing.TB, be backend.Backend) restic.Repository { | 
					
						
							| 
									
										
										
										
											2022-07-02 23:30:26 +02:00
										 |  |  | 	repo, err := New(be, Options{}) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-11-25 09:10:45 -05:00
										 |  |  | 	err = repo.SearchKey(context.TODO(), test.TestPassword, 10, "") | 
					
						
							| 
									
										
										
										
											2016-09-04 14:29:04 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return repo | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-04-29 23:16:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | type VersionedTest func(t *testing.T, version uint) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestAllVersions(t *testing.T, test VersionedTest) { | 
					
						
							|  |  |  | 	for version := restic.MinRepoVersion; version <= restic.MaxRepoVersion; version++ { | 
					
						
							|  |  |  | 		t.Run(fmt.Sprintf("v%d", version), func(t *testing.T) { | 
					
						
							|  |  |  | 			test(t, uint(version)) | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type VersionedBenchmark func(b *testing.B, version uint) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func BenchmarkAllVersions(b *testing.B, bench VersionedBenchmark) { | 
					
						
							|  |  |  | 	for version := restic.MinRepoVersion; version <= restic.MaxRepoVersion; version++ { | 
					
						
							|  |  |  | 		b.Run(fmt.Sprintf("v%d", version), func(b *testing.B) { | 
					
						
							|  |  |  | 			bench(b, uint(version)) | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |