django-journal/django_journal/models.py

169 lines
5.0 KiB
Python

from django.db import models
from django.contrib.contenttypes.models import ContentType
from django.contrib.contenttypes import generic
from django.core.exceptions import ObjectDoesNotExist
from django.utils.translation import ugettext_lazy as _
class TagManager(models.Manager):
def get_by_natural_key(self, name):
return self.get(name=name)
class Tag(models.Model):
'''Tag allows typing event and data linked to events.
name:
the string identifier of the tag
'''
objects = TagManager()
name = models.CharField(max_length=32, unique=True, db_index=True)
def __unicode__(self):
return self.name
def natural_key(self):
return (self.name,)
class Meta:
ordering = ('name',)
verbose_name = _('Journal tag')
class TemplateManager(models.Manager):
def get_by_natural_key(self, content):
return self.get(content=content)
class Template(models.Model):
'''Template for formatting an event.
ex.: Template(
content='{user1} gave group {group} to {user2}')
'''
objects = TemplateManager()
content = models.TextField(unique=True, db_index=True)
def __unicode__(self):
return self.name
def natural_key(self):
return (self.content,)
class Meta:
ordering = ('content',)
class JournalManager(models.Manager):
def for_object(self, obj):
'''Return Journal records linked to this object.'''
content_type = ContentType.objects.get_for_model(obj)
return self.filter(journalobjectdata__content_type=content_type,
journalobjectdata__object_id=obj.pk)
def for_tag(self, tag):
'''Returns Journal records linked to this tag by their own tag or
the tag on their data records.
'''
if not isinstance(tag, Tag):
try:
tag = Tag.objects.get(name=tag)
except Tag.DoesNotExist:
return self.none()
# always remember: multiple join (OR in WHERE) produces duplicate
# lines ! Use .distinct() for safety.
return self.filter(models.Q(tag=tag)|
models.Q(objectdata__tag=tag)|
models.Q(stringdata__tag=tag)) \
.distinct()
class Journal(models.Model):
'''One line of the journal.
Each recorded event in the journal is a Journal instance.
time - the time at which the event was recorded
tag - the tag giving the type of event
template - a format string to present the event
message - a simple string representation of the event, computed using
the template and associated datas.
'''
objects = JournalManager()
time = models.DateTimeField(auto_now_add=True, db_index=True)
tag = models.ForeignKey(Tag, on_delete=models.PROTECT)
template = models.ForeignKey(Template, on_delete=models.PROTECT)
message = models.CharField(max_length=128, db_index=True)
class Meta:
ordering = ('time', 'id')
verbose_name = _('Journal entry')
verbose_name_plural = _('Journal entries')
def message_context(self):
ctx = {}
for data in self.objectdata_set.all():
try:
ctx[data.tag.name] = unicode(data.content_object)
except ObjectDoesNotExist:
ctx[data.tag.name] = '<deleted>'
for data in self.stringdata_set.all():
ctx[data.tag.name] = data.content
return ctx
def __unicode__(self):
if len(self.message) < 125:
return self.message
ctx = self.message_context()
return self.template.content.format(**ctx)
def __repr__(self):
return '<Journal pk:{0} tag:{1} message:{2}>'.format(
self.pk, unicode(self.tag).encode('utf-8'),
unicode(self.message).encode('utf-8'))
class StringData(models.Model):
'''String data associated to a recorded event.
journal:
the recorded event
tag:
the identifier for this data
content:
the string value of the data
'''
journal = models.ForeignKey(Journal)
tag = models.ForeignKey(Tag)
content = models.TextField(db_index=True)
class Meta:
unique_together = (('journal', 'tag'),)
verbose_name = _('Linked text string')
class ObjectData(models.Model):
'''Object data associated with a recorded event.
journal:
the recorded event
tag:
the identifier for this data
content_object:
the object value of the data
'''
journal = models.ForeignKey(Journal)
tag = models.ForeignKey(Tag)
content_type = models.ForeignKey(ContentType)
object_id = models.PositiveIntegerField(db_index=True)
content_object = generic.GenericForeignKey('content_type',
'object_id')
class Meta:
unique_together = (('journal', 'tag'),)
verbose_name = _('Linked object')
def __unicode__(self):
return u'{}:{}:{}'.format(self.journal.id, self.tag, self.content_object)