restic/src/restic/backend/local/local.go

348 lines
7.0 KiB
Go
Raw Normal View History

2015-03-28 10:50:23 +00:00
package local
import (
"errors"
"fmt"
"io"
"io/ioutil"
"os"
"path/filepath"
"restic/backend"
"restic/debug"
"restic/fs"
2015-03-28 10:50:23 +00:00
)
2016-01-24 19:23:50 +00:00
// Local is a backend in a local directory.
2015-03-28 10:50:23 +00:00
type Local struct {
p string
2015-03-28 10:50:23 +00:00
}
2016-01-26 21:09:29 +00:00
func paths(dir string) []string {
return []string{
2015-03-28 10:50:23 +00:00
dir,
filepath.Join(dir, backend.Paths.Data),
filepath.Join(dir, backend.Paths.Snapshots),
2015-04-26 13:48:35 +00:00
filepath.Join(dir, backend.Paths.Index),
2015-03-28 10:50:23 +00:00
filepath.Join(dir, backend.Paths.Locks),
filepath.Join(dir, backend.Paths.Keys),
filepath.Join(dir, backend.Paths.Temp),
}
2016-01-26 21:09:29 +00:00
}
2015-03-28 10:50:23 +00:00
2016-01-26 21:09:29 +00:00
// Open opens the local backend as specified by config.
func Open(dir string) (*Local, error) {
// test if all necessary dirs are there
2016-01-26 21:09:29 +00:00
for _, d := range paths(dir) {
2015-03-28 10:50:23 +00:00
if _, err := os.Stat(d); err != nil {
return nil, fmt.Errorf("%s does not exist", d)
}
}
return &Local{p: dir}, nil
2015-03-28 10:50:23 +00:00
}
// Create creates all the necessary files and directories for a new local
2015-05-04 18:39:45 +00:00
// backend at dir. Afterwards a new config blob should be created.
2015-03-28 10:50:23 +00:00
func Create(dir string) (*Local, error) {
2015-05-04 18:39:45 +00:00
// test if config file already exists
2015-08-26 20:06:52 +00:00
_, err := os.Lstat(filepath.Join(dir, backend.Paths.Config))
2015-03-28 10:50:23 +00:00
if err == nil {
return nil, errors.New("config file already exists")
2015-03-28 10:50:23 +00:00
}
// create paths for data, refs and temp
2016-01-26 21:09:29 +00:00
for _, d := range paths(dir) {
2015-03-28 10:50:23 +00:00
err := os.MkdirAll(d, backend.Modes.Dir)
if err != nil {
return nil, err
}
}
// open backend
return Open(dir)
}
// Location returns this backend's location (the directory name).
func (b *Local) Location() string {
return b.p
}
// Construct path for given Type and name.
func filename(base string, t backend.Type, name string) string {
if t == backend.Config {
return filepath.Join(base, "config")
}
2015-03-28 10:50:23 +00:00
return filepath.Join(dirname(base, t, name), name)
}
// Construct directory for given Type.
func dirname(base string, t backend.Type, name string) string {
var n string
switch t {
case backend.Data:
n = backend.Paths.Data
if len(name) > 2 {
n = filepath.Join(n, name[:2])
}
case backend.Snapshot:
n = backend.Paths.Snapshots
2015-04-26 13:48:35 +00:00
case backend.Index:
n = backend.Paths.Index
2015-03-28 10:50:23 +00:00
case backend.Lock:
n = backend.Paths.Locks
case backend.Key:
n = backend.Paths.Keys
}
return filepath.Join(base, n)
}
2016-01-23 13:12:12 +00:00
// Load returns the data stored in the backend for h at the given offset
// and saves it in p. Load has the same semantics as io.ReaderAt.
func (b *Local) Load(h backend.Handle, p []byte, off int64) (n int, err error) {
2016-01-23 16:08:03 +00:00
if err := h.Valid(); err != nil {
return 0, err
}
2016-01-23 13:12:12 +00:00
f, err := os.Open(filename(b.p, h.Type, h.Name))
if err != nil {
return 0, err
}
defer func() {
e := f.Close()
if err == nil && e != nil {
err = e
}
}()
if off > 0 {
_, err = f.Seek(off, 0)
if err != nil {
return 0, err
}
}
return io.ReadFull(f, p)
}
// writeToTempfile saves p into a tempfile in tempdir.
func writeToTempfile(tempdir string, p []byte) (filename string, err error) {
tmpfile, err := ioutil.TempFile(tempdir, "temp-")
2016-01-24 00:15:35 +00:00
if err != nil {
return "", err
2016-01-24 00:15:35 +00:00
}
2016-01-24 15:59:38 +00:00
n, err := tmpfile.Write(p)
2016-01-24 00:15:35 +00:00
if err != nil {
return "", err
2016-01-24 00:15:35 +00:00
}
if n != len(p) {
return "", errors.New("not all bytes writen")
2016-01-24 00:15:35 +00:00
}
2016-01-24 15:59:38 +00:00
if err = tmpfile.Sync(); err != nil {
return "", err
2016-01-24 15:59:38 +00:00
}
err = fs.ClearCache(tmpfile)
if err != nil {
return "", err
}
2016-01-24 15:59:38 +00:00
err = tmpfile.Close()
if err != nil {
return "", err
}
return tmpfile.Name(), nil
}
// Save stores data in the backend at the handle.
func (b *Local) Save(h backend.Handle, p []byte) (err error) {
if err := h.Valid(); err != nil {
2016-01-24 15:59:38 +00:00
return err
}
tmpfile, err := writeToTempfile(filepath.Join(b.p, backend.Paths.Temp), p)
debug.Log("local.Save", "saved %v (%d bytes) to %v", h, len(p), tmpfile)
filename := filename(b.p, h.Type, h.Name)
2016-01-24 15:59:38 +00:00
// test if new path already exists
if _, err := os.Stat(filename); err == nil {
return fmt.Errorf("Rename(): file %v already exists", filename)
2016-01-24 15:59:38 +00:00
}
// create directories if necessary, ignore errors
if h.Type == backend.Data {
err = os.MkdirAll(filepath.Dir(filename), backend.Modes.Dir)
if err != nil {
return err
}
}
err = os.Rename(tmpfile, filename)
2016-01-24 15:59:38 +00:00
debug.Log("local.Save", "save %v: rename %v -> %v: %v",
h, filepath.Base(tmpfile), filepath.Base(filename), err)
2016-01-24 15:59:38 +00:00
if err != nil {
return err
}
// set mode to read-only
fi, err := os.Stat(filename)
2016-01-24 15:59:38 +00:00
if err != nil {
return err
}
return setNewFileMode(filename, fi)
2016-01-24 00:15:35 +00:00
}
2016-01-23 22:27:58 +00:00
// Stat returns information about a blob.
func (b *Local) Stat(h backend.Handle) (backend.BlobInfo, error) {
if err := h.Valid(); err != nil {
return backend.BlobInfo{}, err
}
fi, err := os.Stat(filename(b.p, h.Type, h.Name))
if err != nil {
return backend.BlobInfo{}, err
}
return backend.BlobInfo{Size: fi.Size()}, nil
}
2015-03-28 10:50:23 +00:00
// Test returns true if a blob of the given type and name exists in the backend.
func (b *Local) Test(t backend.Type, name string) (bool, error) {
_, err := os.Stat(filename(b.p, t, name))
if err != nil {
if os.IsNotExist(err) {
return false, nil
}
return false, err
}
return true, nil
}
// Remove removes the blob with the given name and type.
func (b *Local) Remove(t backend.Type, name string) error {
fn := filename(b.p, t, name)
2015-08-19 20:02:47 +00:00
// reset read-only flag
err := os.Chmod(fn, 0666)
if err != nil {
return err
}
return os.Remove(fn)
2015-03-28 10:50:23 +00:00
}
func isFile(fi os.FileInfo) bool {
return fi.Mode()&(os.ModeType|os.ModeCharDevice) == 0
}
func readdir(d string) (fileInfos []os.FileInfo, err error) {
f, e := os.Open(d)
if e != nil {
return nil, e
}
defer func() {
e := f.Close()
if err == nil {
err = e
}
}()
return f.Readdir(-1)
}
// listDir returns a list of all files in d.
func listDir(d string) (filenames []string, err error) {
fileInfos, err := readdir(d)
if err != nil {
return nil, err
}
for _, fi := range fileInfos {
if isFile(fi) {
filenames = append(filenames, fi.Name())
}
}
return filenames, nil
}
// listDirs returns a list of all files in directories within d.
func listDirs(dir string) (filenames []string, err error) {
fileInfos, err := readdir(dir)
if err != nil {
return nil, err
}
for _, fi := range fileInfos {
if !fi.IsDir() {
continue
}
files, err := listDir(filepath.Join(dir, fi.Name()))
if err != nil {
continue
}
filenames = append(filenames, files...)
}
return filenames, nil
}
2015-03-28 10:50:23 +00:00
// List returns a channel that yields all names of blobs of type t. A
2015-06-28 07:44:06 +00:00
// goroutine is started for this. If the channel done is closed, sending
2015-03-28 10:50:23 +00:00
// stops.
func (b *Local) List(t backend.Type, done <-chan struct{}) <-chan string {
lister := listDir
2015-04-29 20:30:00 +00:00
if t == backend.Data {
lister = listDirs
2015-03-28 10:50:23 +00:00
}
ch := make(chan string)
items, err := lister(filepath.Join(dirname(b.p, t, "")))
2015-03-28 10:50:23 +00:00
if err != nil {
close(ch)
return ch
}
go func() {
defer close(ch)
for _, m := range items {
2015-03-28 10:50:23 +00:00
if m == "" {
continue
}
select {
case ch <- m:
case <-done:
return
}
}
}()
return ch
}
// Delete removes the repository and all files.
func (b *Local) Delete() error {
return os.RemoveAll(b.p)
}
2015-03-28 10:50:23 +00:00
// Close closes all open files.
func (b *Local) Close() error {
// this does not need to do anything, all open files are closed within the
// same function.
return nil
}