2 RouteFS is a base class for developing read-only FUSE filesystems that
3 lets you focus on the directory tree instead of the system calls.
5 RouteFS uses the Routes library developed for Pylons. URLs were
6 inspired by filesystems, and now you can have filesystems inspired by
9 When developing a descendent of RouteFS, any methods defined in that
10 class are considered "controllers", and receive any other parameters
11 specified by the URL as keyword arguments.
19 fuse.fuse_python_api = (0, 2)
21 class RouteStat(fuse.Stat):
23 RouteStat is a descendent of fuse.Stat, defined to make sure that
24 all of the necessary attributes are always defined
38 class RouteMeta(type):
40 Metaclass to calculate controller methods
42 Routes needs to be pre-seeded with a list of "controllers". For
43 all descendents of RouteFS, the list of controllers is defined to
44 be any non-private methods of the class that were not in the
47 def __init__(cls, classname, bases, dict_):
48 super(RouteMeta, cls).__init__(classname, bases, dict_)
49 if bases != (fuse.Fuse,):
50 new_funcs = set(dict_.keys()).difference(dir(RouteFS))
51 cls.controllers([func for func in new_funcs \
52 if not func.startswith('_')])
54 class RouteFS(fuse.Fuse):
56 RouteFS: Web 2.0 for filesystems
58 __metaclass__ = RouteMeta
59 def __init__(self, *args, **kwargs):
60 super(RouteFS, self).__init__(*args, **kwargs)
62 self.map = self.make_map()
63 self.map.create_regs(self.controller_list)
67 This method should be overridden by descendents of RouteFS to
68 define the routing for the filesystem
72 m.connect(':controller')
77 def controllers(cls, lst):
78 cls.controller_list = lst
80 def _get_file(self, path):
82 Find the filesystem entry object for a given path
84 match = self.map.match(path)
87 controller = match.pop('controller')
88 result = getattr(self, controller)(**match)
89 if type(result) is str:
91 if type(result) is list:
92 result = Directory(result)
95 def readdir(self, path, offset):
97 If the path referred to is a directory, return the elements of
100 return self._get_file(path).readdir(offset)
102 def getattr(self, path):
104 Return the stat information for a path
106 The stat information for a directory, symlink, or file is
107 predetermined based on which it is.
109 return self._get_file(path).getattr()
111 def read(self, path, length, offset):
113 If the path specified is a file, return the requested portion
116 return self._get_file(path).read(length, offset)
118 def readlink(self, path):
120 If the path specified is a symlink, return the target
122 return self._get_file(path).readlink()
124 class TreeKey(object):
127 def readdir(self, offset):
129 def read(self, length, offset):
134 class NoEntry(TreeKey):
137 def readdir(self, offset):
139 def read(self, length, offset):
144 class TreeEntry(TreeKey):
147 def __new__(cls, contents, mode=None):
148 return super(TreeEntry, cls).__new__(cls, contents)
150 def __init__(self, contents, mode=None):
152 self.mode = self.default_mode
156 super(TreeEntry, self).__init__(contents)
158 class Directory(TreeEntry, list):
160 A dummy class representing a filesystem entry that should be a
167 st.st_mode = stat.S_IFDIR | self.mode
171 def readdir(self, offset):
172 for member in ['.', '..'] + self:
173 yield fuse.Direntry(str(member))
175 class Symlink(TreeEntry, str):
177 A dummy class representing something that should be a symlink
183 st.st_mode = stat.S_IFLNK | self.mode
185 st.st_size = len(self)
191 class File(TreeEntry, str):
193 A dummy class representing something that should be a file
199 st.st_mode = stat.S_IFREG | self.mode
201 st.st_size = len(self)
204 def read(self, length, offset):
205 return self[offset:offset + length]
209 A convenience function for initializing a RouteFS filesystem
211 server = cls(version="%prog " + fuse.__version__,
212 usage=fuse.Fuse.fusage,
213 dash_s_do='setsingle')
214 server.parse(values=server, errex=1)
217 from dictfs import DictFS
219 __all__ = ['RouteFS', 'DictFS', 'Symlink', 'Directory', 'File', 'main']