2
2
mirror of https://github.com/octoleo/restic.git synced 2024-12-11 21:57:58 +00:00
restic/internal/fuse/file.go

171 lines
4.3 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"
"sort"
"github.com/restic/restic/internal/debug"
"github.com/restic/restic/internal/errors"
2017-07-24 15:42:25 +00:00
"github.com/restic/restic/internal/restic"
"github.com/anacrolix/fuse"
"github.com/anacrolix/fuse/fs"
2015-07-19 12:28:11 +00:00
)
// The default block size to report in stat
const blockSize = 512
// Statically ensure that *file and *openFile implement the given interfaces
var _ = fs.HandleReader(&openFile{})
var _ = fs.NodeListxattrer(&file{})
var _ = fs.NodeGetxattrer(&file{})
var _ = fs.NodeOpener(&file{})
2015-07-19 12:28:11 +00:00
type file struct {
2017-06-18 13:11:32 +00:00
root *Root
node *restic.Node
inode uint64
}
2015-07-19 12:28:11 +00:00
type openFile struct {
file
// cumsize[i] holds the cumulative size of blobs[:i].
cumsize []uint64
2015-07-19 12:28:11 +00:00
}
2022-08-19 18:29:33 +00:00
func newFile(root *Root, inode uint64, node *restic.Node) (fusefile *file, err error) {
2016-09-27 20:35:08 +00:00
debug.Log("create new file for %v with %d blobs", node.Name, len(node.Content))
2015-07-19 12:28:11 +00:00
return &file{
2017-06-18 14:28:55 +00:00
inode: inode,
2017-06-18 13:11:32 +00:00
root: root,
node: node,
2015-07-19 12:28:11 +00:00
}, nil
}
func (f *file) Attr(_ context.Context, a *fuse.Attr) error {
2016-09-27 20:35:08 +00:00
debug.Log("Attr(%v)", f.node.Name)
2017-06-18 13:11:32 +00:00
a.Inode = f.inode
2015-07-19 12:28:11 +00:00
a.Mode = f.node.Mode
a.Size = f.node.Size
a.Blocks = (f.node.Size + blockSize - 1) / blockSize
a.BlockSize = blockSize
2017-02-10 20:16:48 +00:00
a.Nlink = uint32(f.node.Links)
2017-06-18 13:11:32 +00:00
if !f.root.cfg.OwnerIsRoot {
a.Uid = f.node.UID
a.Gid = f.node.GID
}
2015-07-21 20:11:30 +00:00
a.Atime = f.node.AccessTime
a.Ctime = f.node.ChangeTime
a.Mtime = f.node.ModTime
2017-02-11 20:50:03 +00:00
2015-07-19 12:28:11 +00:00
return nil
2017-02-11 20:50:03 +00:00
2015-07-19 12:28:11 +00:00
}
func (f *file) Open(_ context.Context, _ *fuse.OpenRequest, _ *fuse.OpenResponse) (fs.Handle, error) {
debug.Log("open file %v with %d blobs", f.node.Name, len(f.node.Content))
var bytes uint64
cumsize := make([]uint64, 1+len(f.node.Content))
for i, id := range f.node.Content {
size, found := f.root.repo.LookupBlobSize(id, restic.DataBlob)
if !found {
return nil, errors.Errorf("id %v not found in repository", id)
}
bytes += uint64(size)
cumsize[i+1] = bytes
}
var of = openFile{file: *f}
if bytes != f.node.Size {
debug.Log("sizes do not match: node.Size %v != size %v, using real size", f.node.Size, bytes)
// Make a copy of the node with correct size
nodenew := *f.node
nodenew.Size = bytes
of.file.node = &nodenew
}
of.cumsize = cumsize
return &of, nil
}
func (f *openFile) getBlobAt(ctx context.Context, i int) (blob []byte, err error) {
blob, err = f.root.blobCache.GetOrCompute(f.node.Content[i], func() ([]byte, error) {
return f.root.repo.LoadBlob(ctx, restic.DataBlob, f.node.Content[i], nil)
})
if err != nil {
2016-09-27 20:35:08 +00:00
debug.Log("LoadBlob(%v, %v) failed: %v", f.node.Name, f.node.Content[i], err)
return nil, unwrapCtxCanceled(err)
}
return blob, nil
2015-07-19 12:28:11 +00:00
}
func (f *openFile) Read(ctx context.Context, req *fuse.ReadRequest, resp *fuse.ReadResponse) error {
2016-09-27 20:35:08 +00:00
debug.Log("Read(%v, %v, %v), file size %v", f.node.Name, req.Size, req.Offset, f.node.Size)
offset := uint64(req.Offset)
2015-07-19 12:28:11 +00:00
// as stated in https://godoc.org/bazil.org/fuse/fs#HandleReader there
// is no need to check if offset > size
2016-07-29 18:55:09 +00:00
2016-07-29 19:18:32 +00:00
// handle special case: file is empty
if f.node.Size == 0 {
resp.Data = resp.Data[:0]
return nil
}
2015-07-19 12:28:11 +00:00
// Skip blobs before the offset
startContent := -1 + sort.Search(len(f.cumsize), func(i int) bool {
return f.cumsize[i] > offset
})
offset -= f.cumsize[startContent]
2015-07-19 12:28:11 +00:00
dst := resp.Data[0:req.Size]
readBytes := 0
remainingBytes := req.Size
// The documentation of bazil/fuse actually says that synchronization is
// required (see https://godoc.org/bazil.org/fuse#hdr-Service_Methods):
//
// Multiple goroutines may call service methods simultaneously;
// the methods being called are responsible for appropriate synchronization.
//
2023-12-06 12:11:55 +00:00
// However, no lock needed here as getBlobAt can be called concurrently
2023-10-26 06:27:19 +00:00
// (blobCache has its own locking)
for i := startContent; remainingBytes > 0 && i < len(f.cumsize)-1; i++ {
2017-06-04 09:16:55 +00:00
blob, err := f.getBlobAt(ctx, i)
2015-07-19 12:28:11 +00:00
if err != nil {
return err
}
if offset > 0 {
2017-05-16 23:28:39 +00:00
blob = blob[offset:]
offset = 0
2015-07-19 12:28:11 +00:00
}
copied := copy(dst, blob)
remainingBytes -= copied
readBytes += copied
dst = dst[copied:]
}
resp.Data = resp.Data[:readBytes]
return nil
}
func (f *file) Listxattr(_ context.Context, req *fuse.ListxattrRequest, resp *fuse.ListxattrResponse) error {
2023-07-08 15:41:45 +00:00
nodeToXattrList(f.node, req, resp)
return nil
}
func (f *file) Getxattr(_ context.Context, req *fuse.GetxattrRequest, resp *fuse.GetxattrResponse) error {
2023-07-08 15:41:45 +00:00
return nodeGetXattr(f.node, req, resp)
}