1 from django.db import models
2 from django.contrib.contenttypes.models import ContentType
3 from django.contrib.contenttypes import generic
4 from django.utils import simplejson as json
5 from django.core.exceptions import ObjectDoesNotExist
6 from philo.utils import ContentTypeRegistryLimiter
7 from UserDict import DictMixin
10 class Tag(models.Model):
11 name = models.CharField(max_length=250)
12 slug = models.SlugField()
14 def __unicode__(self):
18 class Titled(models.Model):
19 title = models.CharField(max_length=255)
20 slug = models.SlugField()
22 def __unicode__(self):
29 class Attribute(models.Model):
30 entity_content_type = models.ForeignKey(ContentType, verbose_name='Entity type')
31 entity_object_id = models.PositiveIntegerField(verbose_name='Entity ID')
32 entity = generic.GenericForeignKey('entity_content_type', 'entity_object_id')
33 key = models.CharField(max_length=255)
34 json_value = models.TextField(verbose_name='Value (JSON)', help_text='This value must be valid JSON.')
37 return json.loads(self.json_value)
39 def set_value(self, value):
40 self.json_value = json.dumps(value)
42 def delete_value(self):
43 self.json_value = json.dumps(None)
45 value = property(get_value, set_value, delete_value)
47 def __unicode__(self):
48 return u'"%s": %s' % (self.key, self.value)
54 value_content_type_limiter = ContentTypeRegistryLimiter()
57 def register_value_model(model):
58 value_content_type_limiter.register_class(model)
61 def unregister_value_model(model):
62 value_content_type_limiter.unregister_class(model)
66 class Relationship(models.Model):
67 entity_content_type = models.ForeignKey(ContentType, related_name='relationship_entity_set', verbose_name='Entity type')
68 entity_object_id = models.PositiveIntegerField(verbose_name='Entity ID')
69 entity = generic.GenericForeignKey('entity_content_type', 'entity_object_id')
70 key = models.CharField(max_length=255)
71 value_content_type = models.ForeignKey(ContentType, related_name='relationship_value_set', limit_choices_to=value_content_type_limiter, verbose_name='Value type')
72 value_object_id = models.PositiveIntegerField(verbose_name='Value ID')
73 value = generic.GenericForeignKey('value_content_type', 'value_object_id')
75 def __unicode__(self):
76 return u'"%s": %s' % (self.key, self.value)
82 class QuerySetMapper(object, DictMixin):
83 def __init__(self, queryset, passthrough=None):
84 self.queryset = queryset
85 self.passthrough = passthrough
87 def __getitem__(self, key):
89 return self.queryset.get(key__exact=key).value
90 except ObjectDoesNotExist:
91 if self.passthrough is not None:
92 return self.passthrough.__getitem__(key)
96 keys = set(self.queryset.values_list('key', flat=True).distinct())
97 if self.passthrough is not None:
98 keys |= set(self.passthrough.keys())
102 class Entity(models.Model):
103 attribute_set = generic.GenericRelation(Attribute, content_type_field='entity_content_type', object_id_field='entity_object_id')
104 relationship_set = generic.GenericRelation(Relationship, content_type_field='entity_content_type', object_id_field='entity_object_id')
107 def attributes(self):
108 return QuerySetMapper(self.attribute_set)
111 def relationships(self):
112 return QuerySetMapper(self.relationship_set)
119 class TreeManager(models.Manager):
120 use_for_related_fields = True
123 return self.filter(parent__isnull=True)
125 def get_with_path(self, path, root=None, absolute_result=True, pathsep='/'):
127 Returns the object with the path, or None if there is no object with that path,
128 unless absolute_result is set to False, in which case it returns a tuple containing
129 the deepest object found along the path, and the remainder of the path after that
130 object as a string (or None in the case that there is no remaining path).
132 slugs = path.split(pathsep)
134 remaining_slugs = list(slugs)
137 remaining_slugs.remove(slug)
138 if slug: # ignore blank slugs, handles for multiple consecutive pathseps
140 obj = self.get(slug__exact=slug, parent__exact=obj)
141 except self.model.DoesNotExist:
144 remaining_slugs.insert(0, slug)
145 remainder = pathsep.join(remaining_slugs)
151 return (obj, remainder)
152 raise self.model.DoesNotExist('%s matching query does not exist.' % self.model._meta.object_name)
155 class TreeModel(models.Model):
156 objects = TreeManager()
157 parent = models.ForeignKey('self', related_name='children', null=True, blank=True)
158 slug = models.SlugField()
160 def get_path(self, pathsep='/', field='slug'):
161 path = getattr(self, field, '?')
164 path = getattr(parent, field, '?') + pathsep + path
165 parent = parent.parent
167 path = property(get_path)
169 def __unicode__(self):
173 unique_together = (('parent', 'slug'),)
178 class TreeEntity(TreeModel, Entity):
180 def attributes(self):
182 return QuerySetMapper(self.attribute_set, passthrough=self.parent.attributes)
183 return super(TreeEntity, self).attributes
186 def relationships(self):
188 return QuerySetMapper(self.relationship_set, passthrough=self.parent.relationships)
189 return super(TreeEntity, self).relationships