Commit f1a7913d authored by Han-Wen Nienhuys's avatar Han-Wen Nienhuys

Add a multizip file system, that mounts new zip files on

fuse/SUBDIR-NAME by doing

  echo ZIP-FILE-NAME > fuse/config/SUBDIR-NAME

this exercises the PathFileSystemConnector mount functionality.
parent dad9f3f8
...@@ -2,8 +2,9 @@ ...@@ -2,8 +2,9 @@
include $(GOROOT)/src/Make.inc include $(GOROOT)/src/Make.inc
TARG=zipfs TARG=zipfs
GOFILES=zipfs.go GOFILES=multizip.go
DEPS=../examplelib DEPS=../examplelib
include $(GOROOT)/src/Make.cmd include $(GOROOT)/src/Make.cmd
package main
import (
"github.com/hanwen/go-fuse/fuse"
"github.com/hanwen/go-fuse/examplelib"
"fmt"
"flag"
"log"
"os"
"path"
"sync"
"strings"
)
var _ = log.Printf
const (
CONFIG_PREFIX = "config/"
)
type ZipCreateFile struct {
// Basename of the entry in the FS.
Basename string
zfs *MultiZipFs
fuse.DefaultRawFuseFile
}
func (me *ZipCreateFile) Write(input *fuse.WriteIn, nameBytes []byte) (uint32, fuse.Status) {
if me.zfs == nil {
// TODO
return 0, fuse.EPERM
}
zipFile := string(nameBytes)
zipFile = strings.Trim(zipFile, "\n ")
fs := examplelib.NewZipFileFuse(zipFile)
if fs == nil {
// TODO
log.Println("NewZipFileFuse returned nil")
me.zfs.pendingZips[me.Basename] = false, false
return 0, fuse.ENOSYS
}
code := me.zfs.Connector.Mount("/"+path.Base(me.Basename), fs)
if code != fuse.OK {
return 0, code
}
// TODO. locks?
me.zfs.zips[me.Basename] = fs
me.zfs.pendingZips[me.Basename] = false, false
me.zfs = nil
return uint32(len(nameBytes)), code
}
////////////////////////////////////////////////////////////////
type MultiZipFs struct {
Connector *fuse.PathFileSystemConnector
lock sync.RWMutex
zips map[string]*examplelib.ZipFileFuse
pendingZips map[string]bool
zipFileNames map[string]string
fuse.DefaultPathFilesystem
}
func NewMultiZipFs() *MultiZipFs {
m := new(MultiZipFs)
m.zips = make(map[string]*examplelib.ZipFileFuse)
m.pendingZips = make(map[string]bool)
m.zipFileNames = make(map[string]string)
m.Connector = fuse.NewPathFileSystemConnector(m)
return m
}
func (me *MultiZipFs) OpenDir(name string) (stream chan fuse.DirEntry, code fuse.Status) {
me.lock.RLock()
defer me.lock.RUnlock()
// We don't use a goroutine, since we don't want to hold the
// lock.
stream = make(chan fuse.DirEntry,
len(me.pendingZips)+len(me.zips)+2)
submode := uint32(fuse.S_IFDIR | 0700)
if name == "config" {
submode = fuse.S_IFREG | 0600
}
for k, _ := range me.zips {
var d fuse.DirEntry
d.Name = k
d.Mode = submode
stream <- fuse.DirEntry(d)
}
for k, _ := range me.pendingZips {
var d fuse.DirEntry
d.Name = k
d.Mode = submode
stream <- fuse.DirEntry(d)
}
if name == "" {
var d fuse.DirEntry
d.Name = "config"
d.Mode = fuse.S_IFDIR | 0700
stream <- fuse.DirEntry(d)
}
stream <- fuse.DirEntry{Name: ""}
return stream, fuse.OK
}
func (me *MultiZipFs) GetAttr(name string) (*fuse.Attr, fuse.Status) {
a := new(fuse.Attr)
if name == "" {
// Should not write in top dir.
a.Mode = fuse.S_IFDIR | 0500
return a, fuse.OK
}
if name == "config" {
// TODO
a.Mode = fuse.S_IFDIR | 0700
return a, fuse.OK
}
dir, base := path.Split(name)
if dir != "" && dir != CONFIG_PREFIX {
return nil, fuse.ENOENT
}
submode := uint32(fuse.S_IFDIR | 0700)
if dir == CONFIG_PREFIX {
submode = fuse.S_IFREG | 0600
}
me.lock.RLock()
defer me.lock.RUnlock()
a.Mode = submode
entry, hasDir := me.zips[base]
if hasDir {
a.Size = uint64(len(entry.ZipFileName))
return a, fuse.OK
}
_, hasDir = me.pendingZips[base]
if hasDir {
return a, fuse.OK
}
return nil, fuse.ENOENT
}
func (me *MultiZipFs) Unlink(name string) (code fuse.Status) {
dir, basename := path.Split(name)
if dir == CONFIG_PREFIX {
me.lock.Lock()
defer me.lock.Unlock()
_, ok := me.zips[basename]
if ok {
me.zips[basename] = nil, false
return fuse.OK
} else {
return fuse.ENOENT
}
}
return fuse.EPERM
}
func (me *MultiZipFs) Open(name string, flags uint32) (file fuse.RawFuseFile, code fuse.Status) {
if 0 != flags&uint32(os.O_WRONLY|os.O_RDWR|os.O_APPEND) {
return nil, fuse.EPERM
}
dir, basename := path.Split(name)
if dir == CONFIG_PREFIX {
me.lock.RLock()
defer me.lock.RUnlock()
entry, ok := me.zips[basename]
if !ok {
return nil, fuse.ENOENT
}
return fuse.NewReadOnlyFile([]byte(entry.ZipFileName)), fuse.OK
}
return nil, fuse.ENOENT
}
func (me *MultiZipFs) Create(name string, flags uint32, mode uint32) (file fuse.RawFuseFile, code fuse.Status) {
dir, base := path.Split(name)
if dir != CONFIG_PREFIX {
return nil, fuse.EPERM
}
z := new(ZipCreateFile)
z.Basename = base
z.zfs = me
me.lock.Lock()
defer me.lock.Unlock()
me.pendingZips[z.Basename] = true
return z, fuse.OK
}
func main() {
// Scans the arg list and sets up flags
flag.Parse()
if flag.NArg() < 1 {
// TODO - where to get program name?
fmt.Println("usage: main MOUNTPOINT")
os.Exit(2)
}
fs := NewMultiZipFs()
state := fuse.NewMountState(fs.Connector)
mountPoint := flag.Arg(0)
state.Debug = true
state.Mount(mountPoint)
fmt.Printf("Mounted %s\n", mountPoint)
state.Loop(true)
fmt.Println(state.Stats())
}
package main
import (
"github.com/hanwen/go-fuse/fuse"
"github.com/hanwen/go-fuse/examplelib"
"fmt"
"os"
"flag"
"log"
)
var _ = log.Printf
func main() {
// Scans the arg list and sets up flags
flag.Parse()
if flag.NArg() < 2 {
// TODO - where to get program name?
fmt.Println("usage: main ZIPFILE MOUNTPOINT")
os.Exit(2)
}
orig := flag.Arg(0)
fs := examplelib.NewZipFileFuse(orig)
conn := fuse.NewPathFileSystemConnector(fs)
state := fuse.NewMountState(conn)
mountPoint := flag.Arg(1)
state.Debug = true
state.Mount(mountPoint)
fmt.Printf("Mounted %s on %s\n", orig, mountPoint)
state.Loop(true)
fmt.Println(state.Stats())
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment