44
45
import ivle.conf.apps
45
46
from ivle.dispatch.request import Request
46
47
from ivle.dispatch import login
48
from ivle.webapp.base.plugins import ViewPlugin
49
from ivle.webapp.errors import HTTPError
49
import plugins.console # XXX: Relies on www/ being in the Python path.
51
53
# XXX List of plugins, which will eventually be read in from conf
55
'ivle.webapp.core#Plugin',
53
56
'ivle.webapp.admin.user#Plugin',
54
57
'ivle.webapp.tutorial#Plugin',
55
58
'ivle.webapp.admin.subject#Plugin',
56
'ivle.webapp.browser#Plugin',
59
'ivle.webapp.filesystem.browser#Plugin',
60
'ivle.webapp.filesystem.diff#Plugin',
61
'ivle.webapp.filesystem.svnlog#Plugin',
57
62
'ivle.webapp.groups#Plugin',
58
63
'ivle.webapp.console#Plugin',
59
64
'ivle.webapp.security#Plugin',
65
'ivle.webapp.media#Plugin',
66
'ivle.webapp.forum#Plugin',
62
def get_routes_mapper():
69
def generate_route_mapper(view_plugins):
64
71
Build a Mapper object for doing URL matching using 'routes', based on the
72
given plugin registry.
67
74
m = routes.Mapper(explicit=True)
68
for pluginstr in plugins_HACK:
69
plugin_path, classname = pluginstr.split('#')
70
# Load the plugin module from somewhere in the Python path
71
# (Note that plugin_path is a fully-qualified Python module name).
72
plugin = getattr(__import__(plugin_path, fromlist=[classname]),
75
for plugin in view_plugins:
74
76
# Establish a URL pattern for each element of plugin.urls
77
assert hasattr(plugin, 'urls'), "%r does not have any urls" % plugin
75
78
for url in plugin.urls:
77
80
view_class = url[1]
79
82
m.connect(routex, view=view_class, **kwargs_dict)
85
def get_plugin(pluginstr):
86
plugin_path, classname = pluginstr.split('#')
87
# Load the plugin module from somewhere in the Python path
88
# (Note that plugin_path is a fully-qualified Python module name).
90
getattr(__import__(plugin_path, fromlist=[classname]), classname))
83
93
"""Handles a request which may be to anywhere in the site except media.
84
94
Intended to be called by mod_python, as a handler.
123
133
# XXX This should be done ONCE per Python process, not per request.
124
134
# (Wait till WSGI)
125
135
# XXX No authentication is done here
126
req.mapper = get_routes_mapper()
136
req.plugins = dict([get_plugin(pluginstr) for pluginstr in plugins_HACK])
137
# Index the plugins by base class
138
req.plugin_index = {}
139
for plugin in req.plugins.values():
140
# Getmro returns a tuple of all the super-classes of the plugin
141
for base in inspect.getmro(plugin):
142
if base not in req.plugin_index:
143
req.plugin_index[base] = []
144
req.plugin_index[base].append(plugin)
145
req.reverse_plugins = dict([(v, k) for (k, v) in req.plugins.items()])
146
req.mapper = generate_route_mapper(req.plugin_index[ViewPlugin])
127
148
matchdict = req.mapper.match(req.uri)
128
149
if matchdict is not None:
129
150
viewcls = matchdict['view']
131
152
# (The latter two seem to be built-in, and we don't want them).
132
153
kwargs = matchdict.copy()
133
154
del kwargs['view']
134
# Instantiate the view, which should be a BaseView class
135
view = viewcls(req, **kwargs)
156
# Instantiate the view, which should be a BaseView class
157
view = viewcls(req, **kwargs)
161
# A view explicitly raised an HTTP error. Respect it.
164
# Try to find a custom error view.
165
if hasattr(viewcls, 'get_error_view'):
166
errviewcls = viewcls.get_error_view(e)
171
errview = errviewcls(req, e)
178
# A non-HTTPError appeared. We have an unknown exception. Panic.
179
handle_unknown_exception(req, *sys.exc_info())
140
184
### END New plugins framework ###
142
186
# Check req.app to see if it is valid. 404 if not.
206
250
# When done, write out the HTML footer if the app has requested it
207
251
if req.write_html_head_foot:
208
# Show the console if required
209
if logged_in and app.useconsole:
210
plugins.console.present(req, windowpane=True)
211
252
html.write_html_foot(req)
213
254
# Note: Apache will not write custom HTML error messages here.