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