fusego/fuseutil/file_system.go

175 lines
4.6 KiB
Go
Raw Normal View History

// Copyright 2015 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package fuseutil
import (
2015-04-02 03:19:56 +03:00
"flag"
2015-03-25 01:24:44 +03:00
"io"
2015-04-02 03:19:56 +03:00
"math/rand"
"time"
2015-03-25 01:24:44 +03:00
"github.com/jacobsa/fuse"
"github.com/jacobsa/fuse/fuseops"
)
2015-04-02 03:20:53 +03:00
var fRandomDelays = flag.Bool(
"fuseutil.random_delays", false,
2015-04-02 03:19:56 +03:00
"If set, randomly delay each op received, to help expose concurrency issues.")
// An interface with a method for each op type in the fuseops package. This can
// be used in conjunction with NewFileSystemServer to avoid writing a "dispatch
// loop" that switches on op types, instead receiving typed method calls
// directly.
//
2015-06-05 07:13:59 +03:00
// The FileSystem implementation should not call Op.Respond, instead returning
// the error with which the caller should respond.
2015-03-25 01:17:58 +03:00
//
// See NotImplementedFileSystem for a convenient way to embed default
// implementations for methods you don't care about.
type FileSystem interface {
2015-06-05 07:13:59 +03:00
Init(*fuseops.InitOp) error
LookUpInode(*fuseops.LookUpInodeOp) error
GetInodeAttributes(*fuseops.GetInodeAttributesOp) error
SetInodeAttributes(*fuseops.SetInodeAttributesOp) error
ForgetInode(*fuseops.ForgetInodeOp) error
MkDir(*fuseops.MkDirOp) error
CreateFile(*fuseops.CreateFileOp) error
CreateSymlink(*fuseops.CreateSymlinkOp) error
RmDir(*fuseops.RmDirOp) error
Unlink(*fuseops.UnlinkOp) error
OpenDir(*fuseops.OpenDirOp) error
ReadDir(*fuseops.ReadDirOp) error
ReleaseDirHandle(*fuseops.ReleaseDirHandleOp) error
OpenFile(*fuseops.OpenFileOp) error
ReadFile(*fuseops.ReadFileOp) error
WriteFile(*fuseops.WriteFileOp) error
SyncFile(*fuseops.SyncFileOp) error
FlushFile(*fuseops.FlushFileOp) error
ReleaseFileHandle(*fuseops.ReleaseFileHandleOp) error
ReadSymlink(*fuseops.ReadSymlinkOp) error
}
// Create a fuse.Server that handles ops by calling the associated FileSystem
// method.Respond with the resulting error. Unsupported ops are responded to
// directly with ENOSYS.
//
2015-04-02 02:50:44 +03:00
// Each call to a FileSystem method is made on its own goroutine, and is free
// to block.
//
// (It is safe to naively process ops concurrently because the kernel
// guarantees to serialize operations that the user expects to happen in order,
// cf. http://goo.gl/jnkHPO, fuse-devel thread "Fuse guarantees on concurrent
// requests").
2015-03-25 01:24:44 +03:00
func NewFileSystemServer(fs FileSystem) fuse.Server {
return fileSystemServer{fs}
}
type fileSystemServer struct {
fs FileSystem
}
func (s fileSystemServer) ServeOps(c *fuse.Connection) {
for {
op, err := c.ReadOp()
if err == io.EOF {
break
}
if err != nil {
panic(err)
}
2015-04-02 02:52:58 +03:00
go s.handleOp(op)
}
}
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
func (s fileSystemServer) handleOp(op fuseops.Op) {
2015-04-02 03:19:56 +03:00
// Delay if requested.
2015-04-02 03:20:53 +03:00
if *fRandomDelays {
2015-04-02 03:19:56 +03:00
const delayLimit = 100 * time.Microsecond
delay := time.Duration(rand.Int63n(int64(delayLimit)))
time.Sleep(delay)
}
// Dispatch to the appropriate method.
2015-06-05 07:20:39 +03:00
var err error
2015-04-02 02:52:58 +03:00
switch typed := op.(type) {
default:
2015-06-05 07:20:39 +03:00
err = fuse.ENOSYS
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.InitOp:
2015-06-05 07:20:39 +03:00
err = s.fs.Init(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.LookUpInodeOp:
2015-06-05 07:20:39 +03:00
err = s.fs.LookUpInode(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.GetInodeAttributesOp:
2015-06-05 07:20:39 +03:00
err = s.fs.GetInodeAttributes(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.SetInodeAttributesOp:
2015-06-05 07:20:39 +03:00
err = s.fs.SetInodeAttributes(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.ForgetInodeOp:
2015-06-05 07:20:39 +03:00
err = s.fs.ForgetInode(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.MkDirOp:
2015-06-05 07:20:39 +03:00
err = s.fs.MkDir(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.CreateFileOp:
2015-06-05 07:20:39 +03:00
err = s.fs.CreateFile(typed)
2015-03-25 01:24:44 +03:00
2015-05-19 08:33:19 +03:00
case *fuseops.CreateSymlinkOp:
2015-06-05 07:20:39 +03:00
err = s.fs.CreateSymlink(typed)
2015-05-19 08:33:19 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.RmDirOp:
2015-06-05 07:20:39 +03:00
err = s.fs.RmDir(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.UnlinkOp:
2015-06-05 07:20:39 +03:00
err = s.fs.Unlink(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.OpenDirOp:
2015-06-05 07:20:39 +03:00
err = s.fs.OpenDir(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.ReadDirOp:
2015-06-05 07:20:39 +03:00
err = s.fs.ReadDir(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.ReleaseDirHandleOp:
2015-06-05 07:20:39 +03:00
err = s.fs.ReleaseDirHandle(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.OpenFileOp:
2015-06-05 07:20:39 +03:00
err = s.fs.OpenFile(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.ReadFileOp:
2015-06-05 07:20:39 +03:00
err = s.fs.ReadFile(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.WriteFileOp:
2015-06-05 07:20:39 +03:00
err = s.fs.WriteFile(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.SyncFileOp:
2015-06-05 07:20:39 +03:00
err = s.fs.SyncFile(typed)
2015-03-25 01:24:44 +03:00
2015-04-02 02:52:58 +03:00
case *fuseops.FlushFileOp:
2015-06-05 07:20:39 +03:00
err = s.fs.FlushFile(typed)
2015-04-02 02:52:58 +03:00
case *fuseops.ReleaseFileHandleOp:
2015-06-05 07:20:39 +03:00
err = s.fs.ReleaseFileHandle(typed)
2015-05-19 09:06:57 +03:00
case *fuseops.ReadSymlinkOp:
2015-06-05 07:20:39 +03:00
err = s.fs.ReadSymlink(typed)
2015-03-25 01:24:44 +03:00
}
2015-06-05 07:20:39 +03:00
op.Respond(err)
2015-03-25 01:24:44 +03:00
}