source: package_branches/invirt-web/cherrypy/code/view.py @ 2615

Last change on this file since 2615 was 2532, checked in by quentin, 15 years ago

Clarify the operation of MakoHandler?

File size: 5.7 KB
RevLine 
[2485]1import os, sys
[2385]2
3import cherrypy
4from mako.template import Template
5from mako.lookup import TemplateLookup
6import simplejson
7import datetime, decimal
[2485]8from StringIO import StringIO
[2397]9from invirt.config import structs as config
[2399]10from webcommon import State
[2385]11
12class MakoHandler(cherrypy.dispatch.LateParamPageHandler):
[2532]13    """Callable which processes a dictionary, returning the rendered
14    body."""
[2385]15   
16    def __init__(self, template, next_handler, content_type='text/html; charset=utf-8'):
17        self.template = template
18        self.next_handler = next_handler
19        self.content_type = content_type
20   
21    def __call__(self):
22        env = globals().copy()
23        env.update(self.next_handler())
24        cherrypy.response.headers['Content-Type'] = self.content_type
25        return self.template.render(**env)
26       
27
28class MakoLoader(object):
29   
30    def __init__(self):
31        self.lookups = {}
[2485]32
33    def get_lookup(self, directories, module_directory=None,
34                     collection_size=-1, imports=[], **kwargs):
[2385]35        # Find the appropriate template lookup.
36        key = (tuple(directories), module_directory)
37        try:
38            lookup = self.lookups[key]
39        except KeyError:
40            lookup = TemplateLookup(directories=directories,
41                                    module_directory=module_directory,
42                                    collection_size=collection_size,
43                                    default_filters=['decode.utf8'],
44                                    input_encoding='utf-8',
45                                    output_encoding='utf-8',
[2402]46                                    imports=imports,
[2385]47                                    )
48            self.lookups[key] = lookup
[2485]49        return lookup
50
51    def __call__(self, filename, directories, module_directory=None,
52                 collection_size=-1, content_type='text/html; charset=utf-8',
53                 imports=[]):
54        cherrypy.request.lookup = lookup = self.get_lookup(directories, module_directory,
55                                                           collection_size, imports)
[2385]56        cherrypy.request.template = t = lookup.get_template(filename)
57        cherrypy.request.handler = MakoHandler(t, cherrypy.request.handler, content_type)
58
[2518]59cherrypy.tools.mako = cherrypy.Tool('on_start_resource', MakoLoader())
[2385]60
[2485]61def revertStandardError():
62    """Move stderr to stdout, and return the contents of the old stderr."""
63    errio = sys.stderr
64    if not isinstance(errio, StringIO):
65        return ''
66    sys.stderr = sys.stdout
67    errio.seek(0)
68    return errio.read()
69
70def catchStderr():
71    old_handler = cherrypy.request.handler
72    def wrapper(*args, **kwargs):
73        sys.stderr = StringIO()
74        ret = old_handler(*args, **kwargs)
75        e = revertStandardError()
76        if e:
77            if isinstance(ret, dict):
78                ret["error_text"] = e
79        return ret
80    if old_handler:
81        cherrypy.request.handler = wrapper
82
83cherrypy.tools.catch_stderr = cherrypy.Tool('before_handler', catchStderr)
84
[2385]85class JSONEncoder(simplejson.JSONEncoder):
86        def default(self, obj):
87                if isinstance(obj, datetime.datetime):
88                        return str(obj)
89                elif isinstance(obj, decimal.Decimal):
90                        return float(obj)
91                else:
92                        return simplejson.JSONEncoder.default(self, obj)
93
94def jsonify_tool_callback(*args, **kwargs):
95    if not cherrypy.request.cached:
96        response = cherrypy.response
97        response.headers['Content-Type'] = 'text/javascript'
98        response.body = JSONEncoder().iterencode(response.body)
99
100cherrypy.tools.jsonify = cherrypy.Tool('before_finalize', jsonify_tool_callback, priority=30)
101
[2391]102def require_login():
103    """If the user isn't logged in, raise 403 with an error."""
[2531]104    if cherrypy.request.login is False:
[2391]105        raise cherrypy.HTTPError(403,
106            "You are not authorized to access that resource")
107
[2397]108cherrypy.tools.require_login = cherrypy.Tool('on_start_resource', require_login, priority=150)
[2391]109
[2422]110def require_POST():
111    """If the request isn't a POST request, raise 405 Method Not Allowed"""
112    if cherrypy.request.method != "POST":
113        raise cherrypy.HTTPError(405,
114                                 "You must submit this request with POST")
115
116cherrypy.tools.require_POST = cherrypy.Tool('on_start_resource', require_POST, priority=150)
117
[2397]118def remote_user_login():
[2531]119    """Get the current user based on the SSL or GSSAPI environment
120variables and store it in the request object's login variable. This
121conforms to the CherryPy API:
122http://www.cherrypy.org/wiki/RequestObject#login
123
124If the user is logged in successfully, cherrypy.request.login is set
125to the username. If the user failed to log in, cherrypy.request.login
126is set to False. If the user did not attempt authentication,
127cherrypy.request.login is set to None."""
[2397]128    environ = cherrypy.request.wsgi_environ
129    user = environ.get('REMOTE_USER')
130    if user is None:
131        return
[2531]132    else:
133        cherrypy.request.login = None # clear what cherrypy put there
[2397]134
135    if environ.get('AUTH_TYPE') == 'Negotiate':
136        # Convert the krb5 principal into a krb4 username
137        if not user.endswith('@%s' % config.kerberos.realm):
[2531]138            cherrypy.request.login = False # failed to login
[2397]139        else:
140            cherrypy.request.login = user.split('@')[0].replace('/', '.')
141    else:
142        cherrypy.request.login = user
143
144cherrypy.tools.remote_user_login = cherrypy.Tool('on_start_resource', remote_user_login, priority=50)
145
[2399]146def invirtwebstate_init():
147    """Initialize the cherrypy.request.state object from Invirt"""
[2482]148    if not hasattr(cherrypy.request, "state"):
149        cherrypy.request.state = State(cherrypy.request.login)
[2399]150
151cherrypy.tools.invirtwebstate = cherrypy.Tool('on_start_resource', invirtwebstate_init, priority=100)
152
[2385]153class View(object):
154    _cp_config = {'tools.mako.directories': [os.path.join(os.path.dirname(__file__),'templates')]}
Note: See TracBrowser for help on using the repository browser.