Moved the EventQuerySet to its own class instead of nesting it in the Event model.
[philo.git] / middleware.py
1 from django.conf import settings
2 from django.contrib.sites.models import Site
3 from django.http import Http404
4 from philo.models import Node, View
5
6
7 class LazyNode(object):
8         def __get__(self, request, obj_type=None):
9                 if not hasattr(request, '_cached_node_path'):
10                         return None
11                 
12                 if not hasattr(request, '_found_node'):
13                         try:
14                                 current_site = Site.objects.get_current()
15                         except Site.DoesNotExist:
16                                 current_site = None
17                         
18                         path = request._cached_node_path
19                         trailing_slash = False
20                         if path[-1] == '/':
21                                 trailing_slash = True
22                         
23                         try:
24                                 node, subpath = Node.objects.get_with_path(path, root=getattr(current_site, 'root_node', None), absolute_result=False)
25                         except Node.DoesNotExist:
26                                 node = None
27                         
28                         if node:
29                                 if subpath is None:
30                                         subpath = ""
31                                 subpath = "/" + subpath
32                                 
33                                 if trailing_slash and subpath[-1] != "/":
34                                         subpath += "/"
35                                 
36                                 node.subpath = subpath
37                         
38                         request._found_node = node
39                 
40                 return request._found_node
41
42
43 class RequestNodeMiddleware(object):
44         """Middleware to process the request's path and attach the closest ancestor node."""
45         def process_request(self, request):
46                 request.__class__.node = LazyNode()
47         
48         def process_view(self, request, view_func, view_args, view_kwargs):
49                 request._cached_node_path = view_kwargs.get('path', '/')
50         
51         def process_exception(self, request, exception):
52                 if settings.DEBUG or not hasattr(request, 'node') or not request.node:
53                         return
54                 
55                 if isinstance(exception, Http404):
56                         error_view = request.node.attributes.get('Http404', None)
57                 else:
58                         error_view = request.node.attributes.get('Http500', None)
59                 
60                 if error_view is None or not isinstance(error_view, View):
61                         # Should this be duck-typing? Perhaps even no testing?
62                         return
63                 
64                 extra_context = {'exception': exception}
65                 return error_view.render_to_response(request, extra_context)