v2api (go-fuse v2 api): initial noop implementation

Compiles and mounts but does nothing useful.
This commit is contained in:
Jakob Unterwurzacher 2020-06-11 22:20:15 +02:00
parent 3b61244b72
commit 2aad58f9ec
5 changed files with 95 additions and 29 deletions

View File

@ -1,3 +1,6 @@
// +build ignore
// TODO
package main package main
import ( import (

View File

@ -0,0 +1,50 @@
package fusefrontend
import (
"context"
"syscall"
"github.com/hanwen/go-fuse/v2/fs"
"github.com/hanwen/go-fuse/v2/fuse"
"github.com/rfjakob/gocryptfs/internal/contentenc"
"github.com/rfjakob/gocryptfs/internal/nametransform"
)
// Node is a file or directory in the filesystem tree
// in a gocryptfs mount.
type Node struct {
fs.Inode
}
// RootNode is the root of the filesystem tree of Nodes.
type RootNode struct {
Node
// This flag is set to zero each time fs.isFiltered() is called
// (uint32 so that it can be reset with CompareAndSwapUint32).
// When -idle was used when mounting, idleMonitor() sets it to 1
// periodically.
IsIdle uint32
}
func NewRootNode(args Args, c *contentenc.ContentEnc, n nametransform.NameTransformer) *RootNode {
// TODO
return &RootNode{}
}
func (n *Node) path() string {
return n.Path(n.Root())
}
func (n *Node) Lookup(ctx context.Context, name string, out *fuse.EntryOut) (*fs.Inode, syscall.Errno) {
return nil, 1
}
func (n *Node) Getattr(ctx context.Context, f fs.FileHandle, out *fuse.AttrOut) syscall.Errno {
return 1
}
func (n *Node) Readdir(ctx context.Context) (fs.DirStream, syscall.Errno) {
return nil, 1
}

View File

@ -0,0 +1,20 @@
package fusefrontend_reverse
import (
"github.com/hanwen/go-fuse/v2/fs"
"github.com/rfjakob/gocryptfs/internal/contentenc"
"github.com/rfjakob/gocryptfs/internal/fusefrontend"
"github.com/rfjakob/gocryptfs/internal/nametransform"
)
// RNode is a file or directory in the filesystem tree
// in a `gocryptfs -reverse` mount.
type RNode struct {
fs.Inode
}
func NewRootNode(args fusefrontend.Args, c *contentenc.ContentEnc, n nametransform.NameTransformer) *RNode {
// TODO
return &RNode{}
}

View File

@ -313,7 +313,8 @@ func main() {
} }
// "-fsck" // "-fsck"
if args.fsck { if args.fsck {
fsck(&args) // TODO
//fsck(&args)
os.Exit(0) os.Exit(0)
} }
} }

View File

