| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | package test | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 	"math/rand" | 
					
						
							| 
									
										
										
										
											2017-01-25 17:07:36 +01:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 	"reflect" | 
					
						
							|  |  |  | 	"sort" | 
					
						
							| 
									
										
										
										
											2017-01-22 12:32:20 +01:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 	"testing" | 
					
						
							| 
									
										
										
										
											2017-05-28 10:16:29 +02:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-23 14:21:03 +02:00
										 |  |  | 	"github.com/restic/restic/internal/errors" | 
					
						
							| 
									
										
										
										
											2017-07-24 17:42:25 +02:00
										 |  |  | 	"github.com/restic/restic/internal/restic" | 
					
						
							| 
									
										
										
										
											2016-08-29 19:18:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-23 14:21:03 +02:00
										 |  |  | 	"github.com/restic/restic/internal/test" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/backend" | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-28 10:16:29 +02:00
										 |  |  | func seedRand(t testing.TB) { | 
					
						
							|  |  |  | 	seed := time.Now().UnixNano() | 
					
						
							|  |  |  | 	rand.Seed(seed) | 
					
						
							|  |  |  | 	t.Logf("rand initialized with seed %d", seed) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 12:56:10 +02:00
										 |  |  | // TestCreateWithConfig tests that creating a backend in a location which already | 
					
						
							| 
									
										
										
										
											2016-01-23 18:07:15 +01:00
										 |  |  | // has a config file fails. | 
					
						
							| 
									
										
										
										
											2017-05-14 12:56:10 +02:00
										 |  |  | func (s *Suite) TestCreateWithConfig(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2017-05-01 22:23:46 +02:00
										 |  |  | 	b := s.open(t) | 
					
						
							|  |  |  | 	defer s.close(t, b) | 
					
						
							| 
									
										
										
										
											2016-01-23 18:07:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 12:50:20 +02:00
										 |  |  | 	// remove a config if present | 
					
						
							|  |  |  | 	cfgHandle := restic.Handle{Type: restic.ConfigFile} | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	cfgPresent, err := b.Test(context.TODO(), cfgHandle) | 
					
						
							| 
									
										
										
										
											2017-05-14 12:50:20 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("unable to test for config: %+v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if cfgPresent { | 
					
						
							|  |  |  | 		remove(t, b, cfgHandle) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-23 18:07:15 +01:00
										 |  |  | 	// save a config | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | 	store(t, b, restic.ConfigFile, []byte("test config")) | 
					
						
							| 
									
										
										
										
											2016-01-23 18:07:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// now create the backend again, this must fail | 
					
						
							| 
									
										
										
										
											2017-05-14 12:50:20 +02:00
										 |  |  | 	_, err = s.Create(s.Config) | 
					
						
							| 
									
										
										
										
											2016-01-23 18:07:15 +01:00
										 |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		t.Fatalf("expected error not found for creating a backend with an existing config file") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// remove config | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	err = b.Remove(context.TODO(), restic.Handle{Type: restic.ConfigFile, Name: ""}) | 
					
						
							| 
									
										
										
										
											2016-01-23 18:07:15 +01:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 		t.Fatalf("unexpected error removing config: %+v", err) | 
					
						
							| 
									
										
										
										
											2016-01-23 18:07:15 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 12:56:10 +02:00
										 |  |  | // TestLocation tests that a location string is returned. | 
					
						
							|  |  |  | func (s *Suite) TestLocation(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2017-05-01 22:23:46 +02:00
										 |  |  | 	b := s.open(t) | 
					
						
							|  |  |  | 	defer s.close(t, b) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	l := b.Location() | 
					
						
							|  |  |  | 	if l == "" { | 
					
						
							|  |  |  | 		t.Fatalf("invalid location string %q", l) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 12:56:10 +02:00
										 |  |  | // TestConfig saves and loads a config from the backend. | 
					
						
							|  |  |  | func (s *Suite) TestConfig(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2017-05-01 22:23:46 +02:00
										 |  |  | 	b := s.open(t) | 
					
						
							|  |  |  | 	defer s.close(t, b) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var testString = "Config" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// create config and read it back | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	_, err := backend.LoadAll(context.TODO(), b, restic.Handle{Type: restic.ConfigFile}) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		t.Fatalf("did not get expected error for non-existing config") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	err = b.Save(context.TODO(), restic.Handle{Type: restic.ConfigFile}, strings.NewReader(testString)) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 		t.Fatalf("Save() error: %+v", err) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// try accessing the config with different names, should all return the | 
					
						
							|  |  |  | 	// same config | 
					
						
							|  |  |  | 	for _, name := range []string{"", "foo", "bar", "0000000000000000000000000000000000000000000000000000000000000000"} { | 
					
						
							| 
									
										
										
										
											2016-09-01 21:19:30 +02:00
										 |  |  | 		h := restic.Handle{Type: restic.ConfigFile, Name: name} | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		buf, err := backend.LoadAll(context.TODO(), b, h) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 			t.Fatalf("unable to read config with name %q: %+v", name, err) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if string(buf) != testString { | 
					
						
							|  |  |  | 			t.Fatalf("wrong data returned, want %q, got %q", testString, string(buf)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-14 12:50:20 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// remove the config | 
					
						
							|  |  |  | 	remove(t, b, restic.Handle{Type: restic.ConfigFile}) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 12:56:10 +02:00
										 |  |  | // TestLoad tests the backend's Load function. | 
					
						
							|  |  |  | func (s *Suite) TestLoad(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2017-05-28 10:16:29 +02:00
										 |  |  | 	seedRand(t) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 22:23:46 +02:00
										 |  |  | 	b := s.open(t) | 
					
						
							|  |  |  | 	defer s.close(t, b) | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	rd, err := b.Load(context.TODO(), restic.Handle{}, 0, 0) | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 	if err == nil { | 
					
						
							| 
									
										
										
										
											2017-01-23 18:11:10 +01:00
										 |  |  | 		t.Fatalf("Load() did not return an error for invalid handle") | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-28 13:05:35 +02:00
										 |  |  | 	if rd != nil { | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		_ = rd.Close() | 
					
						
							| 
									
										
										
										
											2017-05-28 13:05:35 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-28 13:05:35 +02:00
										 |  |  | 	err = testLoad(b, restic.Handle{Type: restic.DataFile, Name: "foobar"}, 0, 0) | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 	if err == nil { | 
					
						
							| 
									
										
										
										
											2017-01-23 18:11:10 +01:00
										 |  |  | 		t.Fatalf("Load() did not return an error for non-existing blob") | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	length := rand.Intn(1<<24) + 2000 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	data := test.Random(23, length) | 
					
						
							|  |  |  | 	id := restic.Hash(data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	handle := restic.Handle{Type: restic.DataFile, Name: id.String()} | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	err = b.Save(context.TODO(), handle, bytes.NewReader(data)) | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 		t.Fatalf("Save() error: %+v", err) | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-28 12:32:53 +02:00
										 |  |  | 	t.Logf("saved %d bytes as %v", length, handle) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	rd, err = b.Load(context.TODO(), handle, 100, -1) | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 	if err == nil { | 
					
						
							| 
									
										
										
										
											2017-01-23 18:11:10 +01:00
										 |  |  | 		t.Fatalf("Load() returned no error for negative offset!") | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if rd != nil { | 
					
						
							| 
									
										
										
										
											2017-01-23 18:11:10 +01:00
										 |  |  | 		t.Fatalf("Load() returned a non-nil reader for negative offset!") | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 12:42:10 +02:00
										 |  |  | 	loadTests := 50 | 
					
						
							| 
									
										
										
										
											2017-05-01 22:23:46 +02:00
										 |  |  | 	if s.MinimalData { | 
					
						
							| 
									
										
										
										
											2017-05-01 12:42:10 +02:00
										 |  |  | 		loadTests = 10 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for i := 0; i < loadTests; i++ { | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 		l := rand.Intn(length + 2000) | 
					
						
							|  |  |  | 		o := rand.Intn(length + 2000) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		d := data | 
					
						
							|  |  |  | 		if o < len(d) { | 
					
						
							|  |  |  | 			d = d[o:] | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2017-05-28 12:32:42 +02:00
										 |  |  | 			t.Logf("offset == length, skipping test") | 
					
						
							|  |  |  | 			continue | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		getlen := l | 
					
						
							|  |  |  | 		if l >= len(d) && rand.Float32() >= 0.5 { | 
					
						
							|  |  |  | 			getlen = 0 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if l > 0 && l < len(d) { | 
					
						
							|  |  |  | 			d = d[:l] | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		rd, err := b.Load(context.TODO(), handle, getlen, int64(o)) | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2017-05-28 12:32:53 +02:00
										 |  |  | 			t.Logf("Load, l %v, o %v, len(d) %v, getlen %v", l, o, len(d), getlen) | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 			t.Errorf("Load(%d, %d) returned unexpected error: %+v", l, o, err) | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		buf, err := ioutil.ReadAll(rd) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2017-05-28 12:32:53 +02:00
										 |  |  | 			t.Logf("Load, l %v, o %v, len(d) %v, getlen %v", l, o, len(d), getlen) | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 			t.Errorf("Load(%d, %d) ReadAll() returned unexpected error: %+v", l, o, err) | 
					
						
							| 
									
										
										
										
											2017-05-01 12:45:53 +02:00
										 |  |  | 			if err = rd.Close(); err != nil { | 
					
						
							| 
									
										
										
										
											2017-05-17 01:34:33 +02:00
										 |  |  | 				t.Errorf("Load(%d, %d) rd.Close() returned error: %+v", l, o, err) | 
					
						
							| 
									
										
										
										
											2017-05-01 12:45:53 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-26 20:47:15 +02:00
										 |  |  | 		if l == 0 && len(buf) != len(d) { | 
					
						
							| 
									
										
										
										
											2017-05-28 12:32:53 +02:00
										 |  |  | 			t.Logf("Load, l %v, o %v, len(d) %v, getlen %v", l, o, len(d), getlen) | 
					
						
							| 
									
										
										
										
											2017-04-26 20:47:15 +02:00
										 |  |  | 			t.Errorf("Load(%d, %d) wrong number of bytes read: want %d, got %d", l, o, len(d), len(buf)) | 
					
						
							| 
									
										
										
										
											2017-05-01 12:45:53 +02:00
										 |  |  | 			if err = rd.Close(); err != nil { | 
					
						
							| 
									
										
										
										
											2017-05-17 01:34:33 +02:00
										 |  |  | 				t.Errorf("Load(%d, %d) rd.Close() returned error: %+v", l, o, err) | 
					
						
							| 
									
										
										
										
											2017-05-01 12:45:53 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-04-26 20:47:15 +02:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if l > 0 && l <= len(d) && len(buf) != l { | 
					
						
							| 
									
										
										
										
											2017-05-28 12:32:53 +02:00
										 |  |  | 			t.Logf("Load, l %v, o %v, len(d) %v, getlen %v", l, o, len(d), getlen) | 
					
						
							| 
									
										
										
										
											2017-01-23 18:11:10 +01:00
										 |  |  | 			t.Errorf("Load(%d, %d) wrong number of bytes read: want %d, got %d", l, o, l, len(buf)) | 
					
						
							| 
									
										
										
										
											2017-05-01 12:45:53 +02:00
										 |  |  | 			if err = rd.Close(); err != nil { | 
					
						
							| 
									
										
										
										
											2017-05-17 01:34:33 +02:00
										 |  |  | 				t.Errorf("Load(%d, %d) rd.Close() returned error: %+v", l, o, err) | 
					
						
							| 
									
										
										
										
											2017-05-01 12:45:53 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if l > len(d) && len(buf) != len(d) { | 
					
						
							| 
									
										
										
										
											2017-05-28 12:32:53 +02:00
										 |  |  | 			t.Logf("Load, l %v, o %v, len(d) %v, getlen %v", l, o, len(d), getlen) | 
					
						
							| 
									
										
										
										
											2017-01-23 18:11:10 +01:00
										 |  |  | 			t.Errorf("Load(%d, %d) wrong number of bytes read for overlong read: want %d, got %d", l, o, l, len(buf)) | 
					
						
							| 
									
										
										
										
											2017-05-01 12:45:53 +02:00
										 |  |  | 			if err = rd.Close(); err != nil { | 
					
						
							| 
									
										
										
										
											2017-05-17 01:34:33 +02:00
										 |  |  | 				t.Errorf("Load(%d, %d) rd.Close() returned error: %+v", l, o, err) | 
					
						
							| 
									
										
										
										
											2017-05-01 12:45:53 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if !bytes.Equal(buf, d) { | 
					
						
							| 
									
										
										
										
											2017-05-28 12:32:53 +02:00
										 |  |  | 			t.Logf("Load, l %v, o %v, len(d) %v, getlen %v", l, o, len(d), getlen) | 
					
						
							| 
									
										
										
										
											2017-01-23 18:11:10 +01:00
										 |  |  | 			t.Errorf("Load(%d, %d) returned wrong bytes", l, o) | 
					
						
							| 
									
										
										
										
											2017-05-01 12:45:53 +02:00
										 |  |  | 			if err = rd.Close(); err != nil { | 
					
						
							| 
									
										
										
										
											2017-05-17 01:34:33 +02:00
										 |  |  | 				t.Errorf("Load(%d, %d) rd.Close() returned error: %+v", l, o, err) | 
					
						
							| 
									
										
										
										
											2017-05-01 12:45:53 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		err = rd.Close() | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2017-05-28 12:32:53 +02:00
										 |  |  | 			t.Logf("Load, l %v, o %v, len(d) %v, getlen %v", l, o, len(d), getlen) | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 			t.Errorf("Load(%d, %d) rd.Close() returned unexpected error: %+v", l, o, err) | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	test.OK(t, b.Remove(context.TODO(), handle)) | 
					
						
							| 
									
										
										
										
											2017-01-22 22:01:12 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | // TestList makes sure that the backend implements List() pagination correctly. | 
					
						
							| 
									
										
										
										
											2017-09-17 11:09:09 +02:00
										 |  |  | func (s *Suite) TestList(t *testing.T) { | 
					
						
							|  |  |  | 	seedRand(t) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 	numTestFiles := rand.Intn(20) + 20 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-17 11:09:09 +02:00
										 |  |  | 	b := s.open(t) | 
					
						
							|  |  |  | 	defer s.close(t, b) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-27 13:06:15 +00:00
										 |  |  | 	// Check that the backend is empty to start with | 
					
						
							|  |  |  | 	var found []string | 
					
						
							|  |  |  | 	err := b.List(context.TODO(), restic.DataFile, func(fi restic.FileInfo) error { | 
					
						
							|  |  |  | 		found = append(found, fi.Name) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("List returned error %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if found != nil { | 
					
						
							|  |  |  | 		t.Fatalf("backend not empty at start of test - contains: %v", found) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 	list1 := make(map[restic.ID]int64) | 
					
						
							| 
									
										
										
										
											2017-09-17 11:09:09 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for i := 0; i < numTestFiles; i++ { | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 		data := test.Random(rand.Int(), rand.Intn(100)+55) | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 		id := restic.Hash(data) | 
					
						
							|  |  |  | 		h := restic.Handle{Type: restic.DataFile, Name: id.String()} | 
					
						
							|  |  |  | 		err := b.Save(context.TODO(), h, bytes.NewReader(data)) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 		list1[id] = int64(len(data)) | 
					
						
							| 
									
										
										
										
											2017-09-17 11:36:45 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 	t.Logf("wrote %v files", len(list1)) | 
					
						
							| 
									
										
										
										
											2017-09-17 11:36:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 	var tests = []struct { | 
					
						
							|  |  |  | 		maxItems int | 
					
						
							|  |  |  | 	}{ | 
					
						
							| 
									
										
										
										
											2017-09-18 13:18:42 +02:00
										 |  |  | 		{11}, {23}, {numTestFiles}, {numTestFiles + 10}, {numTestFiles + 1123}, | 
					
						
							| 
									
										
										
										
											2017-09-17 11:36:45 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 	for _, test := range tests { | 
					
						
							|  |  |  | 		t.Run(fmt.Sprintf("max-%v", test.maxItems), func(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 			list2 := make(map[restic.ID]int64) | 
					
						
							| 
									
										
										
										
											2017-09-17 11:36:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 			type setter interface { | 
					
						
							|  |  |  | 				SetListMaxItems(int) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-09-17 11:09:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 			if s, ok := b.(setter); ok { | 
					
						
							|  |  |  | 				t.Logf("setting max list items to %d", test.maxItems) | 
					
						
							|  |  |  | 				s.SetListMaxItems(test.maxItems) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-09-17 11:09:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:07 +01:00
										 |  |  | 			err := b.List(context.TODO(), restic.DataFile, func(fi restic.FileInfo) error { | 
					
						
							|  |  |  | 				id, err := restic.ParseID(fi.Name) | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					t.Fatal(err) | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 				list2[id] = fi.Size | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:07 +01:00
										 |  |  | 				return nil | 
					
						
							|  |  |  | 			}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				t.Fatalf("List returned error %v", err) | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-09-17 11:09:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 			t.Logf("loaded %v IDs from backend", len(list2)) | 
					
						
							| 
									
										
										
										
											2017-09-17 11:09:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 			for id, size := range list1 { | 
					
						
							|  |  |  | 				size2, ok := list2[id] | 
					
						
							|  |  |  | 				if !ok { | 
					
						
							|  |  |  | 					t.Errorf("id %v not returned by List()", id.Str()) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if size != size2 { | 
					
						
							|  |  |  | 					t.Errorf("wrong size for id %v returned: want %v, got %v", id.Str(), size, size2) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			for id := range list2 { | 
					
						
							|  |  |  | 				_, ok := list1[id] | 
					
						
							|  |  |  | 				if !ok { | 
					
						
							|  |  |  | 					t.Errorf("extra id %v returned by List()", id.Str()) | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2017-09-17 11:09:09 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-18 12:01:54 +02:00
										 |  |  | 	t.Logf("remove %d files", numTestFiles) | 
					
						
							| 
									
										
										
										
											2017-09-18 13:18:42 +02:00
										 |  |  | 	handles := make([]restic.Handle, 0, len(list1)) | 
					
						
							| 
									
										
										
										
											2017-09-17 11:09:09 +02:00
										 |  |  | 	for id := range list1 { | 
					
						
							| 
									
										
										
										
											2017-09-18 13:18:42 +02:00
										 |  |  | 		handles = append(handles, restic.Handle{Type: restic.DataFile, Name: id.String()}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-27 13:06:15 +00:00
										 |  |  | 	err = s.delayedRemove(t, b, handles...) | 
					
						
							| 
									
										
										
										
											2017-09-18 13:18:42 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							| 
									
										
										
										
											2017-09-17 11:09:09 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:50 +01:00
										 |  |  | // TestListCancel tests that the context is respected and the error is returned by List. | 
					
						
							|  |  |  | func (s *Suite) TestListCancel(t *testing.T) { | 
					
						
							|  |  |  | 	seedRand(t) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	numTestFiles := 5 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	b := s.open(t) | 
					
						
							|  |  |  | 	defer s.close(t, b) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	testFiles := make([]restic.Handle, 0, numTestFiles) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for i := 0; i < numTestFiles; i++ { | 
					
						
							|  |  |  | 		data := []byte(fmt.Sprintf("random test blob %v", i)) | 
					
						
							|  |  |  | 		id := restic.Hash(data) | 
					
						
							|  |  |  | 		h := restic.Handle{Type: restic.DataFile, Name: id.String()} | 
					
						
							|  |  |  | 		err := b.Save(context.TODO(), h, bytes.NewReader(data)) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		testFiles = append(testFiles, h) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t.Run("Cancelled", func(t *testing.T) { | 
					
						
							|  |  |  | 		ctx, cancel := context.WithCancel(context.TODO()) | 
					
						
							|  |  |  | 		cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// pass in a cancelled context | 
					
						
							|  |  |  | 		err := b.List(ctx, restic.DataFile, func(fi restic.FileInfo) error { | 
					
						
							|  |  |  | 			t.Errorf("got FileInfo %v for cancelled context", fi) | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 		if errors.Cause(err) != context.Canceled { | 
					
						
							|  |  |  | 			t.Fatalf("expected error not found, want %v, got %v", context.Canceled, errors.Cause(err)) | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:50 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t.Run("First", func(t *testing.T) { | 
					
						
							|  |  |  | 		ctx, cancel := context.WithCancel(context.TODO()) | 
					
						
							|  |  |  | 		defer cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		i := 0 | 
					
						
							|  |  |  | 		err := b.List(ctx, restic.DataFile, func(fi restic.FileInfo) error { | 
					
						
							|  |  |  | 			i++ | 
					
						
							|  |  |  | 			// cancel the context on the first file | 
					
						
							|  |  |  | 			if i == 1 { | 
					
						
							|  |  |  | 				cancel() | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-25 08:53:50 -05:00
										 |  |  | 		if errors.Cause(err) != context.Canceled { | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:50 +01:00
										 |  |  | 			t.Fatalf("expected error not found, want %v, got %v", context.Canceled, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if i != 1 { | 
					
						
							|  |  |  | 			t.Fatalf("wrong number of files returned by List, want %v, got %v", 1, i) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t.Run("Last", func(t *testing.T) { | 
					
						
							|  |  |  | 		ctx, cancel := context.WithCancel(context.TODO()) | 
					
						
							|  |  |  | 		defer cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		i := 0 | 
					
						
							|  |  |  | 		err := b.List(ctx, restic.DataFile, func(fi restic.FileInfo) error { | 
					
						
							|  |  |  | 			// cancel the context at the last file | 
					
						
							|  |  |  | 			i++ | 
					
						
							|  |  |  | 			if i == numTestFiles { | 
					
						
							|  |  |  | 				cancel() | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-25 08:53:50 -05:00
										 |  |  | 		if errors.Cause(err) != context.Canceled { | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:50 +01:00
										 |  |  | 			t.Fatalf("expected error not found, want %v, got %v", context.Canceled, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if i != numTestFiles { | 
					
						
							|  |  |  | 			t.Fatalf("wrong number of files returned by List, want %v, got %v", numTestFiles, i) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t.Run("Timeout", func(t *testing.T) { | 
					
						
							|  |  |  | 		ctx, cancel := context.WithCancel(context.TODO()) | 
					
						
							|  |  |  | 		defer cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 21:41:21 +01:00
										 |  |  | 		// rather large timeout, let's try to get at least one item | 
					
						
							|  |  |  | 		timeout := time.Second | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:50 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		ctxTimeout, _ := context.WithTimeout(ctx, timeout) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		i := 0 | 
					
						
							| 
									
										
										
										
											2018-01-23 21:41:21 +01:00
										 |  |  | 		// pass in a context with a timeout | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:50 +01:00
										 |  |  | 		err := b.List(ctxTimeout, restic.DataFile, func(fi restic.FileInfo) error { | 
					
						
							|  |  |  | 			i++ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// wait until the context is cancelled | 
					
						
							| 
									
										
										
										
											2018-01-23 21:41:21 +01:00
										 |  |  | 			<-ctxTimeout.Done() | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:50 +01:00
										 |  |  | 			return nil | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-25 08:53:50 -05:00
										 |  |  | 		if errors.Cause(err) != context.DeadlineExceeded { | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:50 +01:00
										 |  |  | 			t.Fatalf("expected error not found, want %#v, got %#v", context.DeadlineExceeded, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-10 12:53:38 +01:00
										 |  |  | 		if i > 2 { | 
					
						
							|  |  |  | 			t.Fatalf("wrong number of files returned by List, want <= 2, got %v", i) | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:50 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err := s.delayedRemove(t, b, testFiles...) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-25 17:07:36 +01:00
										 |  |  | type errorCloser struct { | 
					
						
							|  |  |  | 	io.Reader | 
					
						
							| 
									
										
										
										
											2017-06-16 10:55:04 +02:00
										 |  |  | 	l int | 
					
						
							|  |  |  | 	t testing.TB | 
					
						
							| 
									
										
										
										
											2017-01-25 17:07:36 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (ec errorCloser) Close() error { | 
					
						
							|  |  |  | 	ec.t.Error("forbidden method close was called") | 
					
						
							|  |  |  | 	return errors.New("forbidden method close was called") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-16 10:55:04 +02:00
										 |  |  | func (ec errorCloser) Len() int { | 
					
						
							|  |  |  | 	return ec.l | 
					
						
							| 
									
										
										
										
											2017-05-13 19:56:11 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 12:56:10 +02:00
										 |  |  | // TestSave tests saving data in the backend. | 
					
						
							|  |  |  | func (s *Suite) TestSave(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2017-05-28 10:16:29 +02:00
										 |  |  | 	seedRand(t) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 22:23:46 +02:00
										 |  |  | 	b := s.open(t) | 
					
						
							|  |  |  | 	defer s.close(t, b) | 
					
						
							| 
									
										
										
										
											2016-08-31 22:51:35 +02:00
										 |  |  | 	var id restic.ID | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-01 12:42:10 +02:00
										 |  |  | 	saveTests := 10 | 
					
						
							| 
									
										
										
										
											2017-05-01 22:23:46 +02:00
										 |  |  | 	if s.MinimalData { | 
					
						
							| 
									
										
										
										
											2017-05-01 12:42:10 +02:00
										 |  |  | 		saveTests = 2 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for i := 0; i < saveTests; i++ { | 
					
						
							| 
									
										
										
										
											2016-01-24 17:46:18 +01:00
										 |  |  | 		length := rand.Intn(1<<23) + 200000 | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 		data := test.Random(23, length) | 
					
						
							| 
									
										
										
										
											2016-01-24 17:46:18 +01:00
										 |  |  | 		// use the first 32 byte as the ID | 
					
						
							|  |  |  | 		copy(id[:], data) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | 		h := restic.Handle{ | 
					
						
							| 
									
										
										
										
											2016-09-01 21:19:30 +02:00
										 |  |  | 			Type: restic.DataFile, | 
					
						
							|  |  |  | 			Name: fmt.Sprintf("%s-%d", id, i), | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		err := b.Save(context.TODO(), h, bytes.NewReader(data)) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 		test.OK(t, err) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		buf, err := backend.LoadAll(context.TODO(), b, h) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 		test.OK(t, err) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 		if len(buf) != len(data) { | 
					
						
							|  |  |  | 			t.Fatalf("number of bytes does not match, want %v, got %v", len(data), len(buf)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if !bytes.Equal(buf, data) { | 
					
						
							|  |  |  | 			t.Fatalf("data not equal") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		fi, err := b.Stat(context.TODO(), h) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 		test.OK(t, err) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 		if fi.Name != h.Name { | 
					
						
							|  |  |  | 			t.Errorf("Stat() returned wrong name, want %q, got %q", h.Name, fi.Name) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 		if fi.Size != int64(len(data)) { | 
					
						
							| 
									
										
										
										
											2018-01-20 19:34:38 +01:00
										 |  |  | 			t.Errorf("Stat() returned different size, want %q, got %d", len(data), fi.Size) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		err = b.Remove(context.TODO(), h) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 			t.Fatalf("error removing item: %+v", err) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-01-25 17:07:36 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// test saving from a tempfile | 
					
						
							|  |  |  | 	tmpfile, err := ioutil.TempFile("", "restic-backend-save-test-") | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	length := rand.Intn(1<<23) + 200000 | 
					
						
							|  |  |  | 	data := test.Random(23, length) | 
					
						
							|  |  |  | 	copy(id[:], data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if _, err = tmpfile.Write(data); err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 00:15:17 +02:00
										 |  |  | 	if _, err = tmpfile.Seek(0, io.SeekStart); err != nil { | 
					
						
							| 
									
										
										
										
											2017-01-25 17:07:36 +01:00
										 |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	h := restic.Handle{Type: restic.DataFile, Name: id.String()} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// wrap the tempfile in an errorCloser, so we can detect if the backend | 
					
						
							|  |  |  | 	// closes the reader | 
					
						
							| 
									
										
										
										
											2017-06-16 10:55:04 +02:00
										 |  |  | 	err = b.Save(context.TODO(), h, errorCloser{t: t, l: length, Reader: tmpfile}) | 
					
						
							| 
									
										
										
										
											2017-01-25 17:07:36 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-16 13:59:55 +02:00
										 |  |  | 	err = s.delayedRemove(t, b, h) | 
					
						
							| 
									
										
										
										
											2017-05-14 00:15:17 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("error removing item: %+v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// try again directly with the temp file | 
					
						
							|  |  |  | 	if _, err = tmpfile.Seek(588, io.SeekStart); err != nil { | 
					
						
							| 
									
										
										
										
											2017-01-25 17:07:36 +01:00
										 |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	err = b.Save(context.TODO(), h, tmpfile) | 
					
						
							| 
									
										
										
										
											2017-05-14 00:15:17 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err = tmpfile.Close(); err != nil { | 
					
						
							| 
									
										
										
										
											2017-01-25 17:07:36 +01:00
										 |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	err = b.Remove(context.TODO(), h) | 
					
						
							| 
									
										
										
										
											2017-01-25 17:07:36 +01:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 		t.Fatalf("error removing item: %+v", err) | 
					
						
							| 
									
										
										
										
											2017-01-25 17:07:36 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-14 00:15:17 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if err = os.Remove(tmpfile.Name()); err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var filenameTests = []struct { | 
					
						
							|  |  |  | 	name string | 
					
						
							|  |  |  | 	data string | 
					
						
							|  |  |  | }{ | 
					
						
							|  |  |  | 	{"1dfc6bc0f06cb255889e9ea7860a5753e8eb9665c9a96627971171b444e3113e", "x"}, | 
					
						
							| 
									
										
										
										
											2017-03-16 21:50:26 +01:00
										 |  |  | 	{"f00b4r", "foobar"}, | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		"1dfc6bc0f06cb255889e9ea7860a5753e8eb9665c9a96627971171b444e3113e4bf8f2d9144cc5420a80f04a4880ad6155fc58903a4fb6457c476c43541dcaa6-5", | 
					
						
							|  |  |  | 		"foobar content of data blob", | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 12:56:10 +02:00
										 |  |  | // TestSaveFilenames tests saving data with various file names in the backend. | 
					
						
							|  |  |  | func (s *Suite) TestSaveFilenames(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2017-05-01 22:23:46 +02:00
										 |  |  | 	b := s.open(t) | 
					
						
							|  |  |  | 	defer s.close(t, b) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for i, test := range filenameTests { | 
					
						
							| 
									
										
										
										
											2016-09-01 21:19:30 +02:00
										 |  |  | 		h := restic.Handle{Name: test.name, Type: restic.DataFile} | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		err := b.Save(context.TODO(), h, strings.NewReader(test.data)) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 			t.Errorf("test %d failed: Save() returned %+v", i, err) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		buf, err := backend.LoadAll(context.TODO(), b, h) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 			t.Errorf("test %d failed: Load() returned %+v", i, err) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if !bytes.Equal(buf, []byte(test.data)) { | 
					
						
							|  |  |  | 			t.Errorf("test %d: returned wrong bytes", i) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		err = b.Remove(context.TODO(), h) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 			t.Errorf("test %d failed: Remove() returned %+v", i, err) | 
					
						
							| 
									
										
										
										
											2016-01-24 16:59:38 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | var testStrings = []struct { | 
					
						
							|  |  |  | 	id   string | 
					
						
							|  |  |  | 	data string | 
					
						
							|  |  |  | }{ | 
					
						
							|  |  |  | 	{"c3ab8ff13720e8ad9047dd39466b3c8974e592c2fa383d4a3960714caef0c4f2", "foobar"}, | 
					
						
							|  |  |  | 	{"248d6a61d20638b8e5c026930c3e6039a33ce45964ff2167f6ecedd419db06c1", "abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq"}, | 
					
						
							|  |  |  | 	{"cc5d46bdb4991c6eae3eb739c9c8a7a46fe9654fab79c47b4fe48383b5b25e1c", "foo/bar"}, | 
					
						
							|  |  |  | 	{"4e54d2c721cbdb730f01b10b62dec622962b36966ec685880effa63d71c808f2", "foo/../../baz"}, | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-25 17:48:35 +01:00
										 |  |  | func store(t testing.TB, b restic.Backend, tpe restic.FileType, data []byte) restic.Handle { | 
					
						
							| 
									
										
										
										
											2016-08-31 22:51:35 +02:00
										 |  |  | 	id := restic.Hash(data) | 
					
						
							| 
									
										
										
										
											2017-01-25 17:48:35 +01:00
										 |  |  | 	h := restic.Handle{Name: id.String(), Type: tpe} | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	err := b.Save(context.TODO(), h, bytes.NewReader(data)) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 	test.OK(t, err) | 
					
						
							| 
									
										
										
										
											2017-01-25 17:48:35 +01:00
										 |  |  | 	return h | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-28 13:05:35 +02:00
										 |  |  | // testLoad loads a blob (but discards its contents). | 
					
						
							|  |  |  | func testLoad(b restic.Backend, h restic.Handle, length int, offset int64) error { | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 	rd, err := b.Load(context.TODO(), h, 0, 0) | 
					
						
							| 
									
										
										
										
											2017-05-28 13:05:35 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_, err = io.Copy(ioutil.Discard, rd) | 
					
						
							|  |  |  | 	cerr := rd.Close() | 
					
						
							|  |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		err = cerr | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-16 13:59:55 +02:00
										 |  |  | func (s *Suite) delayedRemove(t testing.TB, be restic.Backend, handles ...restic.Handle) error { | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	// Some backend (swift, I'm looking at you) may implement delayed | 
					
						
							|  |  |  | 	// removal of data. Let's wait a bit if this happens. | 
					
						
							| 
									
										
										
										
											2017-05-01 10:13:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 18:00:17 +02:00
										 |  |  | 	for _, h := range handles { | 
					
						
							|  |  |  | 		err := be.Remove(context.TODO(), h) | 
					
						
							| 
									
										
										
										
											2017-09-16 13:59:55 +02:00
										 |  |  | 		if s.ErrorHandler != nil { | 
					
						
							|  |  |  | 			err = s.ErrorHandler(t, be, err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-06-15 20:05:35 +02:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return err | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-07-21 18:00:17 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, h := range handles { | 
					
						
							|  |  |  | 		start := time.Now() | 
					
						
							|  |  |  | 		attempt := 0 | 
					
						
							|  |  |  | 		var found bool | 
					
						
							|  |  |  | 		var err error | 
					
						
							| 
									
										
										
										
											2017-09-16 13:59:55 +02:00
										 |  |  | 		for time.Since(start) <= s.WaitForDelayedRemoval { | 
					
						
							| 
									
										
										
										
											2017-07-21 18:00:17 +02:00
										 |  |  | 			found, err = be.Test(context.TODO(), h) | 
					
						
							| 
									
										
										
										
											2017-09-16 13:59:55 +02:00
										 |  |  | 			if s.ErrorHandler != nil { | 
					
						
							|  |  |  | 				err = s.ErrorHandler(t, be, err) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-07-21 18:00:17 +02:00
										 |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return err | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-06-15 20:05:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 18:00:17 +02:00
										 |  |  | 			if !found { | 
					
						
							|  |  |  | 				break | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			time.Sleep(2 * time.Second) | 
					
						
							|  |  |  | 			attempt++ | 
					
						
							| 
									
										
										
										
											2017-06-15 20:05:35 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 18:00:17 +02:00
										 |  |  | 		if found { | 
					
						
							|  |  |  | 			t.Fatalf("removed blob %v still present after %v (%d attempts)", h, time.Since(start), attempt) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-06-18 17:36:57 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2017-03-29 23:58:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-18 17:36:57 +02:00
										 |  |  | func delayedList(t testing.TB, b restic.Backend, tpe restic.FileType, max int, maxwait time.Duration) restic.IDs { | 
					
						
							| 
									
										
										
										
											2017-06-15 19:41:07 +02:00
										 |  |  | 	list := restic.NewIDSet() | 
					
						
							| 
									
										
										
										
											2017-06-18 17:36:57 +02:00
										 |  |  | 	start := time.Now() | 
					
						
							| 
									
										
										
										
											2017-06-15 19:41:07 +02:00
										 |  |  | 	for i := 0; i < max; i++ { | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:07 +01:00
										 |  |  | 		err := b.List(context.TODO(), tpe, func(fi restic.FileInfo) error { | 
					
						
							|  |  |  | 			id := restic.TestParseID(fi.Name) | 
					
						
							| 
									
										
										
										
											2017-06-15 19:41:07 +02:00
										 |  |  | 			list.Insert(id) | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:07 +01:00
										 |  |  | 			return nil | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Fatal(err) | 
					
						
							| 
									
										
										
										
											2017-06-15 19:41:07 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-01-20 13:43:07 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-18 17:36:57 +02:00
										 |  |  | 		if len(list) < max && time.Since(start) < maxwait { | 
					
						
							|  |  |  | 			time.Sleep(500 * time.Millisecond) | 
					
						
							| 
									
										
										
										
											2017-06-15 19:41:07 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return list.List() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 12:56:10 +02:00
										 |  |  | // TestBackend tests all functions of the backend. | 
					
						
							|  |  |  | func (s *Suite) TestBackend(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2017-05-01 22:23:46 +02:00
										 |  |  | 	b := s.open(t) | 
					
						
							|  |  |  | 	defer s.close(t, b) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-31 22:39:36 +02:00
										 |  |  | 	for _, tpe := range []restic.FileType{ | 
					
						
							|  |  |  | 		restic.DataFile, restic.KeyFile, restic.LockFile, | 
					
						
							|  |  |  | 		restic.SnapshotFile, restic.IndexFile, | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 	} { | 
					
						
							|  |  |  | 		// detect non-existing files | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 		for _, ts := range testStrings { | 
					
						
							|  |  |  | 			id, err := restic.ParseID(ts.id) | 
					
						
							|  |  |  | 			test.OK(t, err) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// test if blob is already in repository | 
					
						
							| 
									
										
										
										
											2017-01-25 17:48:35 +01:00
										 |  |  | 			h := restic.Handle{Type: tpe, Name: id.String()} | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 			ret, err := b.Test(context.TODO(), h) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 			test.OK(t, err) | 
					
						
							|  |  |  | 			test.Assert(t, !ret, "blob was found to exist before creating") | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-24 01:00:27 +01:00
										 |  |  | 			// try to stat a not existing blob | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 			_, err = b.Stat(context.TODO(), h) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 			test.Assert(t, err != nil, "blob data could be extracted before creation") | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// try to read not existing blob | 
					
						
							| 
									
										
										
										
											2017-05-28 13:05:35 +02:00
										 |  |  | 			err = testLoad(b, h, 0, 0) | 
					
						
							|  |  |  | 			test.Assert(t, err != nil, "blob could be read before creation") | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// try to get string out, should fail | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 			ret, err = b.Test(context.TODO(), h) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 			test.OK(t, err) | 
					
						
							|  |  |  | 			test.Assert(t, !ret, "id %q was found (but should not have)", ts.id) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// add files | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 		for _, ts := range testStrings { | 
					
						
							|  |  |  | 			store(t, b, tpe, []byte(ts.data)) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-24 01:00:27 +01:00
										 |  |  | 			// test Load() | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 			h := restic.Handle{Type: tpe, Name: ts.id} | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 			buf, err := backend.LoadAll(context.TODO(), b, h) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 			test.OK(t, err) | 
					
						
							|  |  |  | 			test.Equals(t, ts.data, string(buf)) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// try to read it out with an offset and a length | 
					
						
							|  |  |  | 			start := 1 | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 			end := len(ts.data) - 2 | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 			length := end - start | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-24 01:00:27 +01:00
										 |  |  | 			buf2 := make([]byte, length) | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 			rd, err := b.Load(context.TODO(), h, len(buf2), int64(start)) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 			test.OK(t, err) | 
					
						
							| 
									
										
										
										
											2017-01-23 17:20:08 +01:00
										 |  |  | 			n, err := io.ReadFull(rd, buf2) | 
					
						
							|  |  |  | 			test.OK(t, err) | 
					
						
							|  |  |  | 			test.Equals(t, len(buf2), n) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			remaining, err := io.Copy(ioutil.Discard, rd) | 
					
						
							|  |  |  | 			test.OK(t, err) | 
					
						
							|  |  |  | 			test.Equals(t, int64(0), remaining) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			test.OK(t, rd.Close()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 			test.Equals(t, ts.data[start:end], string(buf2)) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// test adding the first file again | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 		ts := testStrings[0] | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// create blob | 
					
						
							| 
									
										
										
										
											2017-03-16 21:50:26 +01:00
										 |  |  | 		h := restic.Handle{Type: tpe, Name: ts.id} | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		err := b.Save(context.TODO(), h, strings.NewReader(ts.data)) | 
					
						
							| 
									
										
										
										
											2018-01-27 13:06:15 +00:00
										 |  |  | 		test.Assert(t, err != nil, "backend has allowed overwrite of existing blob: expected error for %v, got %v", h, err) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// remove and recreate | 
					
						
							| 
									
										
										
										
											2017-09-16 13:59:55 +02:00
										 |  |  | 		err = s.delayedRemove(t, b, h) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 		test.OK(t, err) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// test that the blob is gone | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		ok, err := b.Test(context.TODO(), h) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 		test.OK(t, err) | 
					
						
							| 
									
										
										
										
											2017-04-11 20:08:25 +02:00
										 |  |  | 		test.Assert(t, !ok, "removed blob still present") | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// create blob | 
					
						
							| 
									
										
										
										
											2017-06-03 17:39:57 +02:00
										 |  |  | 		err = b.Save(context.TODO(), h, strings.NewReader(ts.data)) | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 		test.OK(t, err) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// list items | 
					
						
							| 
									
										
										
										
											2016-08-31 22:51:35 +02:00
										 |  |  | 		IDs := restic.IDs{} | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-04 14:38:18 +02:00
										 |  |  | 		for _, ts := range testStrings { | 
					
						
							|  |  |  | 			id, err := restic.ParseID(ts.id) | 
					
						
							|  |  |  | 			test.OK(t, err) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 			IDs = append(IDs, id) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-18 17:36:57 +02:00
										 |  |  | 		list := delayedList(t, b, tpe, len(IDs), s.WaitForDelayedRemoval) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 		if len(IDs) != len(list) { | 
					
						
							|  |  |  | 			t.Fatalf("wrong number of IDs returned: want %d, got %d", len(IDs), len(list)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		sort.Sort(IDs) | 
					
						
							|  |  |  | 		sort.Sort(list) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if !reflect.DeepEqual(IDs, list) { | 
					
						
							|  |  |  | 			t.Fatalf("lists aren't equal, want:\n  %v\n  got:\n%v\n", IDs, list) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 15:53:32 +02:00
										 |  |  | 		var handles []restic.Handle | 
					
						
							|  |  |  | 		for _, ts := range testStrings { | 
					
						
							|  |  |  | 			id, err := restic.ParseID(ts.id) | 
					
						
							|  |  |  | 			test.OK(t, err) | 
					
						
							| 
									
										
										
										
											2017-01-25 17:48:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 15:53:32 +02:00
										 |  |  | 			h := restic.Handle{Type: tpe, Name: id.String()} | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 15:53:32 +02:00
										 |  |  | 			found, err := b.Test(context.TODO(), h) | 
					
						
							|  |  |  | 			test.OK(t, err) | 
					
						
							|  |  |  | 			test.Assert(t, found, fmt.Sprintf("id %q not found", id)) | 
					
						
							| 
									
										
										
										
											2017-07-21 18:00:17 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 15:53:32 +02:00
										 |  |  | 			handles = append(handles, h) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-10-14 15:53:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		test.OK(t, s.delayedRemove(t, b, handles...)) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 15:56:46 +02:00
										 |  |  | // TestZZZDelete tests the Delete function. The name ensures that this test is executed last. | 
					
						
							|  |  |  | func (s *Suite) TestZZZDelete(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2017-05-01 22:23:46 +02:00
										 |  |  | 	if !test.TestCleanupTempDirs { | 
					
						
							|  |  |  | 		t.Skipf("not removing backend, TestCleanupTempDirs is false") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	b := s.open(t) | 
					
						
							|  |  |  | 	defer s.close(t, b) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 15:56:46 +02:00
										 |  |  | 	err := b.Delete(context.TODO()) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2017-04-10 21:46:41 +02:00
										 |  |  | 		t.Fatalf("error deleting backend: %+v", err) | 
					
						
							| 
									
										
										
										
											2016-01-23 17:08:03 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | } |