+class InheritableTreeEntity(TreeEntity):
+ instance_type = models.ForeignKey(ContentType, editable=False)
+
+ def save(self, force_insert=False, force_update=False):
+ if not hasattr(self, 'instance_type_ptr'):
+ self.instance_type = ContentType.objects.get_for_model(self.__class__)
+ super(InheritableTreeEntity, self).save(force_insert, force_update)
+
+ @property
+ def instance(self):
+ return self.instance_type.get_object_for_this_type(id=self.id)
+
+ def get_path(self, pathsep='/', field='slug'):
+ path = getattr(self.instance, field, '?')
+ parent = self.parent
+ while parent:
+ path = getattr(parent.instance, field, '?') + pathsep + path
+ parent = parent.parent
+ return path
+ path = property(get_path)
+
+ @property
+ def attributes(self):
+ if self.parent:
+ return QuerySetMapper(self.instance.attribute_set, passthrough=self.parent.instance.attributes)
+ return QuerySetMapper(self.instance.attribute_set)
+
+ @property
+ def relationships(self):
+ if self.parent:
+ return QuerySetMapper(self.instance.relationship_set, passthrough=self.parent.instance.relationships)
+ return QuerySetMapper(self.instance.relationship_set)
+
+ class Meta:
+ abstract = True
+
+
+class Node(InheritableTreeEntity):
+ accepts_subpath = False
+
+ def render_to_response(self, request, path=None, subpath=None):
+ return HttpResponseServerError()
+
+ class Meta:
+ unique_together = (('parent', 'slug'),)
+
+
+class MultiNode(Node):
+ accepts_subpath = True
+
+ urlpatterns = []
+
+ def render_to_response(self, request, path=None, subpath=None):
+ if not subpath:
+ subpath = ""
+ subpath = "/" + subpath
+ from django.core.urlresolvers import resolve
+ view, args, kwargs = resolve(subpath, urlconf=self)
+ return view(request, *args, **kwargs)
+
+ class Meta:
+ abstract = True
+
+
+class Redirect(Node):
+ STATUS_CODES = (
+ (302, 'Temporary'),
+ (301, 'Permanent'),
+ )
+ target = models.URLField(help_text='Must be a valid, absolute URL (i.e. http://)')
+ status_code = models.IntegerField(choices=STATUS_CODES, default=302, verbose_name='redirect type')
+
+ def render_to_response(self, request, path=None, subpath=None):
+ response = HttpResponseRedirect(self.target)
+ response.status_code = self.status_code
+ return response
+
+
+class File(Node):
+ """ For storing arbitrary files """
+ mimetype = models.CharField(max_length=255)
+ file = models.FileField(upload_to='philo/files/%Y/%m/%d')
+
+ def render_to_response(self, request, path=None, subpath=None):
+ wrapper = FileWrapper(self.file)
+ response = HttpResponse(wrapper, content_type=self.mimetype)
+ response['Content-Length'] = self.file.size
+ return response
+
+ def __unicode__(self):
+ return self.file
+
+