| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | package cache | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io/ioutil" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"path/filepath" | 
					
						
							| 
									
										
										
										
											2018-05-01 16:22:12 +02:00
										 |  |  | 	"regexp" | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	"strconv" | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/pkg/errors" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/debug" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/fs" | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/restic" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Cache manages a local cache. | 
					
						
							|  |  |  | type Cache struct { | 
					
						
							| 
									
										
										
										
											2017-09-24 22:24:11 +02:00
										 |  |  | 	Path             string | 
					
						
							|  |  |  | 	Base             string | 
					
						
							|  |  |  | 	PerformReadahead func(restic.Handle) bool | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const dirMode = 0700 | 
					
						
							|  |  |  | const fileMode = 0600 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func readVersion(dir string) (v uint, err error) { | 
					
						
							|  |  |  | 	buf, err := ioutil.ReadFile(filepath.Join(dir, "version")) | 
					
						
							|  |  |  | 	if os.IsNotExist(err) { | 
					
						
							|  |  |  | 		return 0, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return 0, errors.Wrap(err, "ReadFile") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ver, err := strconv.ParseUint(string(buf), 10, 32) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return 0, errors.Wrap(err, "ParseUint") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return uint(ver), nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const cacheVersion = 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ensure Cache implements restic.Cache | 
					
						
							|  |  |  | var _ restic.Cache = &Cache{} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var cacheLayoutPaths = map[restic.FileType]string{ | 
					
						
							|  |  |  | 	restic.DataFile:     "data", | 
					
						
							|  |  |  | 	restic.SnapshotFile: "snapshots", | 
					
						
							|  |  |  | 	restic.IndexFile:    "index", | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-11 18:59:19 +02:00
										 |  |  | const cachedirTagSignature = "Signature: 8a477f597d28d172789f06886806bc55\n" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func writeCachedirTag(dir string) error { | 
					
						
							|  |  |  | 	if err := fs.MkdirAll(dir, dirMode); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	f, err := fs.OpenFile(filepath.Join(dir, "CACHEDIR.TAG"), os.O_CREATE|os.O_EXCL|os.O_WRONLY, 0644) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		if os.IsExist(errors.Cause(err)) { | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return errors.Wrap(err, "OpenFile") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	debug.Log("Create CACHEDIR.TAG at %v", dir) | 
					
						
							|  |  |  | 	if _, err := f.Write([]byte(cachedirTagSignature)); err != nil { | 
					
						
							| 
									
										
										
										
											2017-10-25 12:03:55 -04:00
										 |  |  | 		_ = f.Close() | 
					
						
							| 
									
										
										
										
											2017-09-11 18:59:19 +02:00
										 |  |  | 		return errors.Wrap(err, "Write") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return f.Close() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // New returns a new cache for the repo ID at basedir. If basedir is the empty | 
					
						
							|  |  |  | // string, the default cache location (according to the XDG standard) is used. | 
					
						
							| 
									
										
										
										
											2017-09-24 22:24:11 +02:00
										 |  |  | // | 
					
						
							|  |  |  | // For partial files, the complete file is loaded and stored in the cache when | 
					
						
							|  |  |  | // performReadahead returns true. | 
					
						
							| 
									
										
										
										
											2017-09-11 18:59:19 +02:00
										 |  |  | func New(id string, basedir string) (c *Cache, err error) { | 
					
						
							|  |  |  | 	if basedir == "" { | 
					
						
							| 
									
										
										
										
											2017-11-20 21:58:38 +01:00
										 |  |  | 		basedir, err = DefaultDir() | 
					
						
							| 
									
										
										
										
											2017-11-20 06:10:42 +01:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-11-20 21:58:38 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err = mkdirCacheDir(basedir) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-11 18:59:19 +02:00
										 |  |  | 	// create base dir and tag it as a cache directory | 
					
						
							|  |  |  | 	if err = writeCachedirTag(basedir); err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cachedir := filepath.Join(basedir, id) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	debug.Log("using cache dir %v", cachedir) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	v, err := readVersion(cachedir) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if v > cacheVersion { | 
					
						
							|  |  |  | 		return nil, errors.New("cache version is newer") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 21:58:38 +01:00
										 |  |  | 	// create the repo cache dir if it does not exist yet | 
					
						
							|  |  |  | 	if err = fs.MkdirAll(cachedir, dirMode); err != nil { | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 21:58:38 +01:00
										 |  |  | 	// update the timestamp so that we can detect old cache dirs | 
					
						
							|  |  |  | 	err = updateTimestamp(cachedir) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if v < cacheVersion { | 
					
						
							|  |  |  | 		err = ioutil.WriteFile(filepath.Join(cachedir, "version"), []byte(fmt.Sprintf("%d", cacheVersion)), 0644) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, errors.Wrap(err, "WriteFile") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, p := range cacheLayoutPaths { | 
					
						
							|  |  |  | 		if err = fs.MkdirAll(filepath.Join(cachedir, p), dirMode); err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c = &Cache{ | 
					
						
							|  |  |  | 		Path: cachedir, | 
					
						
							| 
									
										
										
										
											2017-09-11 18:59:19 +02:00
										 |  |  | 		Base: basedir, | 
					
						
							| 
									
										
										
										
											2017-09-24 22:24:11 +02:00
										 |  |  | 		PerformReadahead: func(restic.Handle) bool { | 
					
						
							|  |  |  | 			// do not perform readahead by default | 
					
						
							|  |  |  | 			return false | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return c, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | // updateTimestamp sets the modification timestamp (mtime and atime) for the | 
					
						
							|  |  |  | // directory d to the current time. | 
					
						
							|  |  |  | func updateTimestamp(d string) error { | 
					
						
							|  |  |  | 	t := time.Now() | 
					
						
							|  |  |  | 	return fs.Chtimes(d, t, t) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 16:22:12 +02:00
										 |  |  | // MaxCacheAge is the default age (30 days) after which cache directories are considered old. | 
					
						
							|  |  |  | const MaxCacheAge = 30 * 24 * time.Hour | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 16:22:12 +02:00
										 |  |  | func validCacheDirName(s string) bool { | 
					
						
							|  |  |  | 	r := regexp.MustCompile(`^[a-fA-F0-9]{64}$`) | 
					
						
							|  |  |  | 	if !r.MatchString(s) { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return true | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 16:22:12 +02:00
										 |  |  | // listCacheDirs returns the list of cache directories. | 
					
						
							|  |  |  | func listCacheDirs(basedir string) ([]os.FileInfo, error) { | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | 	f, err := fs.Open(basedir) | 
					
						
							| 
									
										
										
										
											2018-05-01 16:22:12 +02:00
										 |  |  | 	if err != nil && os.IsNotExist(errors.Cause(err)) { | 
					
						
							|  |  |  | 		return nil, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	entries, err := f.Readdir(-1) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 16:22:12 +02:00
										 |  |  | 	err = f.Close() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	result := make([]os.FileInfo, 0, len(entries)) | 
					
						
							|  |  |  | 	for _, entry := range entries { | 
					
						
							|  |  |  | 		if !entry.IsDir() { | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 16:22:12 +02:00
										 |  |  | 		if !validCacheDirName(entry.Name()) { | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 16:22:12 +02:00
										 |  |  | 		result = append(result, entry) | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 16:22:12 +02:00
										 |  |  | 	return result, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // All returns a list of cache directories. | 
					
						
							|  |  |  | func All(basedir string) (dirs []os.FileInfo, err error) { | 
					
						
							|  |  |  | 	return listCacheDirs(basedir) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // OlderThan returns the list of cache directories older than max. | 
					
						
							|  |  |  | func OlderThan(basedir string, max time.Duration) ([]os.FileInfo, error) { | 
					
						
							|  |  |  | 	entries, err := listCacheDirs(basedir) | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 16:22:12 +02:00
										 |  |  | 	var oldCacheDirs []os.FileInfo | 
					
						
							|  |  |  | 	for _, fi := range entries { | 
					
						
							|  |  |  | 		if !IsOld(fi.ModTime(), max) { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		oldCacheDirs = append(oldCacheDirs, fi) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 21:32:25 +01:00
										 |  |  | 	debug.Log("%d old cache dirs found", len(oldCacheDirs)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return oldCacheDirs, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-01 16:22:12 +02:00
										 |  |  | // Old returns a list of cache directories with a modification time of more | 
					
						
							|  |  |  | // than 30 days ago. | 
					
						
							|  |  |  | func Old(basedir string) ([]os.FileInfo, error) { | 
					
						
							|  |  |  | 	return OlderThan(basedir, MaxCacheAge) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // IsOld returns true if the timestamp is considered old. | 
					
						
							|  |  |  | func IsOld(t time.Time, maxAge time.Duration) bool { | 
					
						
							|  |  |  | 	oldest := time.Now().Add(-maxAge) | 
					
						
							|  |  |  | 	return t.Before(oldest) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | // errNoSuchFile is returned when a file is not cached. | 
					
						
							|  |  |  | type errNoSuchFile struct { | 
					
						
							|  |  |  | 	Type string | 
					
						
							|  |  |  | 	Name string | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (e errNoSuchFile) Error() string { | 
					
						
							|  |  |  | 	return fmt.Sprintf("file %v (%v) is not cached", e.Name, e.Type) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // IsNotExist returns true if the error was caused by a non-existing file. | 
					
						
							|  |  |  | func (c *Cache) IsNotExist(err error) bool { | 
					
						
							|  |  |  | 	_, ok := errors.Cause(err).(errNoSuchFile) | 
					
						
							|  |  |  | 	return ok | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Wrap returns a backend with a cache. | 
					
						
							|  |  |  | func (c *Cache) Wrap(be restic.Backend) restic.Backend { | 
					
						
							| 
									
										
										
										
											2017-09-25 15:21:16 +02:00
										 |  |  | 	return newBackend(be, c) | 
					
						
							| 
									
										
										
										
											2017-06-10 13:10:08 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-09-11 21:37:10 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | // BaseDir returns the base directory. | 
					
						
							|  |  |  | func (c *Cache) BaseDir() string { | 
					
						
							|  |  |  | 	return c.Base | 
					
						
							|  |  |  | } |