2
# Copyright (C) 2007-2008 The University of Melbourne
4
# This program is free software; you can redistribute it and/or modify
5
# it under the terms of the GNU General Public License as published by
6
# the Free Software Foundation; either version 2 of the License, or
7
# (at your option) any later version.
9
# This program is distributed in the hope that it will be useful,
10
# but WITHOUT ANY WARRANTY; without even the implied warranty of
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12
# GNU General Public License for more details.
14
# You should have received a copy of the GNU General Public License
15
# along with this program; if not, write to the Free Software
16
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
22
# This is a mod_python handler program. The correct way to call it is to have
23
# Apache send all requests to be handled by the module 'dispatch'.
25
# Top-level handler. Handles all requests to all pages in IVLE.
26
# Handles authentication (not authorization).
27
# Then passes the request along to the appropriate ivle app.
30
from mod_python import apache
42
from request import Request
46
from common import (util, forumutil)
48
import plugins.console
53
"""Handles a request which may be to anywhere in the site except media.
54
Intended to be called by mod_python, as a handler.
56
req: An Apache request object.
58
# Make the request object into an IVLE request which can be passed to apps
61
req = Request(req, html.write_html_head)
63
# Pass the apachereq to error reporter, since ivle req isn't created
65
handle_unknown_exception(apachereq, *sys.exc_info())
66
# Tell Apache not to generate its own errors as well
69
# Run the main handler, and catch all exceptions
71
return handler_(req, apachereq)
72
except mod_python.apache.SERVER_RETURN:
73
# An apache error. We discourage these, but they might still happen.
77
handle_unknown_exception(req, *sys.exc_info())
78
# Tell Apache not to generate its own errors as well
81
def handler_(req, apachereq):
83
Nested handler function. May raise exceptions. The top-level handler is
84
just used to catch exceptions.
85
Takes both an IVLE request and an Apache req.
87
# Hack? Try and get the user login early just in case we throw an error
88
# (most likely 404) to stop us seeing not logged in even when we are.
89
if not req.publicmode:
90
req.user = login.get_user_details(req)
92
# Check req.app to see if it is valid. 404 if not.
93
if req.app is not None and req.app not in conf.apps.app_url:
94
# Maybe it is a special app!
95
if req.app == 'logout':
98
req.throw_error(Request.HTTP_NOT_FOUND,
99
"There is no application called %s." % repr(req.app))
101
# Special handling for public mode - only allow the public app, call it
103
# NOTE: This will not behave correctly if the public app uses
104
# write_html_head_foot, but "serve" does not.
106
if req.app != conf.apps.public_app:
107
req.throw_error(Request.HTTP_FORBIDDEN,
108
"This application is not available on the public site.")
109
app = conf.apps.app_url[conf.apps.public_app]
110
apps.call_app(app.dir, req)
113
# app is the App object for the chosen app
115
app = conf.apps.app_url[conf.apps.default_app]
117
app = conf.apps.app_url[req.app]
119
# Check if app requires auth. If so, perform authentication and login.
120
# This will either return a User object, None, or perform a redirect
121
# which we will not catch here.
123
req.user = login.login(req)
124
logged_in = req.user is not None
126
req.user = login.get_user_details(req)
130
# Keep the user's session alive by writing to the session object.
131
# req.get_session().save()
132
# Well, it's a fine idea, but it creates considerable grief in the
133
# concurrent update department, so instead, we'll just make the
134
# sessions not time out.
136
# If user did not specify an app, HTTP redirect to default app and
139
req.throw_redirect(util.make_path(conf.apps.default_app))
141
# Set the default title to the app's tab name, if any. Otherwise URL
143
if app.name is not None:
148
# Call the specified app with the request object
149
apps.call_app(app.dir, req)
151
# if not logged in, login.login will have written the login box.
152
# Just clean up and exit.
154
# MAKE SURE we write the HTTP (and possibly HTML) header. This
155
# wouldn't happen if nothing else ever got written, so we have to make
157
req.ensure_headers_written()
159
# When done, write out the HTML footer if the app has requested it
160
if req.write_html_head_foot:
161
# Show the console if required
162
if logged_in and app.useconsole:
163
plugins.console.present(req, windowpane=True)
164
html.write_html_foot(req)
166
# Note: Apache will not write custom HTML error messages here.
167
# Use req.throw_error to do that.
171
"""Log out the current user (if any) by destroying the session state.
172
Then redirect to the top-level IVLE page."""
173
session = req.get_session()
176
req.add_cookie(forumutil.invalidated_forum_cookie())
177
req.throw_redirect(util.make_path(''))
179
def handle_unknown_exception(req, exc_type, exc_value, exc_traceback):
181
Given an exception that has just been thrown from IVLE, print its details
183
This is a full handler. It assumes nothing has been written, and writes a
185
req: May be EITHER an IVLE req or an Apache req.
186
IVLE reqs may have the HTML head/foot written (on a 400 error), but
187
the handler code may pass an apache req if an exception occurs before
188
the IVLE request is created.
190
req.content_type = "text/html"
191
logfile = os.path.join(conf.conf.log_path, 'ivle_error.log')
193
# For some reason, some versions of mod_python have "_server" instead of
194
# "main_server". So we check for both.
196
admin_email = apache.main_server.server_admin
197
except AttributeError:
199
admin_email = apache._server.server_admin
200
except AttributeError:
203
httpcode = exc_value.httpcode
204
req.status = httpcode
205
except AttributeError:
207
req.status = apache.HTTP_INTERNAL_SERVER_ERROR
209
login = req.user.login
210
except AttributeError:
215
logging.basicConfig(level=logging.INFO,
216
format='%(asctime)s %(levelname)s: ' +
217
'(HTTP: ' + str(req.status) +
218
', Ref: ' + str(login) + '@' +
219
str(socket.gethostname()) + str(req.uri) +
225
logging.debug('Logging Unhandled Exception')
227
# We handle 3 types of error.
228
# IVLEErrors with 4xx response codes (client error).
229
# IVLEErrors with 5xx response codes (handled server error).
230
# Other exceptions (unhandled server error).
231
# IVLEErrors should not have other response codes than 4xx or 5xx
232
# (eg. throw_redirect should have been used for 3xx codes).
233
# Therefore, that is treated as an unhandled error.
235
if (exc_type == util.IVLEError and httpcode >= 400
236
and httpcode <= 499):
237
# IVLEErrors with 4xx response codes are client errors.
238
# Therefore, these have a "nice" response (we even coat it in the IVLE
241
req.write_html_head_foot = True
242
req.write('<div id="ivle_padding">\n')
244
codename, msg = req.get_http_codename(httpcode)
245
except AttributeError:
246
codename, msg = None, None
247
# Override the default message with the supplied one,
249
if exc_value.message is not None:
250
msg = exc_value.message
251
if codename is not None:
252
req.write("<h1>Error: %s</h1>\n" % cgi.escape(codename))
254
req.write("<h1>Error</h1>\n")
256
req.write("<p>%s</p>\n" % cgi.escape(msg))
258
req.write("<p>An unknown error occured.</p>\n")
261
logging.info(str(msg))
263
req.write("<p>(HTTP error code %d)</p>\n" % httpcode)
265
req.write("<p>Warning: Could not open Error Log: '%s'</p>\n"
266
%cgi.escape(logfile))
267
req.write('</div>\n')
269
# A "bad" error message. We shouldn't get here unless IVLE
270
# misbehaves (which is currently very easy, if things aren't set up
272
# Write the traceback.
273
# If this is a non-4xx IVLEError, get the message and httpcode and
274
# make the error message a bit nicer (but still include the
276
# We also need to special-case IVLEJailError, as we can get another
277
# almost-exception out of it.
279
codename, msg = None, None
281
if exc_type is util.IVLEJailError:
282
msg = exc_value.type_str + ": " + exc_value.message
283
tb = 'Exception information extracted from IVLEJailError:\n'
284
tb += urllib.unquote(exc_value.info)
287
codename, msg = req.get_http_codename(httpcode)
288
except AttributeError:
290
# Override the default message with the supplied one,
292
if hasattr(exc_value, 'message') and exc_value.message is not None:
293
msg = exc_value.message
294
# Prepend the exception type
295
if exc_type != util.IVLEError:
296
msg = exc_type.__name__ + ": " + msg
298
tb = ''.join(traceback.format_exception(exc_type, exc_value,
302
logging.error('%s\n%s'%(str(msg), tb))
305
<head><title>IVLE Internal Server Error</title></head>
307
<h1>IVLE Internal Server Error""")
308
if (codename is not None
309
and httpcode != apache.HTTP_INTERNAL_SERVER_ERROR):
310
req.write(": %s" % cgi.escape(codename))
312
<p>An error has occured which is the fault of the IVLE developers or
316
req.write("<p>%s</p>\n" % cgi.escape(msg))
317
if httpcode is not None:
318
req.write("<p>(HTTP error code %d)</p>\n" % httpcode)
320
<p>Please report this to <a href="mailto:%s">%s</a> (the system
321
administrator). Include the following information:</p>
322
""" % (cgi.escape(admin_email), cgi.escape(admin_email)))
324
req.write("<pre>\n%s\n</pre>\n"%cgi.escape(tb))
326
req.write("<p>Warning: Could not open Error Log: '%s'</p>\n"
327
%cgi.escape(logfile))