2015-07-19 12:28:11 +00:00
|
|
|
package fuse
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
|
|
|
|
|
|
|
"bazil.org/fuse"
|
|
|
|
"bazil.org/fuse/fs"
|
|
|
|
"golang.org/x/net/context"
|
|
|
|
|
|
|
|
"github.com/restic/restic"
|
|
|
|
"github.com/restic/restic/repository"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Statically ensure that *dir implement those interface
|
|
|
|
var _ = fs.HandleReadDirAller(&dir{})
|
|
|
|
var _ = fs.NodeStringLookuper(&dir{})
|
|
|
|
|
|
|
|
type dir struct {
|
|
|
|
repo *repository.Repository
|
|
|
|
children map[string]*restic.Node
|
|
|
|
inode uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
func newDir(repo *repository.Repository, node *restic.Node) (*dir, error) {
|
|
|
|
tree, err := restic.LoadTree(repo, node.Subtree)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
children := make(map[string]*restic.Node)
|
|
|
|
for _, child := range tree.Nodes {
|
|
|
|
children[child.Name] = child
|
|
|
|
}
|
|
|
|
|
|
|
|
return &dir{
|
|
|
|
repo: repo,
|
|
|
|
children: children,
|
|
|
|
inode: node.Inode,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func newDirFromSnapshot(repo *repository.Repository, snapshot SnapshotWithId) (*dir, error) {
|
|
|
|
tree, err := restic.LoadTree(repo, snapshot.Tree)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
children := make(map[string]*restic.Node)
|
|
|
|
for _, node := range tree.Nodes {
|
|
|
|
children[node.Name] = node
|
|
|
|
}
|
|
|
|
|
|
|
|
return &dir{
|
|
|
|
repo: repo,
|
|
|
|
children: children,
|
2015-07-19 14:39:17 +00:00
|
|
|
inode: inodeFromBackendId(snapshot.ID),
|
2015-07-19 12:28:11 +00:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *dir) Attr(ctx context.Context, a *fuse.Attr) error {
|
|
|
|
a.Inode = d.inode
|
|
|
|
a.Mode = os.ModeDir | 0555
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *dir) ReadDirAll(ctx context.Context) ([]fuse.Dirent, error) {
|
|
|
|
ret := make([]fuse.Dirent, 0, len(d.children))
|
|
|
|
|
|
|
|
for _, node := range d.children {
|
|
|
|
var typ fuse.DirentType
|
2015-07-21 19:23:40 +00:00
|
|
|
switch node.Type {
|
|
|
|
case "dir":
|
2015-07-19 12:28:11 +00:00
|
|
|
typ = fuse.DT_Dir
|
2015-07-21 19:23:40 +00:00
|
|
|
case "file":
|
2015-07-19 12:28:11 +00:00
|
|
|
typ = fuse.DT_File
|
2015-07-21 19:25:05 +00:00
|
|
|
case "symlink":
|
|
|
|
typ = fuse.DT_Link
|
2015-07-19 12:28:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = append(ret, fuse.Dirent{
|
|
|
|
Inode: node.Inode,
|
|
|
|
Type: typ,
|
|
|
|
Name: node.Name,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *dir) Lookup(ctx context.Context, name string) (fs.Node, error) {
|
|
|
|
child, ok := d.children[name]
|
|
|
|
if !ok {
|
|
|
|
return nil, fuse.ENOENT
|
|
|
|
}
|
2015-07-21 19:23:40 +00:00
|
|
|
switch child.Type {
|
|
|
|
case "dir":
|
2015-07-19 12:28:11 +00:00
|
|
|
return newDir(d.repo, child)
|
2015-07-21 19:23:40 +00:00
|
|
|
case "file":
|
2015-07-19 12:28:11 +00:00
|
|
|
return newFile(d.repo, child)
|
2015-07-21 19:25:05 +00:00
|
|
|
case "symlink":
|
|
|
|
return newLink(d.repo, child)
|
2015-07-19 12:28:11 +00:00
|
|
|
default:
|
|
|
|
return nil, fuse.ENOENT
|
|
|
|
}
|
|
|
|
}
|