| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | package cache | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2017-09-25 15:21:16 +02:00
										 |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/debug" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/restic" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Backend wraps a restic.Backend and adds a cache. | 
					
						
							|  |  |  | type Backend struct { | 
					
						
							|  |  |  | 	restic.Backend | 
					
						
							| 
									
										
										
										
											2017-09-24 22:24:11 +02:00
										 |  |  | 	*Cache | 
					
						
							| 
									
										
										
										
											2017-09-25 15:21:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// inProgress contains the handle for all files that are currently | 
					
						
							|  |  |  | 	// downloaded. The channel in the value is closed as soon as the download | 
					
						
							|  |  |  | 	// is finished. | 
					
						
							|  |  |  | 	inProgressMutex sync.Mutex | 
					
						
							|  |  |  | 	inProgress      map[restic.Handle]chan struct{} | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ensure cachedBackend implements restic.Backend | 
					
						
							|  |  |  | var _ restic.Backend = &Backend{} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-25 15:21:16 +02:00
										 |  |  | func newBackend(be restic.Backend, c *Cache) *Backend { | 
					
						
							|  |  |  | 	return &Backend{ | 
					
						
							|  |  |  | 		Backend:    be, | 
					
						
							|  |  |  | 		Cache:      c, | 
					
						
							|  |  |  | 		inProgress: make(map[restic.Handle]chan struct{}), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | // Remove deletes a file from the backend and the cache if it has been cached. | 
					
						
							|  |  |  | func (b *Backend) Remove(ctx context.Context, h restic.Handle) error { | 
					
						
							|  |  |  | 	debug.Log("cache Remove(%v)", h) | 
					
						
							|  |  |  | 	err := b.Backend.Remove(ctx, h) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-02 18:27:52 +01:00
										 |  |  | 	return b.Cache.remove(h) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var autoCacheTypes = map[restic.FileType]struct{}{ | 
					
						
							| 
									
										
										
										
											2018-05-11 11:07:16 -04:00
										 |  |  | 	restic.IndexFile:    {}, | 
					
						
							|  |  |  | 	restic.SnapshotFile: {}, | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-17 21:58:24 +02:00
										 |  |  | // Save stores a new file in the backend and the cache. | 
					
						
							| 
									
										
										
										
											2018-03-03 14:20:54 +01:00
										 |  |  | func (b *Backend) Save(ctx context.Context, h restic.Handle, rd restic.RewindReader) error { | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	if _, ok := autoCacheTypes[h.Type]; !ok { | 
					
						
							|  |  |  | 		return b.Backend.Save(ctx, h, rd) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-24 22:24:01 +02:00
										 |  |  | 	debug.Log("Save(%v): auto-store in the cache", h) | 
					
						
							| 
									
										
										
										
											2017-10-17 21:58:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-03 14:20:54 +01:00
										 |  |  | 	// make sure the reader is at the start | 
					
						
							|  |  |  | 	err := rd.Rewind() | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2018-03-03 14:20:54 +01:00
										 |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2017-10-17 21:58:24 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-03 14:20:54 +01:00
										 |  |  | 	// first, save in the backend | 
					
						
							| 
									
										
										
										
											2017-10-17 21:58:24 +02:00
										 |  |  | 	err = b.Backend.Save(ctx, h, rd) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-03 14:20:54 +01:00
										 |  |  | 	// next, save in the cache | 
					
						
							|  |  |  | 	err = rd.Rewind() | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2018-03-03 14:20:54 +01:00
										 |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2017-10-17 21:58:24 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err = b.Cache.Save(h, rd) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		debug.Log("unable to save %v to cache: %v", h, err) | 
					
						
							| 
									
										
										
										
											2020-03-02 18:27:52 +01:00
										 |  |  | 		_ = b.Cache.remove(h) | 
					
						
							| 
									
										
										
										
											2017-10-17 21:58:24 +02:00
										 |  |  | 		return nil | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-10-17 21:58:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var autoCacheFiles = map[restic.FileType]bool{ | 
					
						
							|  |  |  | 	restic.IndexFile:    true, | 
					
						
							|  |  |  | 	restic.SnapshotFile: true, | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-24 22:50:35 +02:00
										 |  |  | func (b *Backend) cacheFile(ctx context.Context, h restic.Handle) error { | 
					
						
							| 
									
										
										
										
											2017-09-25 15:21:16 +02:00
										 |  |  | 	finish := make(chan struct{}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	b.inProgressMutex.Lock() | 
					
						
							|  |  |  | 	other, alreadyDownloading := b.inProgress[h] | 
					
						
							|  |  |  | 	if !alreadyDownloading { | 
					
						
							|  |  |  | 		b.inProgress[h] = finish | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	b.inProgressMutex.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if alreadyDownloading { | 
					
						
							|  |  |  | 		debug.Log("readahead %v is already performed by somebody else, delegating...", h) | 
					
						
							|  |  |  | 		<-other | 
					
						
							|  |  |  | 		debug.Log("download %v finished", h) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-04 14:30:07 +02:00
										 |  |  | 	// test again, maybe the file was cached in the meantime | 
					
						
							| 
									
										
										
										
											2018-10-04 17:09:43 +02:00
										 |  |  | 	if !b.Cache.Has(h) { | 
					
						
							| 
									
										
										
										
											2018-10-04 14:30:07 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-04 17:09:43 +02:00
										 |  |  | 		// nope, it's still not in the cache, pull it from the repo and save it | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		err := b.Backend.Load(ctx, h, 0, 0, func(rd io.Reader) error { | 
					
						
							|  |  |  | 			return b.Cache.Save(h, rd) | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			// try to remove from the cache, ignore errors | 
					
						
							| 
									
										
										
										
											2020-03-02 18:27:52 +01:00
										 |  |  | 			_ = b.Cache.remove(h) | 
					
						
							| 
									
										
										
										
											2018-10-04 17:09:43 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-09-24 22:50:35 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-09 17:22:33 +02:00
										 |  |  | 	// signal other waiting goroutines that the file may now be cached | 
					
						
							|  |  |  | 	close(finish) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// remove the finish channel from the map | 
					
						
							|  |  |  | 	b.inProgressMutex.Lock() | 
					
						
							|  |  |  | 	delete(b.inProgress, h) | 
					
						
							|  |  |  | 	b.inProgressMutex.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-04 14:30:07 +02:00
										 |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2017-09-24 22:50:35 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-25 15:21:16 +02:00
										 |  |  | // loadFromCacheOrDelegate will try to load the file from the cache, and fall | 
					
						
							|  |  |  | // back to the backend if that fails. | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | func (b *Backend) loadFromCacheOrDelegate(ctx context.Context, h restic.Handle, length int, offset int64, consumer func(rd io.Reader) error) error { | 
					
						
							| 
									
										
										
										
											2020-03-02 18:27:52 +01:00
										 |  |  | 	rd, err := b.Cache.load(h, length, offset) | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2018-06-09 17:22:33 +02:00
										 |  |  | 		debug.Log("error caching %v: %v, falling back to backend", h, err) | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 		return b.Backend.Load(ctx, h, length, offset, consumer) | 
					
						
							| 
									
										
										
										
											2017-09-25 15:21:16 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 	err = consumer(rd) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2018-06-09 17:22:33 +02:00
										 |  |  | 		_ = rd.Close() // ignore secondary errors | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return rd.Close() | 
					
						
							| 
									
										
										
										
											2017-09-25 15:21:16 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | // Load loads a file from the cache or the backend. | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | func (b *Backend) Load(ctx context.Context, h restic.Handle, length int, offset int64, consumer func(rd io.Reader) error) error { | 
					
						
							| 
									
										
										
										
											2017-09-25 15:21:16 +02:00
										 |  |  | 	b.inProgressMutex.Lock() | 
					
						
							|  |  |  | 	waitForFinish, inProgress := b.inProgress[h] | 
					
						
							|  |  |  | 	b.inProgressMutex.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if inProgress { | 
					
						
							|  |  |  | 		debug.Log("downloading %v is already in progress, waiting for finish", h) | 
					
						
							|  |  |  | 		<-waitForFinish | 
					
						
							|  |  |  | 		debug.Log("downloading %v finished", h) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	if b.Cache.Has(h) { | 
					
						
							|  |  |  | 		debug.Log("Load(%v, %v, %v) from cache", h, length, offset) | 
					
						
							| 
									
										
										
										
											2020-03-02 18:27:52 +01:00
										 |  |  | 		rd, err := b.Cache.load(h, length, offset) | 
					
						
							| 
									
										
										
										
											2017-09-24 23:06:54 +02:00
										 |  |  | 		if err == nil { | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 			err = consumer(rd) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							| 
									
										
										
										
											2021-01-30 19:35:46 +01:00
										 |  |  | 				_ = rd.Close() // ignore secondary errors | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 				return err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return rd.Close() | 
					
						
							| 
									
										
										
										
											2017-09-24 23:06:54 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		debug.Log("error loading %v from cache: %v", h, err) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-24 22:24:01 +02:00
										 |  |  | 	// partial file requested | 
					
						
							|  |  |  | 	if offset != 0 || length != 0 { | 
					
						
							| 
									
										
										
										
											2017-09-24 22:50:35 +02:00
										 |  |  | 		if b.Cache.PerformReadahead(h) { | 
					
						
							|  |  |  | 			debug.Log("performing readahead for %v", h) | 
					
						
							| 
									
										
										
										
											2017-09-25 15:21:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-24 22:50:35 +02:00
										 |  |  | 			err := b.cacheFile(ctx, h) | 
					
						
							|  |  |  | 			if err == nil { | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 				return b.loadFromCacheOrDelegate(ctx, h, length, offset, consumer) | 
					
						
							| 
									
										
										
										
											2017-09-24 22:50:35 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			debug.Log("error caching %v: %v", h, err) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-24 22:50:35 +02:00
										 |  |  | 		debug.Log("Load(%v, %v, %v): partial file requested, delegating to backend", h, length, offset) | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 		return b.Backend.Load(ctx, h, length, offset, consumer) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-24 22:50:35 +02:00
										 |  |  | 	// if we don't automatically cache this file type, fall back to the backend | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	if _, ok := autoCacheFiles[h.Type]; !ok { | 
					
						
							| 
									
										
										
										
											2017-09-24 22:50:35 +02:00
										 |  |  | 		debug.Log("Load(%v, %v, %v): delegating to backend", h, length, offset) | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 		return b.Backend.Load(ctx, h, length, offset, consumer) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	debug.Log("auto-store %v in the cache", h) | 
					
						
							| 
									
										
										
										
											2017-09-24 22:50:35 +02:00
										 |  |  | 	err := b.cacheFile(ctx, h) | 
					
						
							|  |  |  | 	if err == nil { | 
					
						
							| 
									
										
										
										
											2018-06-09 17:22:33 +02:00
										 |  |  | 		return b.loadFromCacheOrDelegate(ctx, h, length, offset, consumer) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-24 22:50:35 +02:00
										 |  |  | 	debug.Log("error caching %v: %v, falling back to backend", h, err) | 
					
						
							| 
									
										
										
										
											2018-01-16 23:59:16 -05:00
										 |  |  | 	return b.Backend.Load(ctx, h, length, offset, consumer) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Stat tests whether the backend has a file. If it does not exist but still | 
					
						
							|  |  |  | // exists in the cache, it is removed from the cache. | 
					
						
							|  |  |  | func (b *Backend) Stat(ctx context.Context, h restic.Handle) (restic.FileInfo, error) { | 
					
						
							|  |  |  | 	debug.Log("cache Stat(%v)", h) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fi, err := b.Backend.Stat(ctx, h) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		if b.Backend.IsNotExist(err) { | 
					
						
							|  |  |  | 			// try to remove from the cache, ignore errors | 
					
						
							| 
									
										
										
										
											2020-03-02 18:27:52 +01:00
										 |  |  | 			_ = b.Cache.remove(h) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return fi, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return fi, err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // IsNotExist returns true if the error is caused by a non-existing file. | 
					
						
							|  |  |  | func (b *Backend) IsNotExist(err error) bool { | 
					
						
							|  |  |  | 	return b.Backend.IsNotExist(err) | 
					
						
							|  |  |  | } |