2
2
mirror of https://github.com/octoleo/restic.git synced 2025-01-26 16:48:29 +00:00

283 lines
6.1 KiB
Go
Raw Normal View History

2015-03-28 11:50:23 +01:00
package local
import (
2017-06-03 17:39:57 +02:00
"context"
2015-03-28 11:50:23 +01:00
"io"
"os"
"path/filepath"
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"
2017-07-23 14:21:03 +02:00
"github.com/restic/restic/internal/backend"
"github.com/restic/restic/internal/debug"
"github.com/restic/restic/internal/fs"
2015-03-28 11:50:23 +01:00
)
2016-01-24 20:23:50 +01:00
// Local is a backend in a local directory.
2015-03-28 11:50:23 +01:00
type Local struct {
2017-03-25 13:20:03 +01:00
Config
backend.Layout
2015-03-28 11:50:23 +01:00
}
2017-03-26 22:14:00 +02:00
// ensure statically that *Local implements restic.Backend.
2016-08-31 22:39:36 +02:00
var _ restic.Backend = &Local{}
2017-04-02 19:54:11 +02:00
const defaultLayout = "default"
// dirExists returns true if the name exists and is a directory.
func dirExists(name string) bool {
f, err := fs.Open(name)
if err != nil {
return false
}
fi, err := f.Stat()
if err != nil {
return false
}
if err = f.Close(); err != nil {
return false
}
return fi.IsDir()
}
2016-01-26 22:09:29 +01:00
// Open opens the local backend as specified by config.
2017-03-25 13:20:03 +01:00
func Open(cfg Config) (*Local, error) {
2017-04-02 19:54:11 +02:00
debug.Log("open local backend at %v (layout %q)", cfg.Path, cfg.Layout)
l, err := backend.ParseLayout(&backend.LocalFilesystem{}, cfg.Layout, defaultLayout, cfg.Path)
2017-04-02 17:57:28 +02:00
if err != nil {
return nil, err
}
2017-04-02 17:57:28 +02:00
be := &Local{Config: cfg, Layout: l}
// if data dir exists, make sure that all subdirs also exist
datadir := be.Dirname(restic.Handle{Type: restic.DataFile})
if dirExists(datadir) {
for _, d := range be.Paths() {
if _, err := filepath.Rel(datadir, d); err != nil {
continue
}
err := fs.MkdirAll(d, backend.Modes.Dir)
if err != nil {
return nil, errors.Wrap(err, "MkdirAll")
}
2017-07-02 15:58:03 +02:00
}
}
return be, nil
2015-03-28 11:50:23 +01:00
}
// Create creates all the necessary files and directories for a new local
2015-05-04 20:39:45 +02:00
// backend at dir. Afterwards a new config blob should be created.
2017-03-25 13:20:03 +01:00
func Create(cfg Config) (*Local, error) {
2017-04-02 19:54:11 +02:00
debug.Log("create local backend at %v (layout %q)", cfg.Path, cfg.Layout)
l, err := backend.ParseLayout(&backend.LocalFilesystem{}, cfg.Layout, defaultLayout, cfg.Path)
if err != nil {
return nil, err
}
be := &Local{
Config: cfg,
2017-04-02 19:54:11 +02:00
Layout: l,
}
2015-05-04 20:39:45 +02:00
// test if config file already exists
2017-04-02 19:54:11 +02:00
_, err = fs.Lstat(be.Filename(restic.Handle{Type: restic.ConfigFile}))
2015-03-28 11:50:23 +01:00
if err == nil {
return nil, errors.New("config file already exists")
2015-03-28 11:50:23 +01:00
}
2017-04-19 18:56:01 +02:00
// create paths for data and refs
for _, d := range be.Paths() {
err := fs.MkdirAll(d, backend.Modes.Dir)
2015-03-28 11:50:23 +01:00
if err != nil {
2016-08-29 21:54:50 +02:00
return nil, errors.Wrap(err, "MkdirAll")
2015-03-28 11:50:23 +01:00
}
}
return be, nil
2015-03-28 11:50:23 +01:00
}
// Location returns this backend's location (the directory name).
func (b *Local) Location() string {
2017-03-25 13:20:03 +01:00
return b.Path
2015-03-28 11:50:23 +01:00
}
2017-06-15 13:40:27 +02:00
// IsNotExist returns true if the error is caused by a non existing file.
func (b *Local) IsNotExist(err error) bool {
return os.IsNotExist(errors.Cause(err))
}
// Save stores data in the backend at the handle.
2017-06-03 17:39:57 +02:00
func (b *Local) Save(ctx context.Context, h restic.Handle, rd io.Reader) (err error) {
debug.Log("Save %v", h)
if err := h.Valid(); err != nil {
2016-01-24 16:59:38 +01:00
return err
}
filename := b.Filename(h)
2016-01-24 16:59:38 +01:00
// create new file
f, err := fs.OpenFile(filename, os.O_CREATE|os.O_EXCL|os.O_WRONLY, backend.Modes.File)
if err != nil {
return errors.Wrap(err, "OpenFile")
}
// save data, then sync
_, err = io.Copy(f, rd)
if err != nil {
2017-06-03 17:39:57 +02:00
_ = f.Close()
return errors.Wrap(err, "Write")
}
if err = f.Sync(); err != nil {
2017-06-03 17:39:57 +02:00
_ = f.Close()
return errors.Wrap(err, "Sync")
}
2016-01-24 16:59:38 +01:00
err = f.Close()
2016-01-24 16:59:38 +01:00
if err != nil {
return errors.Wrap(err, "Close")
2016-01-24 16:59:38 +01:00
}
// set mode to read-only
fi, err := fs.Stat(filename)
2016-01-24 16:59:38 +01:00
if err != nil {
2016-08-29 21:54:50 +02:00
return errors.Wrap(err, "Stat")
2016-01-24 16:59:38 +01:00
}
return setNewFileMode(filename, fi)
2016-01-24 01:15:35 +01:00
}
2017-01-23 18:11:10 +01:00
// Load returns a reader that yields the contents of the file at h at the
2017-01-22 22:01:12 +01:00
// given offset. If length is nonzero, only a portion of the file is
// returned. rd must be closed after use.
2017-06-03 17:39:57 +02:00
func (b *Local) Load(ctx context.Context, h restic.Handle, length int, offset int64) (io.ReadCloser, error) {
2017-01-23 18:11:10 +01:00
debug.Log("Load %v, length %v, offset %v", h, length, offset)
2017-01-22 22:01:12 +01:00
if err := h.Valid(); err != nil {
return nil, err
}
if offset < 0 {
return nil, errors.New("offset is negative")
}
f, err := fs.Open(b.Filename(h))
2017-01-22 22:01:12 +01:00
if err != nil {
return nil, err
}
if offset > 0 {
_, err = f.Seek(offset, 0)
if err != nil {
2017-06-03 17:39:57 +02:00
_ = f.Close()
2017-01-22 22:01:12 +01:00
return nil, err
}
}
if length > 0 {
return backend.LimitReadCloser(f, int64(length)), nil
}
return f, nil
}
2016-01-23 23:27:58 +01:00
// Stat returns information about a blob.
2017-06-03 17:39:57 +02:00
func (b *Local) Stat(ctx context.Context, h restic.Handle) (restic.FileInfo, error) {
2016-09-27 22:35:08 +02:00
debug.Log("Stat %v", h)
2016-01-23 23:27:58 +01:00
if err := h.Valid(); err != nil {
2016-08-31 22:39:36 +02:00
return restic.FileInfo{}, err
2016-01-23 23:27:58 +01:00
}
fi, err := fs.Stat(b.Filename(h))
2016-01-23 23:27:58 +01:00
if err != nil {
2016-08-31 22:39:36 +02:00
return restic.FileInfo{}, errors.Wrap(err, "Stat")
2016-01-23 23:27:58 +01:00
}
2016-08-31 22:39:36 +02:00
return restic.FileInfo{Size: fi.Size()}, nil
2016-01-23 23:27:58 +01:00
}
2015-03-28 11:50:23 +01:00
// Test returns true if a blob of the given type and name exists in the backend.
2017-06-03 17:39:57 +02:00
func (b *Local) Test(ctx context.Context, h restic.Handle) (bool, error) {
debug.Log("Test %v", h)
_, err := fs.Stat(b.Filename(h))
2015-03-28 11:50:23 +01:00
if err != nil {
if os.IsNotExist(errors.Cause(err)) {
2015-03-28 11:50:23 +01:00
return false, nil
}
2016-08-29 21:54:50 +02:00
return false, errors.Wrap(err, "Stat")
2015-03-28 11:50:23 +01:00
}
return true, nil
}
// Remove removes the blob with the given name and type.
2017-06-03 17:39:57 +02:00
func (b *Local) Remove(ctx context.Context, h restic.Handle) error {
debug.Log("Remove %v", h)
fn := b.Filename(h)
2015-08-19 22:02:47 +02:00
// reset read-only flag
err := fs.Chmod(fn, 0666)
2015-08-19 22:02:47 +02:00
if err != nil {
2016-08-29 21:54:50 +02:00
return errors.Wrap(err, "Chmod")
2015-08-19 22:02:47 +02:00
}
return fs.Remove(fn)
2015-03-28 11:50:23 +01:00
}
func isFile(fi os.FileInfo) bool {
return fi.Mode()&(os.ModeType|os.ModeCharDevice) == 0
}
2015-03-28 11:50:23 +01:00
// List returns a channel that yields all names of blobs of type t. A
2017-06-03 17:39:57 +02:00
// goroutine is started for this.
func (b *Local) List(ctx context.Context, t restic.FileType) <-chan string {
2016-09-27 22:35:08 +02:00
debug.Log("List %v", t)
2015-03-28 11:50:23 +01:00
ch := make(chan string)
go func() {
defer close(ch)
2017-04-10 23:31:13 +02:00
fs.Walk(b.Basedir(t), func(path string, fi os.FileInfo, err error) error {
if err != nil {
return err
}
2017-04-10 23:31:13 +02:00
if !isFile(fi) {
return err
2015-03-28 11:50:23 +01:00
}
select {
2017-04-10 23:31:13 +02:00
case ch <- filepath.Base(path):
2017-06-03 17:39:57 +02:00
case <-ctx.Done():
2017-04-10 23:31:13 +02:00
return err
2015-03-28 11:50:23 +01:00
}
2017-04-10 23:31:13 +02:00
return err
})
2015-03-28 11:50:23 +01:00
}()
return ch
}
// Delete removes the repository and all files.
func (b *Local) Delete() error {
2016-09-27 22:35:08 +02:00
debug.Log("Delete()")
2017-03-25 13:20:03 +01:00
return fs.RemoveAll(b.Path)
}
2015-03-28 11:50:23 +01:00
// Close closes all open files.
func (b *Local) Close() error {
2016-09-27 22:35:08 +02:00
debug.Log("Close()")
// this does not need to do anything, all open files are closed within the
// same function.
return nil
}