MultiViews now clear_url_caches before resolving URLs, needed to allow them to change...
[philo.git] / models / nodes.py
1 from django.db import models
2 from django.contrib.contenttypes.models import ContentType
3 from django.contrib.contenttypes import generic
4 from django.contrib.sites.models import Site
5 from django.http import HttpResponse, HttpResponseServerError, HttpResponseRedirect
6 from django.core.servers.basehttp import FileWrapper
7 from django.core.urlresolvers import resolve, clear_url_caches
8 from django.template import add_to_builtins as register_templatetags
9 from inspect import getargspec
10 from philo.models.base import TreeEntity, Entity, QuerySetMapper
11 from philo.utils import ContentTypeSubclassLimiter
12 from philo.validators import RedirectValidator
13 from philo.exceptions import ViewDoesNotProvideSubpaths
14
15
16 _view_content_type_limiter = ContentTypeSubclassLimiter(None)
17
18
19 class Node(TreeEntity):
20         view_content_type = models.ForeignKey(ContentType, related_name='node_view_set', limit_choices_to=_view_content_type_limiter)
21         view_object_id = models.PositiveIntegerField()
22         view = generic.GenericForeignKey('view_content_type', 'view_object_id')
23         
24         @property
25         def accepts_subpath(self):
26                 if self.view:
27                         return self.view.accepts_subpath
28                 return False
29         
30         def render_to_response(self, request, path=None, subpath=None, extra_context=None):
31                 return self.view.render_to_response(self, request, path, subpath, extra_context)
32         
33         class Meta:
34                 app_label = 'philo'
35
36
37 # the following line enables the selection of a node as the root for a given django.contrib.sites Site object
38 models.ForeignKey(Node, related_name='sites', null=True, blank=True).contribute_to_class(Site, 'root_node')
39
40
41 class View(Entity):
42         nodes = generic.GenericRelation(Node, content_type_field='view_content_type', object_id_field='view_object_id')
43         
44         accepts_subpath = False
45         
46         def get_subpath(self, obj):
47                 raise ViewDoesNotProvideSubpaths
48         
49         def attributes_with_node(self, node):
50                 return QuerySetMapper(self.attribute_set, passthrough=node.attributes)
51         
52         def relationships_with_node(self, node):
53                 return QuerySetMapper(self.relationship_set, passthrough=node.relationships)
54         
55         def render_to_response(self, node, request, path=None, subpath=None, extra_context=None):
56                 raise NotImplementedError('View subclasses must implement render_to_response.')
57         
58         class Meta:
59                 abstract = True
60
61
62 _view_content_type_limiter.cls = View
63
64
65 class MultiView(View):
66         accepts_subpath = True
67         
68         urlpatterns = []
69         
70         def render_to_response(self, node, request, path=None, subpath=None, extra_context=None):
71                 clear_url_caches()
72                 if not subpath:
73                         subpath = ""
74                 subpath = "/" + subpath
75                 view, args, kwargs = resolve(subpath, urlconf=self)
76                 view_args = getargspec(view)[0]
77                 if extra_context is not None and 'extra_context' in view_args:
78                         if 'extra_context' in kwargs:
79                                 extra_context.update(kwargs['extra_context'])
80                         kwargs['extra_context'] = extra_context
81                 if 'node' in view_args:
82                         kwargs['node'] = node
83                 return view(request, *args, **kwargs)
84         
85         class Meta:
86                 abstract = True
87
88
89 class Redirect(View):
90         STATUS_CODES = (
91                 (302, 'Temporary'),
92                 (301, 'Permanent'),
93         )
94         target = models.CharField(max_length=200, validators=[RedirectValidator()])
95         status_code = models.IntegerField(choices=STATUS_CODES, default=302, verbose_name='redirect type')
96         
97         def render_to_response(self, node, request, path=None, subpath=None, extra_context=None):
98                 response = HttpResponseRedirect(self.target)
99                 response.status_code = self.status_code
100                 return response
101         
102         class Meta:
103                 app_label = 'philo'
104
105
106 class File(View):
107         """ For storing arbitrary files """
108         
109         mimetype = models.CharField(max_length=255)
110         file = models.FileField(upload_to='philo/files/%Y/%m/%d')
111         
112         def render_to_response(self, node, request, path=None, subpath=None, extra_context=None):
113                 wrapper = FileWrapper(self.file)
114                 response = HttpResponse(wrapper, content_type=self.mimetype)
115                 response['Content-Length'] = self.file.size
116                 return response
117         
118         class Meta:
119                 app_label = 'philo'
120         
121         def __unicode__(self):
122                 return self.file.name
123
124
125 register_templatetags('philo.templatetags.nodes')