Commit 3fed732a authored by Han-Wen Nienhuys's avatar Han-Wen Nienhuys

Revise deletion handling. More tests.

parent 2121012d
...@@ -27,6 +27,11 @@ type MemUnionFs struct { ...@@ -27,6 +27,11 @@ type MemUnionFs struct {
readonly fuse.FileSystem readonly fuse.FileSystem
openWritable int openWritable int
// All paths that have been renamed or deleted will be marked
// here. After deletion, entries may be recreated, but they
// will be treated as new.
deleted map[string]bool
} }
type memNode struct { type memNode struct {
...@@ -40,7 +45,6 @@ type memNode struct { ...@@ -40,7 +45,6 @@ type memNode struct {
changed bool changed bool
link string link string
info os.FileInfo info os.FileInfo
deleted map[string]bool
} }
type Result struct { type Result struct {
...@@ -69,6 +73,35 @@ func (me *MemUnionFs) Reap() map[string]*Result { ...@@ -69,6 +73,35 @@ func (me *MemUnionFs) Reap() map[string]*Result {
} }
m := map[string]*Result{} m := map[string]*Result{}
for name, _ := range me.deleted {
fi, code := me.readonly.GetAttr(name, nil)
if !code.Ok() {
continue
}
m[name] = &Result{}
if !fi.IsDirectory() {
continue
}
todo := []string{name}
for len(todo) > 0 {
l := len(todo)-1
n := todo[l]
todo = todo[:l]
s, _ := me.readonly.OpenDir(n, nil)
for e := range s {
full := filepath.Join(n, e.Name)
m[full] = &Result{}
if e.Mode & fuse.S_IFDIR != 0 {
todo = append(todo, full)
}
}
}
}
me.root.Reap("", m) me.root.Reap("", m)
return m return m
} }
...@@ -76,6 +109,7 @@ func (me *MemUnionFs) Reap() map[string]*Result { ...@@ -76,6 +109,7 @@ func (me *MemUnionFs) Reap() map[string]*Result {
func (me *MemUnionFs) Clear() { func (me *MemUnionFs) Clear() {
me.mutex.Lock() me.mutex.Lock()
defer me.mutex.Unlock() defer me.mutex.Unlock()
me.deleted = make(map[string]bool)
me.root.Clear("") me.root.Clear("")
} }
...@@ -145,9 +179,6 @@ func (me *MemUnionFs) newNode(isdir bool) *memNode { ...@@ -145,9 +179,6 @@ func (me *MemUnionFs) newNode(isdir bool) *memNode {
fs: me, fs: me,
mutex: &me.mutex, mutex: &me.mutex,
} }
if isdir {
n.deleted = map[string]bool{}
}
now := time.Nanoseconds() now := time.Nanoseconds()
n.info.Mtime_ns = now n.info.Mtime_ns = now
n.info.Atime_ns = now n.info.Atime_ns = now
...@@ -157,16 +188,18 @@ func (me *MemUnionFs) newNode(isdir bool) *memNode { ...@@ -157,16 +188,18 @@ func (me *MemUnionFs) newNode(isdir bool) *memNode {
func NewMemUnionFs(backingStore string, roFs fuse.FileSystem) *MemUnionFs { func NewMemUnionFs(backingStore string, roFs fuse.FileSystem) *MemUnionFs {
me := &MemUnionFs{} me := &MemUnionFs{}
me.deleted = make(map[string]bool)
me.backingStore = backingStore me.backingStore = backingStore
me.readonly = roFs me.readonly = roFs
me.root = me.newNode(true) me.root = me.newNode(true)
me.root.info.Mode = fuse.S_IFDIR | 0755 me.root.info.Mode = fuse.S_IFDIR | 0755
me.cond = sync.NewCond(&me.mutex) me.cond = sync.NewCond(&me.mutex)
return me return me
} }
func (me *memNode) Deletable() bool { func (me *memNode) Deletable() bool {
return !me.changed return !me.changed && me.original == ""
} }
func (me *memNode) touch() { func (me *memNode) touch() {
...@@ -194,16 +227,20 @@ func (me *memNode) Readlink(c *fuse.Context) ([]byte, fuse.Status) { ...@@ -194,16 +227,20 @@ func (me *memNode) Readlink(c *fuse.Context) ([]byte, fuse.Status) {
func (me *memNode) Lookup(name string, context *fuse.Context) (fi *os.FileInfo, node fuse.FsNode, code fuse.Status) { func (me *memNode) Lookup(name string, context *fuse.Context) (fi *os.FileInfo, node fuse.FsNode, code fuse.Status) {
me.mutex.RLock() me.mutex.RLock()
defer me.mutex.RUnlock() defer me.mutex.RUnlock()
return me.lookup(name, context)
}
if _, del := me.deleted[name]; del { // Must run with mutex held.
return nil, nil, fuse.ENOENT func (me *memNode) lookup(name string, context *fuse.Context) (fi *os.FileInfo, node fuse.FsNode, code fuse.Status) {
}
if me.original == "" && me != me.fs.root { if me.original == "" && me != me.fs.root {
return nil, nil, fuse.ENOENT return nil, nil, fuse.ENOENT
} }
fn := filepath.Join(me.original, name) fn := filepath.Join(me.original, name)
if _, del := me.fs.deleted[fn]; del {
return nil, nil, fuse.ENOENT
}
fi, code = me.fs.readonly.GetAttr(fn, context) fi, code = me.fs.readonly.GetAttr(fn, context)
if !code.Ok() { if !code.Ok() {
return nil, nil, code return nil, nil, code
...@@ -224,7 +261,6 @@ func (me *memNode) Mkdir(name string, mode uint32, context *fuse.Context) (fi *o ...@@ -224,7 +261,6 @@ func (me *memNode) Mkdir(name string, mode uint32, context *fuse.Context) (fi *o
me.mutex.Lock() me.mutex.Lock()
defer me.mutex.Unlock() defer me.mutex.Unlock()
me.deleted[name] = false, false
n := me.newNode(true) n := me.newNode(true)
n.changed = true n.changed = true
n.info.Mode = mode | fuse.S_IFDIR n.info.Mode = mode | fuse.S_IFDIR
...@@ -237,7 +273,9 @@ func (me *memNode) Unlink(name string, context *fuse.Context) (code fuse.Status) ...@@ -237,7 +273,9 @@ func (me *memNode) Unlink(name string, context *fuse.Context) (code fuse.Status)
me.mutex.Lock() me.mutex.Lock()
defer me.mutex.Unlock() defer me.mutex.Unlock()
me.deleted[name] = true if me.original != "" || me == me.fs.root {
me.fs.deleted[filepath.Join(me.original, name)] = true
}
ch := me.Inode().RmChild(name) ch := me.Inode().RmChild(name)
if ch == nil { if ch == nil {
return fuse.ENOENT return fuse.ENOENT
...@@ -260,24 +298,60 @@ func (me *memNode) Symlink(name string, content string, context *fuse.Context) ( ...@@ -260,24 +298,60 @@ func (me *memNode) Symlink(name string, content string, context *fuse.Context) (
n.changed = true n.changed = true
me.Inode().AddChild(name, n.Inode()) me.Inode().AddChild(name, n.Inode())
me.touch() me.touch()
me.deleted[name] = false, false
return &n.info, n, fuse.OK return &n.info, n, fuse.OK
} }
// Expand the original fs as a tree.
func (me *memNode) materializeSelf() {
me.changed = true
if !me.Inode().IsDir() {
return
}
s, _ := me.fs.readonly.OpenDir(me.original, nil)
for e := range s {
me.lookup(e.Name, nil)
}
me.original = ""
}
func (me *memNode) materialize() {
me.materializeSelf()
for _, n := range me.Inode().FsChildren() {
if n.IsDir() {
n.FsNode().(*memNode).materialize()
}
}
}
func (me *memNode) Rename(oldName string, newParent fuse.FsNode, newName string, context *fuse.Context) (code fuse.Status) { func (me *memNode) Rename(oldName string, newParent fuse.FsNode, newName string, context *fuse.Context) (code fuse.Status) {
me.mutex.Lock() me.mutex.Lock()
defer me.mutex.Unlock() defer me.mutex.Unlock()
ch := me.Inode().RmChild(oldName) ch := me.Inode().RmChild(oldName)
me.deleted[oldName] = true if ch == nil {
return fuse.ENOENT
}
if me.original != "" || me == me.fs.root {
me.fs.deleted[filepath.Join(me.original, oldName)] = true
}
mn := ch.FsNode().(*memNode)
if mn.original != "" || mn == me.fs.root {
if newParent.(*memNode).original != "" {
me.fs.deleted[filepath.Join(newParent.(*memNode).original, newName)] = true
}
log.Println("materialize")
mn.materialize()
mn.markChanged()
}
newParent.Inode().RmChild(newName) newParent.Inode().RmChild(newName)
newParent.Inode().AddChild(newName, ch) newParent.Inode().AddChild(newName, ch)
me.deleted[newName] = false, false
me.markChanged()
me.touch() me.touch()
return fuse.OK return fuse.OK
} }
// TODO - test this.
func (me *memNode) markChanged() { func (me *memNode) markChanged() {
me.changed = true me.changed = true
for _, n := range me.Inode().FsChildren() { for _, n := range me.Inode().FsChildren() {
...@@ -285,7 +359,6 @@ func (me *memNode) markChanged() { ...@@ -285,7 +359,6 @@ func (me *memNode) markChanged() {
} }
} }
func (me *memNode) Link(name string, existing fuse.FsNode, context *fuse.Context) (fi *os.FileInfo, newNode fuse.FsNode, code fuse.Status) { func (me *memNode) Link(name string, existing fuse.FsNode, context *fuse.Context) (fi *os.FileInfo, newNode fuse.FsNode, code fuse.Status) {
me.Inode().AddChild(name, existing.Inode()) me.Inode().AddChild(name, existing.Inode())
fi, code = existing.GetAttr(nil, context) fi, code = existing.GetAttr(nil, context)
...@@ -293,7 +366,6 @@ func (me *memNode) Link(name string, existing fuse.FsNode, context *fuse.Context ...@@ -293,7 +366,6 @@ func (me *memNode) Link(name string, existing fuse.FsNode, context *fuse.Context
me.mutex.Lock() me.mutex.Lock()
defer me.mutex.Unlock() defer me.mutex.Unlock()
me.touch() me.touch()
me.deleted[name] = false, false
return fi, existing, code return fi, existing, code
} }
...@@ -311,7 +383,6 @@ func (me *memNode) Create(name string, flags uint32, mode uint32, context *fuse. ...@@ -311,7 +383,6 @@ func (me *memNode) Create(name string, flags uint32, mode uint32, context *fuse.
} }
me.Inode().AddChild(name, n.Inode()) me.Inode().AddChild(name, n.Inode())
me.touch() me.touch()
me.deleted[name] = false, false
me.fs.openWritable++ me.fs.openWritable++
return n.newFile(&fuse.LoopbackFile{File: f}, true), &n.info, n, fuse.OK return n.newFile(&fuse.LoopbackFile{File: f}, true), &n.info, n, fuse.OK
} }
...@@ -468,10 +539,13 @@ func (me *memNode) OpenDir(context *fuse.Context) (stream chan fuse.DirEntry, co ...@@ -468,10 +539,13 @@ func (me *memNode) OpenDir(context *fuse.Context) (stream chan fuse.DirEntry, co
if me.original != "" || me == me.fs.root { if me.original != "" || me == me.fs.root {
stream, code = me.fs.readonly.OpenDir(me.original, context) stream, code = me.fs.readonly.OpenDir(me.original, context)
for e := range stream { for e := range stream {
ch[e.Name] = e.Mode fn := filepath.Join(me.original, e.Name)
if !me.fs.deleted[fn] {
ch[e.Name] = e.Mode
}
} }
} }
for k, n := range me.Inode().FsChildren() { for k, n := range me.Inode().FsChildren() {
fi, code := n.FsNode().GetAttr(nil, nil) fi, code := n.FsNode().GetAttr(nil, nil)
if !code.Ok() { if !code.Ok() {
...@@ -480,10 +554,6 @@ func (me *memNode) OpenDir(context *fuse.Context) (stream chan fuse.DirEntry, co ...@@ -480,10 +554,6 @@ func (me *memNode) OpenDir(context *fuse.Context) (stream chan fuse.DirEntry, co
ch[k] = fi.Mode ch[k] = fi.Mode
} }
for k, _ := range me.deleted {
ch[k] = 0, false
}
stream = make(chan fuse.DirEntry, len(ch)) stream = make(chan fuse.DirEntry, len(ch))
for k, v := range ch { for k, v := range ch {
stream <- fuse.DirEntry{Name: k, Mode: v} stream <- fuse.DirEntry{Name: k, Mode: v}
...@@ -493,11 +563,6 @@ func (me *memNode) OpenDir(context *fuse.Context) (stream chan fuse.DirEntry, co ...@@ -493,11 +563,6 @@ func (me *memNode) OpenDir(context *fuse.Context) (stream chan fuse.DirEntry, co
} }
func (me *memNode) Reap(path string, results map[string]*Result) { func (me *memNode) Reap(path string, results map[string]*Result) {
for name, _ := range me.deleted {
p := filepath.Join(path, name)
results[p] = &Result{}
}
if me.changed { if me.changed {
info := me.info info := me.info
results[path] = &Result{ results[path] = &Result{
...@@ -518,7 +583,6 @@ func (me *memNode) Clear(path string) { ...@@ -518,7 +583,6 @@ func (me *memNode) Clear(path string) {
me.original = path me.original = path
me.changed = false me.changed = false
me.backing = "" me.backing = ""
me.deleted = make(map[string]bool)
for n, ch := range me.Inode().FsChildren() { for n, ch := range me.Inode().FsChildren() {
p := filepath.Join(path, n) p := filepath.Join(path, n)
mn := ch.FsNode().(*memNode) mn := ch.FsNode().(*memNode)
......
...@@ -205,7 +205,6 @@ func TestMemUnionFsBasic(t *testing.T) { ...@@ -205,7 +205,6 @@ func TestMemUnionFsBasic(t *testing.T) {
checkMapEq(t, names, map[string]bool{ checkMapEq(t, names, map[string]bool{
"rw": true, "ro2": true, "rw": true, "ro2": true,
}) })
} }
func TestMemUnionFsPromote(t *testing.T) { func TestMemUnionFsPromote(t *testing.T) {
...@@ -661,3 +660,104 @@ func TestMemUnionFsTruncGetAttr(t *testing.T) { ...@@ -661,3 +660,104 @@ func TestMemUnionFsTruncGetAttr(t *testing.T) {
t.Fatalf("Length mismatch got %d want %d", fi.Size, len(c)) t.Fatalf("Length mismatch got %d want %d", fi.Size, len(c))
} }
} }
func TestMemUnionFsRenameDirBasic(t *testing.T) {
wd, ufs, clean := setupMemUfs(t)
defer clean()
err := os.MkdirAll(wd+"/ro/dir/subdir", 0755)
CheckSuccess(err)
err = os.Rename(wd+"/mount/dir", wd+"/mount/renamed")
CheckSuccess(err)
if fi, _ := os.Lstat(wd + "/mount/dir"); fi != nil {
t.Fatalf("%s/mount/dir should have disappeared: %v", wd, fi)
}
if fi, _ := os.Lstat(wd + "/mount/renamed"); fi == nil || !fi.IsDirectory() {
t.Fatalf("%s/mount/renamed should be directory: %v", wd, fi)
}
entries, err := ioutil.ReadDir(wd + "/mount/renamed")
if err != nil || len(entries) != 1 || entries[0].Name != "subdir" {
t.Errorf("readdir(%s/mount/renamed) should have one entry: %v, err %v", wd, entries, err)
}
r := ufs.Reap()
if r["dir"] == nil || r["dir"].FileInfo != nil || r["renamed/subdir"] == nil || !r["renamed/subdir"].FileInfo.IsDirectory() {
t.Errorf("Reap should del dir, and add renamed/subdir: %v", r)
}
if err = os.Mkdir(wd+"/mount/dir", 0755); err != nil {
t.Errorf("mkdir should succeed %v", err)
}
}
func TestMemUnionFsRenameDirAllSourcesGone(t *testing.T) {
wd, ufs, clean := setupMemUfs(t)
defer clean()
err := os.MkdirAll(wd+"/ro/dir", 0755)
CheckSuccess(err)
err = ioutil.WriteFile(wd+"/ro/dir/file.txt", []byte{42}, 0644)
CheckSuccess(err)
err = os.Rename(wd+"/mount/dir", wd+"/mount/renamed")
CheckSuccess(err)
r := ufs.Reap()
if r["dir"] == nil || r["dir"].FileInfo != nil || r["dir/file.txt"] == nil || r["dir/file.txt"].FileInfo != nil {
t.Errorf("Expected 2 deletion entries in %v", r)
}
}
func TestMemUnionFsRenameDirWithDeletions(t *testing.T) {
wd, _, clean := setupMemUfs(t)
defer clean()
err := os.MkdirAll(wd+"/ro/dir/subdir", 0755)
CheckSuccess(err)
err = ioutil.WriteFile(wd+"/ro/dir/file.txt", []byte{42}, 0644)
CheckSuccess(err)
err = ioutil.WriteFile(wd+"/ro/dir/subdir/file.txt", []byte{42}, 0644)
CheckSuccess(err)
if fi, _ := os.Lstat(wd + "/mount/dir/subdir/file.txt"); fi == nil || !fi.IsRegular() {
t.Fatalf("%s/mount/dir/subdir/file.txt should be file: %v", wd, fi)
}
err = os.Remove(wd + "/mount/dir/file.txt")
CheckSuccess(err)
err = os.Rename(wd+"/mount/dir", wd+"/mount/renamed")
CheckSuccess(err)
if fi, _ := os.Lstat(wd + "/mount/dir/subdir/file.txt"); fi != nil {
t.Fatalf("%s/mount/dir/subdir/file.txt should have disappeared: %v", wd, fi)
}
if fi, _ := os.Lstat(wd + "/mount/dir"); fi != nil {
t.Fatalf("%s/mount/dir should have disappeared: %v", wd, fi)
}
if fi, _ := os.Lstat(wd + "/mount/renamed"); fi == nil || !fi.IsDirectory() {
t.Fatalf("%s/mount/renamed should be directory: %v", wd, fi)
}
if fi, _ := os.Lstat(wd + "/mount/renamed/file.txt"); fi != nil {
t.Fatalf("%s/mount/renamed/file.txt should have disappeared %#v", wd, fi)
}
if err = os.Mkdir(wd+"/mount/dir", 0755); err != nil {
t.Errorf("mkdir should succeed %v", err)
}
if fi, _ := os.Lstat(wd + "/mount/dir/subdir"); fi != nil {
t.Fatalf("%s/mount/dir/subdir should have disappeared %#v", wd, fi)
}
}
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