1
0
Fork 0
mirror of https://github.com/restic/restic.git synced 2024-12-23 00:07:25 +00:00
restic/internal/fuse/dir.go

245 lines
5.5 KiB
Go
Raw Normal View History

2022-03-28 20:23:47 +00:00
//go:build darwin || freebsd || linux
// +build darwin freebsd linux
2015-08-16 20:27:07 +00:00
2015-07-19 12:28:11 +00:00
package fuse
import (
"context"
"errors"
2015-07-19 12:28:11 +00:00
"os"
2017-12-13 19:21:18 +00:00
"path/filepath"
"sync"
"syscall"
2015-07-19 12:28:11 +00:00
"github.com/anacrolix/fuse"
"github.com/anacrolix/fuse/fs"
2015-07-19 12:28:11 +00:00
2017-07-23 12:21:03 +00:00
"github.com/restic/restic/internal/debug"
2017-07-24 15:42:25 +00:00
"github.com/restic/restic/internal/restic"
2015-07-19 12:28:11 +00:00
)
// Statically ensure that *dir implement those interface
var _ = fs.HandleReadDirAller(&dir{})
var _ = fs.NodeStringLookuper(&dir{})
type dir struct {
root *Root
items map[string]*restic.Node
inode uint64
parentInode uint64
node *restic.Node
m sync.Mutex
cache treeCache
2015-07-19 12:28:11 +00:00
}
2017-12-13 19:21:18 +00:00
func cleanupNodeName(name string) string {
return filepath.Base(name)
}
2022-08-19 18:29:33 +00:00
func newDir(root *Root, inode, parentInode uint64, node *restic.Node) (*dir, error) {
2018-01-25 19:49:41 +00:00
debug.Log("new dir for %v (%v)", node.Name, node.Subtree)
2015-07-19 12:28:11 +00:00
return &dir{
root: root,
node: node,
inode: inode,
parentInode: parentInode,
cache: *newTreeCache(),
2015-07-19 12:28:11 +00:00
}, nil
}
2023-12-06 12:11:55 +00:00
// returning a wrapped context.Canceled error will instead result in returning
// an input / output error to the user. Thus unwrap the error to match the
// expectations of bazil/fuse
func unwrapCtxCanceled(err error) error {
if errors.Is(err, context.Canceled) {
return context.Canceled
}
return err
}
// replaceSpecialNodes replaces nodes with name "." and "/" by their contents.
// Otherwise, the node is returned.
func replaceSpecialNodes(ctx context.Context, repo restic.BlobLoader, node *restic.Node) ([]*restic.Node, error) {
2024-07-09 17:51:44 +00:00
if node.Type != restic.NodeTypeDir || node.Subtree == nil {
return []*restic.Node{node}, nil
}
if node.Name != "." && node.Name != "/" {
return []*restic.Node{node}, nil
}
tree, err := restic.LoadTree(ctx, repo, *node.Subtree)
if err != nil {
return nil, unwrapCtxCanceled(err)
}
return tree.Nodes, nil
}
2022-08-19 18:29:33 +00:00
func newDirFromSnapshot(root *Root, inode uint64, snapshot *restic.Snapshot) (*dir, error) {
2018-01-25 19:49:41 +00:00
debug.Log("new dir for snapshot %v (%v)", snapshot.ID(), snapshot.Tree)
return &dir{
root: root,
node: &restic.Node{
AccessTime: snapshot.Time,
ModTime: snapshot.Time,
ChangeTime: snapshot.Time,
Mode: os.ModeDir | 0555,
Subtree: snapshot.Tree,
},
inode: inode,
cache: *newTreeCache(),
}, nil
}
func (d *dir) open(ctx context.Context) error {
d.m.Lock()
defer d.m.Unlock()
if d.items != nil {
return nil
}
debug.Log("open dir %v (%v)", d.node.Name, d.node.Subtree)
tree, err := restic.LoadTree(ctx, d.root.repo, *d.node.Subtree)
2015-07-19 12:28:11 +00:00
if err != nil {
debug.Log(" error loading tree %v: %v", d.node.Subtree, err)
return unwrapCtxCanceled(err)
2015-07-19 12:28:11 +00:00
}
items := make(map[string]*restic.Node)
for _, n := range tree.Nodes {
2024-07-31 17:30:47 +00:00
if ctx.Err() != nil {
return ctx.Err()
}
nodes, err := replaceSpecialNodes(ctx, d.root.repo, n)
if err != nil {
2016-09-27 20:35:08 +00:00
debug.Log(" replaceSpecialNodes(%v) failed: %v", n, err)
return err
}
for _, node := range nodes {
2017-12-13 19:21:18 +00:00
items[cleanupNodeName(node.Name)] = node
}
2015-07-19 12:28:11 +00:00
}
d.items = items
return nil
2015-07-19 12:28:11 +00:00
}
func (d *dir) Attr(_ context.Context, a *fuse.Attr) error {
2020-02-17 15:26:08 +00:00
debug.Log("Attr()")
2015-07-19 12:28:11 +00:00
a.Inode = d.inode
2015-07-21 20:11:30 +00:00
a.Mode = os.ModeDir | d.node.Mode
if !d.root.cfg.OwnerIsRoot {
a.Uid = d.node.UID
a.Gid = d.node.GID
}
2015-07-21 20:11:30 +00:00
a.Atime = d.node.AccessTime
a.Ctime = d.node.ChangeTime
a.Mtime = d.node.ModTime
2017-02-11 20:50:03 +00:00
a.Nlink = d.calcNumberOfLinks()
2015-07-19 12:28:11 +00:00
return nil
}
2017-02-11 20:50:03 +00:00
func (d *dir) calcNumberOfLinks() uint32 {
// a directory d has 2 hardlinks + the number
// of directories contained by d
count := uint32(2)
2017-02-11 20:50:03 +00:00
for _, node := range d.items {
2024-07-09 17:51:44 +00:00
if node.Type == restic.NodeTypeDir {
2017-02-11 20:50:03 +00:00
count++
}
}
return count
}
2015-07-19 12:28:11 +00:00
func (d *dir) ReadDirAll(ctx context.Context) ([]fuse.Dirent, error) {
2020-02-17 15:26:08 +00:00
debug.Log("ReadDirAll()")
err := d.open(ctx)
if err != nil {
return nil, err
}
ret := make([]fuse.Dirent, 0, len(d.items)+2)
ret = append(ret, fuse.Dirent{
Inode: d.inode,
Name: ".",
Type: fuse.DT_Dir,
})
ret = append(ret, fuse.Dirent{
Inode: d.parentInode,
Name: "..",
Type: fuse.DT_Dir,
})
2015-07-19 12:28:11 +00:00
for _, node := range d.items {
2024-07-31 17:30:47 +00:00
if ctx.Err() != nil {
return nil, ctx.Err()
}
2017-12-13 19:21:18 +00:00
name := cleanupNodeName(node.Name)
2015-07-19 12:28:11 +00:00
var typ fuse.DirentType
2016-09-01 19:20:03 +00:00
switch node.Type {
2024-07-09 17:51:44 +00:00
case restic.NodeTypeDir:
2015-07-19 12:28:11 +00:00
typ = fuse.DT_Dir
2024-07-09 17:51:44 +00:00
case restic.NodeTypeFile:
2015-07-19 12:28:11 +00:00
typ = fuse.DT_File
2024-07-09 17:51:44 +00:00
case restic.NodeTypeSymlink:
2015-07-21 19:25:05 +00:00
typ = fuse.DT_Link
2015-07-19 12:28:11 +00:00
}
ret = append(ret, fuse.Dirent{
Inode: inodeFromNode(d.inode, node),
2015-07-19 12:28:11 +00:00
Type: typ,
2017-12-13 19:21:18 +00:00
Name: name,
2015-07-19 12:28:11 +00:00
})
}
return ret, nil
}
func (d *dir) Lookup(ctx context.Context, name string) (fs.Node, error) {
2016-09-27 20:35:08 +00:00
debug.Log("Lookup(%v)", name)
err := d.open(ctx)
if err != nil {
return nil, err
}
return d.cache.lookupOrCreate(name, func() (fs.Node, error) {
node, ok := d.items[name]
if !ok {
debug.Log(" Lookup(%v) -> not found", name)
return nil, syscall.ENOENT
}
inode := inodeFromNode(d.inode, node)
switch node.Type {
case restic.NodeTypeDir:
return newDir(d.root, inode, d.inode, node)
case restic.NodeTypeFile:
return newFile(d.root, inode, node)
case restic.NodeTypeSymlink:
return newLink(d.root, inode, node)
case restic.NodeTypeDev, restic.NodeTypeCharDev, restic.NodeTypeFifo, restic.NodeTypeSocket:
return newOther(d.root, inode, node)
default:
debug.Log(" node %v has unknown type %v", name, node.Type)
return nil, syscall.ENOENT
}
})
2015-07-19 12:28:11 +00:00
}
func (d *dir) Listxattr(_ context.Context, req *fuse.ListxattrRequest, resp *fuse.ListxattrResponse) error {
2023-07-08 15:41:45 +00:00
nodeToXattrList(d.node, req, resp)
return nil
}
func (d *dir) Getxattr(_ context.Context, req *fuse.GetxattrRequest, resp *fuse.GetxattrResponse) error {
2023-07-08 15:41:45 +00:00
return nodeGetXattr(d.node, req, resp)
}