Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
go-fuse
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Levin Zimmermann
go-fuse
Commits
3fed732a
Commit
3fed732a
authored
Sep 26, 2011
by
Han-Wen Nienhuys
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Revise deletion handling. More tests.
parent
2121012d
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
196 additions
and
32 deletions
+196
-32
unionfs/memunionfs.go
unionfs/memunionfs.go
+95
-31
unionfs/memunionfs_test.go
unionfs/memunionfs_test.go
+101
-1
No files found.
unionfs/memunionfs.go
View file @
3fed732a
...
...
@@ -27,6 +27,11 @@ type MemUnionFs struct {
readonly
fuse
.
FileSystem
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
{
...
...
@@ -40,7 +45,6 @@ type memNode struct {
changed
bool
link
string
info
os
.
FileInfo
deleted
map
[
string
]
bool
}
type
Result
struct
{
...
...
@@ -69,6 +73,35 @@ func (me *MemUnionFs) Reap() 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
)
return
m
}
...
...
@@ -76,6 +109,7 @@ func (me *MemUnionFs) Reap() map[string]*Result {
func
(
me
*
MemUnionFs
)
Clear
()
{
me
.
mutex
.
Lock
()
defer
me
.
mutex
.
Unlock
()
me
.
deleted
=
make
(
map
[
string
]
bool
)
me
.
root
.
Clear
(
""
)
}
...
...
@@ -145,9 +179,6 @@ func (me *MemUnionFs) newNode(isdir bool) *memNode {
fs
:
me
,
mutex
:
&
me
.
mutex
,
}
if
isdir
{
n
.
deleted
=
map
[
string
]
bool
{}
}
now
:=
time
.
Nanoseconds
()
n
.
info
.
Mtime_ns
=
now
n
.
info
.
Atime_ns
=
now
...
...
@@ -157,16 +188,18 @@ func (me *MemUnionFs) newNode(isdir bool) *memNode {
func
NewMemUnionFs
(
backingStore
string
,
roFs
fuse
.
FileSystem
)
*
MemUnionFs
{
me
:=
&
MemUnionFs
{}
me
.
deleted
=
make
(
map
[
string
]
bool
)
me
.
backingStore
=
backingStore
me
.
readonly
=
roFs
me
.
root
=
me
.
newNode
(
true
)
me
.
root
.
info
.
Mode
=
fuse
.
S_IFDIR
|
0755
me
.
cond
=
sync
.
NewCond
(
&
me
.
mutex
)
return
me
}
func
(
me
*
memNode
)
Deletable
()
bool
{
return
!
me
.
changed
return
!
me
.
changed
&&
me
.
original
==
""
}
func
(
me
*
memNode
)
touch
()
{
...
...
@@ -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
)
{
me
.
mutex
.
RLock
()
defer
me
.
mutex
.
RUnlock
()
return
me
.
lookup
(
name
,
context
)
}
if
_
,
del
:=
me
.
deleted
[
name
];
del
{
return
nil
,
nil
,
fuse
.
ENOENT
}
// Must run with mutex held.
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
{
return
nil
,
nil
,
fuse
.
ENOENT
}
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
)
if
!
code
.
Ok
()
{
return
nil
,
nil
,
code
...
...
@@ -224,7 +261,6 @@ func (me *memNode) Mkdir(name string, mode uint32, context *fuse.Context) (fi *o
me
.
mutex
.
Lock
()
defer
me
.
mutex
.
Unlock
()
me
.
deleted
[
name
]
=
false
,
false
n
:=
me
.
newNode
(
true
)
n
.
changed
=
true
n
.
info
.
Mode
=
mode
|
fuse
.
S_IFDIR
...
...
@@ -237,7 +273,9 @@ func (me *memNode) Unlink(name string, context *fuse.Context) (code fuse.Status)
me
.
mutex
.
Lock
()
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
)
if
ch
==
nil
{
return
fuse
.
ENOENT
...
...
@@ -260,24 +298,60 @@ func (me *memNode) Symlink(name string, content string, context *fuse.Context) (
n
.
changed
=
true
me
.
Inode
()
.
AddChild
(
name
,
n
.
Inode
())
me
.
touch
()
me
.
deleted
[
name
]
=
false
,
false
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
)
{
me
.
mutex
.
Lock
()
defer
me
.
mutex
.
Unlock
()
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
()
.
AddChild
(
newName
,
ch
)
me
.
deleted
[
newName
]
=
false
,
false
me
.
markChanged
()
me
.
touch
()
return
fuse
.
OK
}
// TODO - test this.
func
(
me
*
memNode
)
markChanged
()
{
me
.
changed
=
true
for
_
,
n
:=
range
me
.
Inode
()
.
FsChildren
()
{
...
...
@@ -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
)
{
me
.
Inode
()
.
AddChild
(
name
,
existing
.
Inode
())
fi
,
code
=
existing
.
GetAttr
(
nil
,
context
)
...
...
@@ -293,7 +366,6 @@ func (me *memNode) Link(name string, existing fuse.FsNode, context *fuse.Context
me
.
mutex
.
Lock
()
defer
me
.
mutex
.
Unlock
()
me
.
touch
()
me
.
deleted
[
name
]
=
false
,
false
return
fi
,
existing
,
code
}
...
...
@@ -311,7 +383,6 @@ func (me *memNode) Create(name string, flags uint32, mode uint32, context *fuse.
}
me
.
Inode
()
.
AddChild
(
name
,
n
.
Inode
())
me
.
touch
()
me
.
deleted
[
name
]
=
false
,
false
me
.
fs
.
openWritable
++
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
if
me
.
original
!=
""
||
me
==
me
.
fs
.
root
{
stream
,
code
=
me
.
fs
.
readonly
.
OpenDir
(
me
.
original
,
context
)
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
()
{
fi
,
code
:=
n
.
FsNode
()
.
GetAttr
(
nil
,
nil
)
if
!
code
.
Ok
()
{
...
...
@@ -480,10 +554,6 @@ func (me *memNode) OpenDir(context *fuse.Context) (stream chan fuse.DirEntry, co
ch
[
k
]
=
fi
.
Mode
}
for
k
,
_
:=
range
me
.
deleted
{
ch
[
k
]
=
0
,
false
}
stream
=
make
(
chan
fuse
.
DirEntry
,
len
(
ch
))
for
k
,
v
:=
range
ch
{
stream
<-
fuse
.
DirEntry
{
Name
:
k
,
Mode
:
v
}
...
...
@@ -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
)
{
for
name
,
_
:=
range
me
.
deleted
{
p
:=
filepath
.
Join
(
path
,
name
)
results
[
p
]
=
&
Result
{}
}
if
me
.
changed
{
info
:=
me
.
info
results
[
path
]
=
&
Result
{
...
...
@@ -518,7 +583,6 @@ func (me *memNode) Clear(path string) {
me
.
original
=
path
me
.
changed
=
false
me
.
backing
=
""
me
.
deleted
=
make
(
map
[
string
]
bool
)
for
n
,
ch
:=
range
me
.
Inode
()
.
FsChildren
()
{
p
:=
filepath
.
Join
(
path
,
n
)
mn
:=
ch
.
FsNode
()
.
(
*
memNode
)
...
...
unionfs/memunionfs_test.go
View file @
3fed732a
...
...
@@ -205,7 +205,6 @@ func TestMemUnionFsBasic(t *testing.T) {
checkMapEq
(
t
,
names
,
map
[
string
]
bool
{
"rw"
:
true
,
"ro2"
:
true
,
})
}
func
TestMemUnionFsPromote
(
t
*
testing
.
T
)
{
...
...
@@ -661,3 +660,104 @@ func TestMemUnionFsTruncGetAttr(t *testing.T) {
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
)
}
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment