2018-12-26 14:45:44 +08:00
|
|
|
package filesys
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-09-24 18:06:44 +08:00
|
|
|
"github.com/chrislusf/seaweedfs/weed/util"
|
2018-12-26 14:45:44 +08:00
|
|
|
"os"
|
|
|
|
"syscall"
|
|
|
|
"time"
|
|
|
|
|
2020-09-01 15:21:19 +08:00
|
|
|
"github.com/chrislusf/seaweedfs/weed/filer"
|
2018-12-26 14:45:44 +08:00
|
|
|
"github.com/chrislusf/seaweedfs/weed/glog"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/pb/filer_pb"
|
2019-01-17 09:17:19 +08:00
|
|
|
"github.com/seaweedfs/fuse"
|
|
|
|
"github.com/seaweedfs/fuse/fs"
|
2018-12-26 14:45:44 +08:00
|
|
|
)
|
|
|
|
|
2020-09-24 18:06:44 +08:00
|
|
|
var _ = fs.NodeLinker(&Dir{})
|
2018-12-26 14:45:44 +08:00
|
|
|
var _ = fs.NodeSymlinker(&Dir{})
|
|
|
|
var _ = fs.NodeReadlinker(&File{})
|
|
|
|
|
2020-09-24 18:06:44 +08:00
|
|
|
func (dir *Dir) Link(ctx context.Context, req *fuse.LinkRequest, old fs.Node) (fs.Node, error) {
|
|
|
|
|
|
|
|
oldFile, ok := old.(*File)
|
|
|
|
if !ok {
|
|
|
|
glog.Errorf("old node is not a file: %+v", old)
|
|
|
|
}
|
|
|
|
|
|
|
|
glog.V(4).Infof("Link: %v/%v -> %v/%v", oldFile.dir.FullPath(), oldFile.Name, dir.FullPath(), req.NewName)
|
|
|
|
|
|
|
|
if err := oldFile.maybeLoadEntry(ctx); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// update old file to hardlink mode
|
|
|
|
var updateOldEntryRequest *filer_pb.UpdateEntryRequest
|
|
|
|
var hardLinkId filer.HardLinkId
|
|
|
|
if oldFile.entry.HardLinkId != 0 {
|
|
|
|
hardLinkId = filer.HardLinkId(oldFile.entry.HardLinkId)
|
|
|
|
} else {
|
|
|
|
// CreateLink 1.1 : split source entry into hardlink+empty_entry
|
|
|
|
hardLinkId = filer.HardLinkId(util.RandomInt64())
|
|
|
|
updateOldEntryRequest = &filer_pb.UpdateEntryRequest{
|
|
|
|
Directory: oldFile.dir.FullPath(),
|
|
|
|
Entry: &filer_pb.Entry{
|
|
|
|
Name: oldFile.entry.Name,
|
|
|
|
IsDirectory: oldFile.entry.IsDirectory,
|
|
|
|
HardLinkId: int64(hardLinkId),
|
|
|
|
},
|
|
|
|
Signatures: []int32{dir.wfs.signature},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateLink 1.2 : update new file to hardlink mode
|
|
|
|
request := &filer_pb.CreateEntryRequest{
|
|
|
|
Directory: dir.FullPath(),
|
|
|
|
Entry: &filer_pb.Entry{
|
|
|
|
Name: req.NewName,
|
|
|
|
IsDirectory: false,
|
|
|
|
HardLinkId: int64(hardLinkId),
|
|
|
|
},
|
|
|
|
Signatures: []int32{dir.wfs.signature},
|
|
|
|
}
|
|
|
|
|
|
|
|
// apply changes to the filer, and also apply to local metaCache
|
|
|
|
err := dir.wfs.WithFilerClient(func(client filer_pb.SeaweedFilerClient) error {
|
|
|
|
|
|
|
|
dir.wfs.mapPbIdFromLocalToFiler(request.Entry)
|
|
|
|
defer dir.wfs.mapPbIdFromFilerToLocal(request.Entry)
|
|
|
|
|
|
|
|
if updateOldEntryRequest != nil {
|
|
|
|
if err := filer_pb.UpdateEntry(client, updateOldEntryRequest); err != nil {
|
|
|
|
glog.V(0).Infof("Link %v/%v -> %s/%s: %v", oldFile.dir.FullPath(), oldFile.Name, dir.FullPath(), req.NewName, err)
|
|
|
|
return fuse.EIO
|
|
|
|
}
|
|
|
|
dir.wfs.metaCache.UpdateEntry(context.Background(), filer.FromPbEntry(updateOldEntryRequest.Directory, updateOldEntryRequest.Entry))
|
|
|
|
oldFile.entry.HardLinkId = int64(hardLinkId)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := filer_pb.CreateEntry(client, request); err != nil {
|
|
|
|
glog.V(0).Infof("Link %v/%v -> %s/%s: %v", oldFile.dir.FullPath(), oldFile.Name, dir.FullPath(), req.NewName, err)
|
|
|
|
return fuse.EIO
|
|
|
|
}
|
|
|
|
dir.wfs.metaCache.InsertEntry(context.Background(), filer.FromPbEntry(request.Directory, request.Entry))
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
// create new file node
|
|
|
|
newNode := dir.newFile(req.NewName, request.Entry)
|
|
|
|
newFile := newNode.(*File)
|
|
|
|
if err := newFile.maybeLoadEntry(ctx); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return newFile, err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-12-26 14:45:44 +08:00
|
|
|
func (dir *Dir) Symlink(ctx context.Context, req *fuse.SymlinkRequest) (fs.Node, error) {
|
|
|
|
|
2020-08-14 15:22:21 +08:00
|
|
|
glog.V(4).Infof("Symlink: %v/%v to %v", dir.FullPath(), req.NewName, req.Target)
|
2018-12-26 14:45:44 +08:00
|
|
|
|
|
|
|
request := &filer_pb.CreateEntryRequest{
|
2020-03-26 15:08:14 +08:00
|
|
|
Directory: dir.FullPath(),
|
2018-12-26 14:45:44 +08:00
|
|
|
Entry: &filer_pb.Entry{
|
|
|
|
Name: req.NewName,
|
|
|
|
IsDirectory: false,
|
|
|
|
Attributes: &filer_pb.FuseAttributes{
|
|
|
|
Mtime: time.Now().Unix(),
|
|
|
|
Crtime: time.Now().Unix(),
|
2019-07-24 15:03:05 +08:00
|
|
|
FileMode: uint32((os.FileMode(0777) | os.ModeSymlink) &^ dir.wfs.option.Umask),
|
2018-12-26 14:45:44 +08:00
|
|
|
Uid: req.Uid,
|
|
|
|
Gid: req.Gid,
|
|
|
|
SymlinkTarget: req.Target,
|
|
|
|
},
|
|
|
|
},
|
2020-08-29 14:48:48 +08:00
|
|
|
Signatures: []int32{dir.wfs.signature},
|
2018-12-26 14:45:44 +08:00
|
|
|
}
|
|
|
|
|
2020-02-26 13:50:12 +08:00
|
|
|
err := dir.wfs.WithFilerClient(func(client filer_pb.SeaweedFilerClient) error {
|
2020-09-03 15:07:22 +08:00
|
|
|
|
|
|
|
dir.wfs.mapPbIdFromLocalToFiler(request.Entry)
|
|
|
|
defer dir.wfs.mapPbIdFromFilerToLocal(request.Entry)
|
|
|
|
|
2020-02-26 13:50:12 +08:00
|
|
|
if err := filer_pb.CreateEntry(client, request); err != nil {
|
2020-03-26 15:08:14 +08:00
|
|
|
glog.V(0).Infof("symlink %s/%s: %v", dir.FullPath(), req.NewName, err)
|
2018-12-26 14:45:44 +08:00
|
|
|
return fuse.EIO
|
|
|
|
}
|
2020-04-23 06:40:47 +08:00
|
|
|
|
2020-09-01 15:21:19 +08:00
|
|
|
dir.wfs.metaCache.InsertEntry(context.Background(), filer.FromPbEntry(request.Directory, request.Entry))
|
2020-04-23 06:40:47 +08:00
|
|
|
|
2018-12-26 14:45:44 +08:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
symlink := dir.newFile(req.NewName, request.Entry)
|
|
|
|
|
|
|
|
return symlink, err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func (file *File) Readlink(ctx context.Context, req *fuse.ReadlinkRequest) (string, error) {
|
|
|
|
|
2019-12-16 13:07:01 +08:00
|
|
|
if err := file.maybeLoadEntry(ctx); err != nil {
|
2018-12-26 14:45:44 +08:00
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
if os.FileMode(file.entry.Attributes.FileMode)&os.ModeSymlink == 0 {
|
|
|
|
return "", fuse.Errno(syscall.EINVAL)
|
|
|
|
}
|
|
|
|
|
2020-08-14 15:22:21 +08:00
|
|
|
glog.V(4).Infof("Readlink: %v/%v => %v", file.dir.FullPath(), file.Name, file.entry.Attributes.SymlinkTarget)
|
2018-12-26 14:45:44 +08:00
|
|
|
|
|
|
|
return file.entry.Attributes.SymlinkTarget, nil
|
|
|
|
|
|
|
|
}
|