@ -22,8 +22,8 @@ import (
"golang.org/x/sys/unix" "golang.org/x/sys/unix"
"github.com/hanwen/go-fuse/v2/fs"
"github.com/hanwen/go-fuse/v2/fuse" "github.com/hanwen/go-fuse/v2/fuse"
"github.com/hanwen/go-fuse/v2/fuse/nodefs"
"github.com/hanwen/go-fuse/v2/fuse/pathfs" "github.com/hanwen/go-fuse/v2/fuse/pathfs"
"github.com/rfjakob/gocryptfs/internal/configfile" "github.com/rfjakob/gocryptfs/internal/configfile"
@ -162,7 +162,7 @@ func doMount(args *argContainer) {
// Set up autounmount, if requested. // Set up autounmount, if requested.
if args.idle > 0 && !args.reverse { if args.idle > 0 && !args.reverse {
// Not being in reverse mode means we always have a forward file system. // Not being in reverse mode means we always have a forward file system.
fwdFs := fs.(*fusefrontend.FS) fwdFs := fs.(*fusefrontend.RootNode)
go idleMonitor(args.idle, fwdFs, srv, args.mountpoint) go idleMonitor(args.idle, fwdFs, srv, args.mountpoint)
} }
// Jump into server loop. Returns when it gets an umount request from the kernel. // Jump into server loop. Returns when it gets an umount request from the kernel.
@ -175,7 +175,7 @@ func doMount(args *argContainer) {
// filesystem idleness and unmounts if we've been idle for long enough. // filesystem idleness and unmounts if we've been idle for long enough.
const checksDuringTimeoutPeriod = 4 const checksDuringTimeoutPeriod = 4
func idleMonitor(idleTimeout time.Duration, fs *fusefrontend.FS, srv *fuse.Server, mountpoint string) { func idleMonitor(idleTimeout time.Duration, fs *fusefrontend.RootNode, srv *fuse.Server, mountpoint string) {
sleepTimeBetweenChecks := contentenc.MinUint64( sleepTimeBetweenChecks := contentenc.MinUint64(
uint64(idleTimeout/checksDuringTimeoutPeriod), uint64(idleTimeout/checksDuringTimeoutPeriod),
uint64(2*time.Minute)) uint64(2*time.Minute))
@ -222,16 +222,9 @@ func setOpenFileLimit() {
} }
} }
// ctlsockFs satisfies both the pathfs.FileSystem and the ctlsocksrv.Interface
// interfaces
type ctlsockFs interface {
pathfs.FileSystem
ctlsocksrv.Interface
}
// initFuseFrontend - initialize gocryptfs/fusefrontend // initFuseFrontend - initialize gocryptfs/fusefrontend
// Calls os.Exit on errors // Calls os.Exit on errors
func initFuseFrontend(args *argContainer) (pfs pathfs.FileSystem, wipeKeys func()) { func initFuseFrontend(args *argContainer) (rootNode fs.InodeEmbedder, wipeKeys func()) {
var err error var err error
var confFile *configfile.ConfFile var confFile *configfile.ConfFile
// Get the masterkey from the command line if it was specified // Get the masterkey from the command line if it was specified
@ -318,25 +311,24 @@ func initFuseFrontend(args *argContainer) (pfs pathfs.FileSystem, wipeKeys func(
} }
masterkey = nil masterkey = nil
// Spawn fusefrontend // Spawn fusefrontend
var fs ctlsockFs
if args.reverse { if args.reverse {
if cryptoBackend != cryptocore.BackendAESSIV { if cryptoBackend != cryptocore.BackendAESSIV {
log.Panic("reverse mode must use AES-SIV, everything else is insecure") log.Panic("reverse mode must use AES-SIV, everything else is insecure")
} }
fs = fusefrontend_reverse.NewFS(frontendArgs, cEnc, nameTransform) rootNode = fusefrontend_reverse.NewRootNode(frontendArgs, cEnc, nameTransform)
} else { } else {
fs = fusefrontend.NewFS(frontendArgs, cEnc, nameTransform) rootNode = fusefrontend.NewRootNode(frontendArgs, cEnc, nameTransform)
} }
// We have opened the socket early so that we cannot fail here after // We have opened the socket early so that we cannot fail here after
// asking the user for the password // asking the user for the password
if args._ctlsockFd != nil { if args._ctlsockFd != nil {
go ctlsocksrv.Serve(args._ctlsockFd, fs) go ctlsocksrv.Serve(args._ctlsockFd, rootNode.(ctlsocksrv.Interface))
} }
return fs, func() { cCore.Wipe() } return rootNode, func() { cCore.Wipe() }
} }
func initGoFuse(fs pathfs.FileSystem, args *argContainer) *fuse.Server { func initGoFuse(rootNode fs.InodeEmbedder, args *argContainer) *fuse.Server {
// pathFsOpts are passed into go-fuse/pathfs // pathFsOpts are passed into go-fuse/pathfs
pathFsOpts := &pathfs.PathNodeFsOptions{ClientInodes: true} pathFsOpts := &pathfs.PathNodeFsOptions{ClientInodes: true}
if args.sharedstorage { if args.sharedstorage {
@ -351,23 +343,22 @@ func initGoFuse(fs pathfs.FileSystem, args *argContainer) *fuse.Server {
// inode numbers ( https://github.com/rfjakob/gocryptfs/issues/149 ). // inode numbers ( https://github.com/rfjakob/gocryptfs/issues/149 ).
pathFsOpts.ClientInodes = false pathFsOpts.ClientInodes = false
} }
pathFs := pathfs.NewPathNodeFs(fs, pathFsOpts) var fuseOpts *fs.Options
var fuseOpts *nodefs.Options sec := time.Second
if args.sharedstorage { if args.sharedstorage {
// sharedstorage mode sets all cache timeouts to zero so changes to the // sharedstorage mode sets all cache timeouts to zero so changes to the
// backing shared storage show up immediately. // backing shared storage show up immediately.
fuseOpts = &nodefs.Options{} fuseOpts = &fs.Options{}
} else { } else {
fuseOpts = &nodefs.Options{ fuseOpts = &fs.Options{
// These options are to be compatible with libfuse defaults, // These options are to be compatible with libfuse defaults,
// making benchmarking easier. // making benchmarking easier.
NegativeTimeout: time.Second, NegativeTimeout: &sec,
AttrTimeout: time.Second, AttrTimeout: &sec,
EntryTimeout: time.Second, EntryTimeout: &sec,
} }
} }
conn := nodefs.NewFileSystemConnector(pathFs.Root(), fuseOpts) fuseOpts.MountOptions = fuse.MountOptions{
mOpts := fuse.MountOptions{
// Writes and reads are usually capped at 128kiB on Linux through // Writes and reads are usually capped at 128kiB on Linux through
// the FUSE_MAX_PAGES_PER_REQ kernel constant in fuse_i.h. Our // the FUSE_MAX_PAGES_PER_REQ kernel constant in fuse_i.h. Our
// sync.Pool buffer pools are sized acc. to the default. Users may set // sync.Pool buffer pools are sized acc. to the default. Users may set
@ -377,6 +368,7 @@ func initGoFuse(fs pathfs.FileSystem, args *argContainer) *fuse.Server {
MaxWrite: fuse.MAX_KERNEL_WRITE, MaxWrite: fuse.MAX_KERNEL_WRITE,
Options: []string{fmt.Sprintf("max_read=%d", fuse.MAX_KERNEL_WRITE)}, Options: []string{fmt.Sprintf("max_read=%d", fuse.MAX_KERNEL_WRITE)},
} }
mOpts := &fuseOpts.MountOptions
if args.allow_other { if args.allow_other {
tlog.Info.Printf(tlog.ColorYellow + "The option \"-allow_other\" is set. Make sure the file " + tlog.Info.Printf(tlog.ColorYellow + "The option \"-allow_other\" is set. Make sure the file " +
"permissions protect your data from unwanted access." + tlog.ColorReset) "permissions protect your data from unwanted access." + tlog.ColorReset)
@ -446,9 +438,9 @@ func initGoFuse(fs pathfs.FileSystem, args *argContainer) *fuse.Server {
tlog.Debug.Printf("Adding -ko mount options: %v", parts) tlog.Debug.Printf("Adding -ko mount options: %v", parts)
mOpts.Options = append(mOpts.Options, parts...) mOpts.Options = append(mOpts.Options, parts...)
} }
srv, err := fuse.NewServer(conn.RawFS(), args.mountpoint, &mOpts) srv, err := fs.Mount(args.mountpoint, rootNode, fuseOpts)
if err != nil { if err != nil {
tlog.Fatal.Printf("fuse.NewServer failed: %s", strings.TrimSpace(err.Error())) tlog.Fatal.Printf("fs.Mount failed: %s", strings.TrimSpace(err.Error()))
if runtime.GOOS == "darwin" { if runtime.GOOS == "darwin" {
tlog.Info.Printf("Maybe you should run: /Library/Filesystems/osxfuse.fs/Contents/Resources/load_osxfuse") tlog.Info.Printf("Maybe you should run: /Library/Filesystems/osxfuse.fs/Contents/Resources/load_osxfuse")
} }