restic/src/restic/fuse/file.go

174 lines
3.8 KiB
Go
Raw Normal View History

2015-08-16 22:27:07 +02:00
// +build !openbsd
2015-08-17 19:40:34 +02:00
// +build !windows
2015-08-16 22:27:07 +02:00
2015-07-19 14:28:11 +02:00
package fuse
import (
2016-07-29 20:55:09 +02:00
"errors"
"sync"
"restic"
"restic/backend"
2016-05-08 22:20:46 +02:00
"restic/debug"
"restic/pack"
2015-07-19 14:28:11 +02:00
"bazil.org/fuse"
"bazil.org/fuse/fs"
"golang.org/x/net/context"
)
// The default block size to report in stat
const blockSize = 512
2015-07-19 14:28:11 +02:00
// Statically ensure that *file implements the given interface
var _ = fs.HandleReader(&file{})
var _ = fs.HandleReleaser(&file{})
2015-07-19 14:28:11 +02:00
// BlobLoader is an abstracted repository with a reduced set of methods used
// for fuse operations.
type BlobLoader interface {
LookupBlobSize(backend.ID) (uint, error)
LoadBlob(pack.BlobType, backend.ID, []byte) ([]byte, error)
}
2015-07-19 14:28:11 +02:00
type file struct {
repo BlobLoader
node *restic.Node
ownerIsRoot bool
2015-07-19 14:28:11 +02:00
sizes []uint
2015-07-19 14:28:11 +02:00
blobs [][]byte
}
const defaultBlobSize = 128 * 1024
var blobPool = sync.Pool{
New: func() interface{} {
return make([]byte, defaultBlobSize)
},
}
func newFile(repo BlobLoader, node *restic.Node, ownerIsRoot bool) (*file, error) {
2016-05-08 22:20:46 +02:00
debug.Log("newFile", "create new file for %v with %d blobs", node.Name, len(node.Content))
var bytes uint64
sizes := make([]uint, len(node.Content))
for i, id := range node.Content {
size, err := repo.LookupBlobSize(id)
2015-07-19 14:28:11 +02:00
if err != nil {
return nil, err
}
sizes[i] = size
bytes += uint64(size)
}
if bytes != node.Size {
debug.Log("newFile", "sizes do not match: node.Size %v != size %v, using real size", node.Size, bytes)
node.Size = bytes
2015-07-19 14:28:11 +02:00
}
return &file{
repo: repo,
node: node,
sizes: sizes,
blobs: make([][]byte, len(node.Content)),
ownerIsRoot: ownerIsRoot,
2015-07-19 14:28:11 +02:00
}, nil
}
func (f *file) Attr(ctx context.Context, a *fuse.Attr) error {
2016-05-08 22:20:46 +02:00
debug.Log("file.Attr", "Attr(%v)", f.node.Name)
2015-07-19 14:28:11 +02:00
a.Inode = f.node.Inode
a.Mode = f.node.Mode
a.Size = f.node.Size
a.Blocks = (f.node.Size / blockSize) + 1
a.BlockSize = blockSize
if !f.ownerIsRoot {
a.Uid = f.node.UID
a.Gid = f.node.GID
}
2015-07-21 22:11:30 +02:00
a.Atime = f.node.AccessTime
a.Ctime = f.node.ChangeTime
a.Mtime = f.node.ModTime
2015-07-19 14:28:11 +02:00
return nil
}
func (f *file) getBlobAt(i int) (blob []byte, err error) {
2016-05-08 22:20:46 +02:00
debug.Log("file.getBlobAt", "getBlobAt(%v, %v)", f.node.Name, i)
2015-07-19 14:28:11 +02:00
if f.blobs[i] != nil {
return f.blobs[i], nil
}
buf := blobPool.Get().([]byte)
buf = buf[:cap(buf)]
if uint(len(buf)) < f.sizes[i] {
if len(buf) > defaultBlobSize {
blobPool.Put(buf)
2015-07-19 14:28:11 +02:00
}
buf = make([]byte, f.sizes[i])
2015-07-19 14:28:11 +02:00
}
blob, err = f.repo.LoadBlob(pack.Data, f.node.Content[i], buf)
if err != nil {
2016-05-08 22:20:46 +02:00
debug.Log("file.getBlobAt", "LoadBlob(%v, %v) failed: %v", f.node.Name, f.node.Content[i], err)
return nil, err
}
f.blobs[i] = blob
2015-07-19 14:28:11 +02:00
return blob, nil
}
func (f *file) Read(ctx context.Context, req *fuse.ReadRequest, resp *fuse.ReadResponse) error {
2016-07-29 20:55:09 +02:00
debug.Log("file.Read", "Read(%v, %v, %v), file size %v", f.node.Name, req.Size, req.Offset, f.node.Size)
offset := req.Offset
2015-07-19 14:28:11 +02:00
2016-07-29 21:05:36 +02:00
if uint64(offset) > f.node.Size {
debug.Log("file.Read", "Read(%v): offset is greater than file size: %v > %v",
f.node.Name, req.Offset, f.node.Size)
2016-07-29 20:55:09 +02:00
return errors.New("offset greater than files size")
}
2015-07-19 14:28:11 +02:00
// Skip blobs before the offset
startContent := 0
for offset > int64(f.sizes[startContent]) {
offset -= int64(f.sizes[startContent])
2015-07-19 14:28:11 +02:00
startContent++
}
dst := resp.Data[0:req.Size]
readBytes := 0
remainingBytes := req.Size
for i := startContent; remainingBytes > 0 && i < len(f.sizes); i++ {
2015-07-19 14:28:11 +02:00
blob, err := f.getBlobAt(i)
if err != nil {
return err
}
if offset > 0 {
blob = blob[offset:len(blob)]
offset = 0
2015-07-19 14:28:11 +02:00
}
copied := copy(dst, blob)
remainingBytes -= copied
readBytes += copied
dst = dst[copied:]
}
resp.Data = resp.Data[:readBytes]
return nil
}
func (f *file) Release(ctx context.Context, req *fuse.ReleaseRequest) error {
for i := range f.blobs {
if f.blobs[i] != nil {
blobPool.Put(f.blobs[i])
f.blobs[i] = nil
2015-07-19 14:28:11 +02:00
}
}
return nil
}