| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | package cache | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"math/rand" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/backend" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/backend/mem" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/restic" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/test" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func loadAndCompare(t testing.TB, be restic.Backend, h restic.Handle, data []byte) { | 
					
						
							|  |  |  | 	buf, err := backend.LoadAll(context.TODO(), be, h) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if len(buf) != len(data) { | 
					
						
							|  |  |  | 		t.Fatalf("wrong number of bytes read, want %v, got %v", len(data), len(buf)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !bytes.Equal(buf, data) { | 
					
						
							|  |  |  | 		t.Fatalf("wrong data returned, want:\n  %02x\ngot:\n  %02x", data[:16], buf[:16]) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func save(t testing.TB, be restic.Backend, h restic.Handle, data []byte) { | 
					
						
							| 
									
										
										
										
											2018-03-03 14:20:54 +01:00
										 |  |  | 	err := be.Save(context.TODO(), h, restic.NewByteReader(data)) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func remove(t testing.TB, be restic.Backend, h restic.Handle) { | 
					
						
							|  |  |  | 	err := be.Remove(context.TODO(), h) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func randomData(n int) (restic.Handle, []byte) { | 
					
						
							|  |  |  | 	data := test.Random(rand.Int(), n) | 
					
						
							|  |  |  | 	id := restic.Hash(data) | 
					
						
							|  |  |  | 	copy(id[:], data) | 
					
						
							|  |  |  | 	h := restic.Handle{ | 
					
						
							|  |  |  | 		Type: restic.IndexFile, | 
					
						
							|  |  |  | 		Name: id.String(), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return h, data | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestBackend(t *testing.T) { | 
					
						
							|  |  |  | 	be := mem.New() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c, cleanup := TestNewCache(t) | 
					
						
							|  |  |  | 	defer cleanup() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wbe := c.Wrap(be) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	h, data := randomData(5234142) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// save directly in backend | 
					
						
							|  |  |  | 	save(t, be, h, data) | 
					
						
							|  |  |  | 	if c.Has(h) { | 
					
						
							|  |  |  | 		t.Errorf("cache has file too early") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// load data via cache | 
					
						
							|  |  |  | 	loadAndCompare(t, wbe, h, data) | 
					
						
							|  |  |  | 	if !c.Has(h) { | 
					
						
							| 
									
										
										
										
											2018-01-24 12:09:41 +00:00
										 |  |  | 		t.Errorf("cache doesn't have file after load") | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// remove via cache | 
					
						
							|  |  |  | 	remove(t, wbe, h) | 
					
						
							|  |  |  | 	if c.Has(h) { | 
					
						
							|  |  |  | 		t.Errorf("cache has file after remove") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// save via cache | 
					
						
							|  |  |  | 	save(t, wbe, h, data) | 
					
						
							|  |  |  | 	if !c.Has(h) { | 
					
						
							| 
									
										
										
										
											2018-01-24 12:09:41 +00:00
										 |  |  | 		t.Errorf("cache doesn't have file after load") | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// load data directly from backend | 
					
						
							|  |  |  | 	loadAndCompare(t, be, h, data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// load data via cache | 
					
						
							|  |  |  | 	loadAndCompare(t, be, h, data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// remove directly | 
					
						
							|  |  |  | 	remove(t, be, h) | 
					
						
							|  |  |  | 	if !c.Has(h) { | 
					
						
							|  |  |  | 		t.Errorf("file not in cache any more") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// run stat | 
					
						
							|  |  |  | 	_, err := wbe.Stat(context.TODO(), h) | 
					
						
							|  |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		t.Errorf("expected error for removed file not found, got nil") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !wbe.IsNotExist(err) { | 
					
						
							|  |  |  | 		t.Errorf("Stat() returned error that does not match IsNotExist(): %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if c.Has(h) { | 
					
						
							|  |  |  | 		t.Errorf("removed file still in cache after stat") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |