mirror of
https://github.com/tahoe-lafs/tahoe-lafs.git
synced 2025-01-15 17:30:01 +00:00
173 lines
4.8 KiB
Python
173 lines
4.8 KiB
Python
|
from handler import Handler
|
||
|
import stat, errno, os, time
|
||
|
from cStringIO import StringIO
|
||
|
from kernel import *
|
||
|
|
||
|
|
||
|
UID = os.getuid()
|
||
|
GID = os.getgid()
|
||
|
UMASK = os.umask(0); os.umask(UMASK)
|
||
|
INFINITE = 86400.0
|
||
|
|
||
|
|
||
|
class Node(object):
|
||
|
__slots__ = ['attr', 'data']
|
||
|
|
||
|
def __init__(self, attr, data=None):
|
||
|
self.attr = attr
|
||
|
self.data = data
|
||
|
|
||
|
def type(self):
|
||
|
return mode2type(self.attr.mode)
|
||
|
|
||
|
def modified(self):
|
||
|
self.attr.mtime = self.attr.atime = time.time()
|
||
|
t = self.type()
|
||
|
if t == TYPE_REG:
|
||
|
f = self.data
|
||
|
pos = f.tell()
|
||
|
f.seek(0, 2)
|
||
|
self.attr.size = f.tell()
|
||
|
f.seek(pos)
|
||
|
elif t == TYPE_DIR:
|
||
|
nsubdirs = 0
|
||
|
for nodeid in self.data.values():
|
||
|
nsubdirs += nodeid & 1
|
||
|
self.attr.nlink = 2 + nsubdirs
|
||
|
|
||
|
|
||
|
def newattr(s, mode=0666):
|
||
|
now = time.time()
|
||
|
return fuse_attr(ino = INVALID_INO,
|
||
|
size = 0,
|
||
|
mode = s | (mode & ~UMASK),
|
||
|
nlink = 1 + (s == stat.S_IFDIR),
|
||
|
atime = now,
|
||
|
mtime = now,
|
||
|
ctime = now,
|
||
|
uid = UID,
|
||
|
gid = GID)
|
||
|
|
||
|
# ____________________________________________________________
|
||
|
|
||
|
class Filesystem:
|
||
|
|
||
|
def __init__(self, rootnode):
|
||
|
self.nodes = {FUSE_ROOT_ID: rootnode}
|
||
|
self.nextid = 2
|
||
|
assert self.nextid > FUSE_ROOT_ID
|
||
|
|
||
|
def getnode(self, nodeid):
|
||
|
try:
|
||
|
return self.nodes[nodeid]
|
||
|
except KeyError:
|
||
|
raise IOError(errno.ESTALE, nodeid)
|
||
|
|
||
|
def forget(self, nodeid):
|
||
|
pass
|
||
|
|
||
|
def cachenode(self, node):
|
||
|
id = self.nextid
|
||
|
self.nextid += 2
|
||
|
if node.type() == TYPE_DIR:
|
||
|
id += 1
|
||
|
self.nodes[id] = node
|
||
|
return id
|
||
|
|
||
|
def getattr(self, node):
|
||
|
return node.attr, INFINITE
|
||
|
|
||
|
def setattr(self, node, mode=None, uid=None, gid=None,
|
||
|
size=None, atime=None, mtime=None):
|
||
|
if mode is not None: node.attr.mode = (node.attr.mode&~0777) | mode
|
||
|
if uid is not None: node.attr.uid = uid
|
||
|
if gid is not None: node.attr.gid = gid
|
||
|
if atime is not None: node.attr.atime = atime
|
||
|
if mtime is not None: node.attr.mtime = mtime
|
||
|
if size is not None and node.type() == TYPE_REG:
|
||
|
node.data.seek(size)
|
||
|
node.data.truncate()
|
||
|
|
||
|
def listdir(self, node):
|
||
|
for name, subnodeid in node.data.items():
|
||
|
subnode = self.nodes[subnodeid]
|
||
|
yield name, subnode.type()
|
||
|
|
||
|
def lookup(self, node, name):
|
||
|
try:
|
||
|
return node.data[name], INFINITE
|
||
|
except KeyError:
|
||
|
pass
|
||
|
if hasattr(node, 'findnode'):
|
||
|
try:
|
||
|
subnode = node.findnode(name)
|
||
|
except KeyError:
|
||
|
pass
|
||
|
else:
|
||
|
id = self.cachenode(subnode)
|
||
|
node.data[name] = id
|
||
|
return id, INFINITE
|
||
|
raise IOError(errno.ENOENT, name)
|
||
|
|
||
|
def open(self, node, mode):
|
||
|
return node.data
|
||
|
|
||
|
def mknod(self, node, name, mode):
|
||
|
subnode = Node(newattr(mode & 0170000, mode & 0777))
|
||
|
if subnode.type() == TYPE_REG:
|
||
|
subnode.data = StringIO()
|
||
|
else:
|
||
|
raise NotImplementedError
|
||
|
id = self.cachenode(subnode)
|
||
|
node.data[name] = id
|
||
|
node.modified()
|
||
|
return id, INFINITE
|
||
|
|
||
|
def mkdir(self, node, name, mode):
|
||
|
subnode = Node(newattr(stat.S_IFDIR, mode & 0777), {})
|
||
|
id = self.cachenode(subnode)
|
||
|
node.data[name] = id
|
||
|
node.modified()
|
||
|
return id, INFINITE
|
||
|
|
||
|
def symlink(self, node, linkname, target):
|
||
|
subnode = Node(newattr(stat.S_IFLNK, 0777), target)
|
||
|
id = self.cachenode(subnode)
|
||
|
node.data[linkname] = id
|
||
|
node.modified()
|
||
|
return id, INFINITE
|
||
|
|
||
|
def readlink(self, node):
|
||
|
assert node.type() == TYPE_LNK
|
||
|
return node.data
|
||
|
|
||
|
def unlink(self, node, name):
|
||
|
try:
|
||
|
del node.data[name]
|
||
|
except KeyError:
|
||
|
raise IOError(errno.ENOENT, name)
|
||
|
node.modified()
|
||
|
|
||
|
rmdir = unlink
|
||
|
|
||
|
def rename(self, oldnode, oldname, newnode, newname):
|
||
|
if newnode.type() != TYPE_DIR:
|
||
|
raise IOError(errno.ENOTDIR, newnode)
|
||
|
try:
|
||
|
nodeid = oldnode.data.pop(oldname)
|
||
|
except KeyError:
|
||
|
raise IOError(errno.ENOENT, oldname)
|
||
|
oldnode.modified()
|
||
|
newnode.data[newname] = nodeid
|
||
|
newnode.modified()
|
||
|
|
||
|
def modified(self, node):
|
||
|
node.modified()
|
||
|
|
||
|
# ____________________________________________________________
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
root = Node(newattr(stat.S_IFDIR), {})
|
||
|
handler = Handler('/home/arigo/mnt', Filesystem(root))
|
||
|
handler.loop_forever()
|