fusego/samples/flushfs/flush_fs.go

191 lines
4.2 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 flushfs
2015-03-20 03:21:38 +03:00
import (
2015-03-20 03:49:45 +03:00
"fmt"
2015-03-20 03:34:16 +03:00
"os"
2015-03-20 03:38:40 +03:00
"sync"
2015-03-20 03:34:16 +03:00
2015-03-20 03:21:38 +03:00
"github.com/jacobsa/fuse"
"github.com/jacobsa/fuse/fuseutil"
2015-03-20 03:22:16 +03:00
"golang.org/x/net/context"
2015-03-20 03:21:38 +03:00
)
// Create a file system containing a single file named "foo".
//
// The file may be opened for reading and/or writing. Its initial contents are
// empty. Whenever a flush or fsync is received, the supplied function will be
2015-03-20 03:52:54 +03:00
// called with the current contents of the file and its status returned.
func NewFileSystem(
2015-03-20 03:52:54 +03:00
reportFlush func(string) error,
reportFsync func(string) error) (fs fuse.FileSystem, err error) {
2015-03-20 05:27:27 +03:00
fs = &flushFS{
reportFlush: reportFlush,
reportFsync: reportFsync,
}
2015-03-20 03:21:38 +03:00
return
}
2015-03-20 03:38:40 +03:00
const fooID = fuse.RootInodeID + 1
2015-03-20 03:21:38 +03:00
type flushFS struct {
fuseutil.NotImplementedFileSystem
2015-03-20 05:27:27 +03:00
reportFlush func(string) error
reportFsync func(string) error
2015-03-20 03:34:16 +03:00
2015-03-20 03:49:45 +03:00
mu sync.Mutex
fooContents []byte // GUARDED_BY(mu)
2015-03-20 03:21:38 +03:00
}
2015-03-20 03:22:16 +03:00
2015-03-20 03:38:40 +03:00
////////////////////////////////////////////////////////////////////////
// Helpers
////////////////////////////////////////////////////////////////////////
2015-03-20 03:40:14 +03:00
// LOCKS_REQUIRED(fs.mu)
func (fs *flushFS) rootAttributes() fuse.InodeAttributes {
return fuse.InodeAttributes{
Nlink: 1,
Mode: 0777 | os.ModeDir,
}
}
2015-03-20 03:38:40 +03:00
// LOCKS_REQUIRED(fs.mu)
func (fs *flushFS) fooAttributes() fuse.InodeAttributes {
return fuse.InodeAttributes{
Nlink: 1,
Mode: 0777,
2015-03-20 05:28:51 +03:00
Size: uint64(len(fs.fooContents)),
2015-03-20 03:38:40 +03:00
}
}
////////////////////////////////////////////////////////////////////////
// File system methods
////////////////////////////////////////////////////////////////////////
2015-03-20 03:22:16 +03:00
func (fs *flushFS) Init(
ctx context.Context,
req *fuse.InitRequest) (
resp *fuse.InitResponse, err error) {
resp = &fuse.InitResponse{}
return
}
2015-03-20 03:38:40 +03:00
func (fs *flushFS) LookUpInode(
ctx context.Context,
req *fuse.LookUpInodeRequest) (
resp *fuse.LookUpInodeResponse, err error) {
resp = &fuse.LookUpInodeResponse{}
fs.mu.Lock()
defer fs.mu.Unlock()
// Sanity check.
if req.Parent != fuse.RootInodeID || req.Name != "foo" {
err = fuse.ENOENT
return
}
resp.Entry = fuse.ChildInodeEntry{
Child: fooID,
Attributes: fs.fooAttributes(),
}
return
}
2015-03-20 03:40:14 +03:00
func (fs *flushFS) GetInodeAttributes(
ctx context.Context,
req *fuse.GetInodeAttributesRequest) (
resp *fuse.GetInodeAttributesResponse, err error) {
resp = &fuse.GetInodeAttributesResponse{}
fs.mu.Lock()
defer fs.mu.Unlock()
switch req.Inode {
case fuse.RootInodeID:
resp.Attributes = fs.rootAttributes()
return
case fooID:
resp.Attributes = fs.fooAttributes()
return
default:
err = fuse.ENOENT
return
}
}
2015-03-20 03:40:53 +03:00
func (fs *flushFS) OpenFile(
ctx context.Context,
req *fuse.OpenFileRequest) (
resp *fuse.OpenFileResponse, err error) {
resp = &fuse.OpenFileResponse{}
fs.mu.Lock()
defer fs.mu.Unlock()
// Sanity check.
if req.Inode != fooID {
err = fuse.ENOSYS
return
}
return
}
2015-03-20 03:49:45 +03:00
func (fs *flushFS) WriteFile(
ctx context.Context,
req *fuse.WriteFileRequest) (
resp *fuse.WriteFileResponse, err error) {
resp = &fuse.WriteFileResponse{}
fs.mu.Lock()
defer fs.mu.Unlock()
// Ensure that the contents slice is long enough.
newLen := int(req.Offset) + len(req.Data)
if len(fs.fooContents) < newLen {
padding := make([]byte, newLen-len(fs.fooContents))
fs.fooContents = append(fs.fooContents, padding...)
}
// Copy in the data.
n := copy(fs.fooContents[req.Offset:], req.Data)
// Sanity check.
if n != len(req.Data) {
panic(fmt.Sprintf("Unexpected short copy: %v", n))
}
return
}
2015-03-20 05:27:27 +03:00
func (fs *flushFS) FlushFile(
ctx context.Context,
req *fuse.FlushFileRequest) (
resp *fuse.FlushFileResponse, err error) {
resp = &fuse.FlushFileResponse{}
fs.mu.Lock()
defer fs.mu.Unlock()
err = fs.reportFlush(string(fs.fooContents))
return
}