src/cm/models.py
author Simon Descarpentries <sid@sopinspace.com>
Tue, 06 May 2014 13:52:01 +0200
changeset 651 9bbc657f6837
parent 605 830993bdf983
permissions -rw-r--r--
Replace DISABLE_TRACKING and TRACKING_HTML by a TRACKING_ID variable in configuration files
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
156
6d447220fd1e bug fix: removed \r in textversion's content because of a comment position bug in pre_edit. Added a save function to TextVersion to do so.
rbernard
parents: 155
diff changeset
     1
import re
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
     2
from cm.converters.pandoc_converters import \
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
     3
    CHOICES_INPUT_FORMATS as CHOICES_INPUT_FORMATS_PANDOC, \
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
     4
    DEFAULT_INPUT_FORMAT as DEFAULT_INPUT_FORMAT_PANDOC, pandoc_convert
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
     5
from cm.models_base import PermanentModel, KeyManager, Manager, KeyModel, AuthorModel
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
     6
from cm.models_utils import *
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
     7
from cm.utils.dj import absolute_reverse
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
     8
from cm.utils.date import datetime_to_user_str
175
4f072edc51a1 BUG FIX : handling html
rbernard
parents: 160
diff changeset
     9
from cm.utils.html import on_content_receive
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    10
from cm.utils.comment_positioning import compute_new_comment_positions
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    11
from django import forms
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    12
from django.db.models import Q
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    13
from django.template.loader import render_to_string
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    14
from django.conf import settings
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    15
from django.template import RequestContext
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    16
from django.contrib.auth.models import Permission
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    17
from django.contrib.contenttypes import generic
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    18
from django.contrib.contenttypes.models import ContentType
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    19
from django.core.files.base import ContentFile
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    20
from django.core.urlresolvers import reverse
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    21
from django.template.defaultfilters import timesince
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    22
from django.db import models
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    23
from django.utils.translation import ugettext as _, ugettext_lazy, ugettext_noop
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    24
from tagging.fields import TagField
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    25
import pickle
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    26
from django.db import connection
72
b0c777412d0a keep comment dates (created / modified) when duplicating (fixes #12)
raph
parents: 38
diff changeset
    27
from datetime import datetime
605
830993bdf983 Clear cache when creating/deleting text/comment/user.
gibus
parents: 518
diff changeset
    28
from django.core.cache import cache
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    29
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    30
class TextManager(Manager):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    31
    def create_text(self, title, format, content, note, name, email, tags, user=None, state='approved', **kwargs):
175
4f072edc51a1 BUG FIX : handling html
rbernard
parents: 160
diff changeset
    32
        content = on_content_receive(content, format)
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    33
        text = self.create(name=name, email=email, user=user, state=state)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    34
        text_version = TextVersion.objects.create(title=title, format=format, content=content, text=text, note=note, name=name, email=email, tags=tags, user=user)
605
830993bdf983 Clear cache when creating/deleting text/comment/user.
gibus
parents: 518
diff changeset
    35
        cache.clear()
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    36
        return text
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    37
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    38
    def create_new_version(self, text, title, format, content, note, name, email, tags, user=None, **kwargs):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    39
        text_version = TextVersion.objects.create(title=title, format=format, content=content, text=text, note=note, name=name, email=email, tags=tags, user=user)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    40
        return text_version
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    41
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    42
class Text(PermanentModel, AuthorModel):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    43
    modified = models.DateTimeField(auto_now=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    44
    created = models.DateTimeField(auto_now_add=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    45
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    46
    private_feed_key = models.CharField(max_length=20, db_index=True, unique=True, blank=True, null=True, default=None)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    47
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    48
    # denormalized fields
407
2d14a80716e2 When last_version is deleted, do not delete text and previous versions in cascade, but really delete text only when there is no previous version, otherwise update text.last_version with previous version.
gibus
parents: 347
diff changeset
    49
    last_text_version = models.ForeignKey("TextVersion", related_name='related_text', null=True, blank=True, on_delete=models.SET_NULL)
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    50
    title = models.TextField()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    51
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    52
    objects = TextManager()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    53
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    54
    def get_latest_version(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    55
        return self.last_text_version
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    56
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    57
    def fetch_latest_version(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    58
        versions = self.get_versions()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    59
        if versions:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    60
            return versions[0]
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    61
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    62
            return None
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    63
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    64
    def update_denorm_fields(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    65
        real_last_text_version = self.fetch_latest_version()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    66
    
289
285e67006eb0 fix error when last text_version is deleted
raph
parents: 266
diff changeset
    67
        try:
285e67006eb0 fix error when last text_version is deleted
raph
parents: 266
diff changeset
    68
            last_text_version = self.last_text_version
285e67006eb0 fix error when last text_version is deleted
raph
parents: 266
diff changeset
    69
        except TextVersion.DoesNotExist:
285e67006eb0 fix error when last text_version is deleted
raph
parents: 266
diff changeset
    70
            # the text version has just been deleted
285e67006eb0 fix error when last text_version is deleted
raph
parents: 266
diff changeset
    71
            last_text_version = None
285e67006eb0 fix error when last text_version is deleted
raph
parents: 266
diff changeset
    72
            
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    73
        modif = False
289
285e67006eb0 fix error when last text_version is deleted
raph
parents: 266
diff changeset
    74
        if real_last_text_version and real_last_text_version != last_text_version:
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    75
            self.last_text_version = real_last_text_version
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    76
            modif = True
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    77
            
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    78
        if real_last_text_version and real_last_text_version.title and real_last_text_version.title != self.title:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    79
            self.title = real_last_text_version.title
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    80
            modif = True
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    81
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    82
        if real_last_text_version and real_last_text_version.modified != self.modified:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    83
            self.modified = real_last_text_version.modified
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    84
            modif = True
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    85
            
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    86
        if modif:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    87
            self.save()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    88
407
2d14a80716e2 When last_version is deleted, do not delete text and previous versions in cascade, but really delete text only when there is no previous version, otherwise update text.last_version with previous version.
gibus
parents: 347
diff changeset
    89
        # GIB: there is no more version for this text => delete it
2d14a80716e2 When last_version is deleted, do not delete text and previous versions in cascade, but really delete text only when there is no previous version, otherwise update text.last_version with previous version.
gibus
parents: 347
diff changeset
    90
        if real_last_text_version == None and last_text_version == None:
2d14a80716e2 When last_version is deleted, do not delete text and previous versions in cascade, but really delete text only when there is no previous version, otherwise update text.last_version with previous version.
gibus
parents: 347
diff changeset
    91
          self.real_delete()
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    92
                
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    93
    def get_title(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    94
        return self.get_latest_version().title
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    95
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    96
    def get_versions(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    97
        """
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    98
        Versions with most recent first
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
    99
        """
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   100
        versions = TextVersion.objects.filter(text__exact=self).order_by('-created')
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   101
        # TODO: use new postgresql 8.4 row_number as extra select to do that
145
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   102
        #for index in xrange(len(versions)):
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   103
        #    v = versions[index]
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   104
        #    # version_number is 1-based
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   105
        #    setattr(v, 'version_number', len(versions) - index)
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   106
        return versions
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   107
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   108
    def get_version(self, version_number):        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   109
        """
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   110
        Get version number 'version_number' (1-based)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   111
        """
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   112
        version = TextVersion.objects.filter(text__exact=self).order_by('created')[version_number - 1:version_number][0]
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   113
        return version
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   114
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   115
    def get_inversed_versions(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   116
        versions = TextVersion.objects.filter(text__exact=self).order_by('created')
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   117
        # TODO: use new postgresql 8.4 row_number as extra select to do that
145
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   118
        #for index in xrange(len(versions)):
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   119
        #    v = versions[index]
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   120
        #    # version_number is 1-based
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   121
        #    setattr(v, 'version_number', index + 1)
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   122
        return versions
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   123
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   124
    def get_versions_number(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   125
        return self.get_versions().count()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   126
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   127
    def is_admin(self, adminkey=None):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   128
        if adminkey and self.adminkey == adminkey:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   129
            return True
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   130
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   131
            return False
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   132
145
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   133
    def revert_to_version(self, text_version_key):
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   134
        text_version = TextVersion.objects.get(key=text_version_key)
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   135
        if text_version.text != self:
fb5b1e6ad179 refactor version explorer
raph
parents: 140
diff changeset
   136
            raise Exception('Invalid revert attempt')
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   137
        new_text_version = TextVersion.objects.duplicate(text_version, True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   138
        return new_text_version
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   139
        
103
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   140
    def edit(self, new_title, new_format, new_content, new_tags=None, new_note=None, keep_comments=True, cancel_modified_scopes=True, new_version=True):
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   141
        text_version = self.get_latest_version()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   142
            
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   143
        if new_version:        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   144
            text_version = TextVersion.objects.duplicate(text_version, keep_comments)
103
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   145
        text_version.edit(new_title, new_format, new_content, new_tags, new_note, keep_comments, cancel_modified_scopes)        
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   146
        return text_version 
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   147
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   148
    def __unicode__(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   149
        return self.title    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   150
514
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   151
DEFAULT_INPUT_FORMAT = getattr(settings, 'DEFAULT_INPUT_FORMAT', DEFAULT_INPUT_FORMAT_PANDOC)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   152
CHOICES_INPUT_FORMATS = getattr(settings, 'CHOICES_INPUT_FORMATS', CHOICES_INPUT_FORMATS_PANDOC)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   153
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   154
class TextVersionManager(KeyManager):
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   155
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   156
    def duplicate(self, text_version, duplicate_comments=True):
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   157
        old_comment_set = set(text_version.comment_set.all())
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   158
        text_version.id = None
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   159
        
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   160
        # generate new key
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   161
        text_version.key = self._gen_key()
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   162
        text_version.adminkey = self._gen_adminkey()
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   163
        
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   164
        text_version.save()
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   165
        
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   166
        duplicate_text_version = text_version
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   167
        
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   168
        if duplicate_comments:
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   169
            old_comment_map = {}
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   170
            while len(old_comment_set):
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   171
                for c in old_comment_set:
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   172
                    if not c.reply_to or c.reply_to.id in old_comment_map:
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   173
                        old_id = c.id
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   174
                        old_comment_set.remove(c)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   175
                        reply_to = None
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   176
                        if c.reply_to:                            
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   177
                            reply_to = old_comment_map[c.reply_to.id]  
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   178
                        c2 = Comment.objects.duplicate(c, duplicate_text_version, reply_to, keep_dates=True)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   179
                        old_comment_map[old_id] = c2
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   180
                        break
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   181
                 
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   182
        return duplicate_text_version
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   183
        
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   184
class TextVersion(AuthorModel, KeyModel):
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   185
    modified = models.DateTimeField(auto_now=True)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   186
    created = models.DateTimeField(auto_now_add=True)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   187
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   188
    title = models.TextField(ugettext_lazy("Title"))
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   189
    format = models.CharField(ugettext_lazy("Format"), max_length=20, blank=False, default=DEFAULT_INPUT_FORMAT, choices=CHOICES_INPUT_FORMATS)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   190
    content = models.TextField(ugettext_lazy("Content"))
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   191
    tags = TagField(ugettext_lazy("Tags"), max_length=1000)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   192
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   193
    note = models.CharField(ugettext_lazy("Note"), max_length=100, null=True, blank=True)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   194
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   195
    mod_posteriori = models.BooleanField(ugettext_lazy('Moderation a posteriori?'), default=True)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   196
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   197
    from django.utils.safestring import mark_safe
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   198
518
9172f4c6ad70 Better translation of help text
gibus
parents: 515
diff changeset
   199
    category_1 = models.CharField(ugettext_lazy("Label for the first category of comments"), max_length=20, null=True, blank=True)
9172f4c6ad70 Better translation of help text
gibus
parents: 515
diff changeset
   200
    category_2 = models.CharField(ugettext_lazy("Label for the second category of comments"), max_length=20, null=True, blank=True)
9172f4c6ad70 Better translation of help text
gibus
parents: 515
diff changeset
   201
    category_3 = models.CharField(ugettext_lazy("Label for the third category of comments"), max_length=20, null=True, blank=True)
9172f4c6ad70 Better translation of help text
gibus
parents: 515
diff changeset
   202
    category_4 = models.CharField(ugettext_lazy("Label for the fourth category of comments"), max_length=20, null=True, blank=True)
9172f4c6ad70 Better translation of help text
gibus
parents: 515
diff changeset
   203
    category_5 = models.CharField(ugettext_lazy("Label for the fifth category of comments"), max_length=20, null=True, blank=True)
514
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   204
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   205
    text = models.ForeignKey("Text")
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   206
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   207
    objects = TextVersionManager()
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   208
    
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   209
    def get_content(self, format='html'):
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   210
        return pandoc_convert(self.content, self.format, format)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   211
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   212
    def get_comments(self):
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   213
        "Warning: data access without security"
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   214
        return self.comment_set.filter(reply_to=None, deleted=False)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   215
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   216
    def get_replies(self):
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   217
        "Warning: data access without security"
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   218
        return self.comment_set.filter(~Q(reply_to == None), Q(deleted=False))
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   219
    
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   220
    def __unicode__(self):
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   221
        return '<%d> %s' % (self.id, self.title)    
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   222
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   223
    def edit(self, new_title, new_format, new_content, new_tags=None, new_note=None, keep_comments=True, cancel_modified_scopes=True):
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   224
        new_content = on_content_receive(new_content, new_format) 
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   225
        if not keep_comments :
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   226
            self.comment_set.all().delete()
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   227
        elif self.content != new_content or new_format != self.format:
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   228
            comments = self.get_comments() ;
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   229
            tomodify_comments, toremove_comments = compute_new_comment_positions(self.content, self.format, new_content, new_format, comments)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   230
            [comment.save(keep_dates=True) for comment in tomodify_comments]
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   231
            if cancel_modified_scopes :
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   232
                [comment.remove_scope() for comment in toremove_comments]
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   233
            else :
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   234
                [comment.delete() for comment in toremove_comments]
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   235
                
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   236
        self.title = new_title
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   237
        if new_tags:
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   238
            self.tags = new_tags
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   239
        if new_note:
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   240
            self.note = new_note
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   241
        self.content = new_content
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   242
        self.format = new_format
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   243
        self.save()
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   244
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   245
    def get_next_version(self):        
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   246
        other_versions = TextVersion.objects.filter(text__exact=self.text).order_by('created').filter(created__gt=self.created)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   247
        return other_versions[0] if other_versions else None 
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   248
        if other_versions:
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   249
            return 
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   250
        else:
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   251
            return None
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   252
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   253
    def get_previous_version(self):
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   254
        other_versions = TextVersion.objects.filter(text__exact=self.text).order_by('-created').filter(created__lt=self.created)
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   255
        return other_versions[0] if other_versions else None
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   256
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   257
    def get_version_number(self):
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   258
        return TextVersion.objects.filter(text__exact=self.text).order_by('created').filter(created__lte=self.created).count()
f99c7f7401b0 Reorder classes in models.py.
Production Moz <dev@sopinspace.com>
parents: 513
diff changeset
   259
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   260
class CommentManager(Manager):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   261
    
72
b0c777412d0a keep comment dates (created / modified) when duplicating (fixes #12)
raph
parents: 38
diff changeset
   262
    def duplicate(self, comment, text_version, reply_to=None, keep_dates=False):
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   263
        comment.id = None
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   264
        comment.text_version = text_version
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   265
        if reply_to:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   266
            comment.reply_to = reply_to
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   267
        self.update_keys(comment)
72
b0c777412d0a keep comment dates (created / modified) when duplicating (fixes #12)
raph
parents: 38
diff changeset
   268
        comment.save(keep_dates=keep_dates)
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   269
        return comment
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   270
    
130
f257ad33bed5 model change+migration (add id_key to comment): run ./bin/django migrate
raph
parents: 103
diff changeset
   271
from cm.models_base import KEY_MAX_SIZE, generate_key
f257ad33bed5 model change+migration (add id_key to comment): run ./bin/django migrate
raph
parents: 103
diff changeset
   272
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   273
class Comment(PermanentModel, AuthorModel):
72
b0c777412d0a keep comment dates (created / modified) when duplicating (fixes #12)
raph
parents: 38
diff changeset
   274
    modified = models.DateTimeField()
b0c777412d0a keep comment dates (created / modified) when duplicating (fixes #12)
raph
parents: 38
diff changeset
   275
    created = models.DateTimeField()
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   276
130
f257ad33bed5 model change+migration (add id_key to comment): run ./bin/django migrate
raph
parents: 103
diff changeset
   277
    # key to identify same comments across versions
f257ad33bed5 model change+migration (add id_key to comment): run ./bin/django migrate
raph
parents: 103
diff changeset
   278
    id_key = models.CharField(max_length=KEY_MAX_SIZE, db_index=True, default=generate_key)
f257ad33bed5 model change+migration (add id_key to comment): run ./bin/django migrate
raph
parents: 103
diff changeset
   279
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   280
    text_version = models.ForeignKey("TextVersion")
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   281
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   282
    # comment_set will be replies
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   283
    reply_to = models.ForeignKey("Comment", null=True, blank=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   284
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   285
    title = models.TextField()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   286
    content = models.TextField()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   287
    content_html = models.TextField()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   288
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   289
    format = models.CharField(_("Format:"), max_length=20, blank=False, default=DEFAULT_INPUT_FORMAT, choices=CHOICES_INPUT_FORMATS)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   290
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   291
    tags = TagField()
504
b2e0186daa5b Adds a category to comments, painted with colored vertical bar.
gibus
parents: 478
diff changeset
   292
b2e0186daa5b Adds a category to comments, painted with colored vertical bar.
gibus
parents: 478
diff changeset
   293
    category = models.PositiveSmallIntegerField(default=0)
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   294
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   295
    start_wrapper = models.IntegerField(null=True, blank=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   296
    end_wrapper = models.IntegerField(null=True, blank=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   297
    start_offset = models.IntegerField(null=True, blank=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   298
    end_offset = models.IntegerField(null=True, blank=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   299
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   300
    objects = CommentManager()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   301
    
327
d858c3bb07fe save prototype for django new version (1.2.3)
gibus
parents: 305
diff changeset
   302
    def save(self, keep_dates=False, **kwargs):
72
b0c777412d0a keep comment dates (created / modified) when duplicating (fixes #12)
raph
parents: 38
diff changeset
   303
        if not keep_dates:
b0c777412d0a keep comment dates (created / modified) when duplicating (fixes #12)
raph
parents: 38
diff changeset
   304
            now = datetime.now()
b0c777412d0a keep comment dates (created / modified) when duplicating (fixes #12)
raph
parents: 38
diff changeset
   305
            if not self.id: 
b0c777412d0a keep comment dates (created / modified) when duplicating (fixes #12)
raph
parents: 38
diff changeset
   306
                self.created = now  
b0c777412d0a keep comment dates (created / modified) when duplicating (fixes #12)
raph
parents: 38
diff changeset
   307
            self.modified = now 
327
d858c3bb07fe save prototype for django new version (1.2.3)
gibus
parents: 305
diff changeset
   308
        super(PermanentModel, self).save(**kwargs) 
72
b0c777412d0a keep comment dates (created / modified) when duplicating (fixes #12)
raph
parents: 38
diff changeset
   309
            
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   310
    def __unicode__(self):
262
9644e27ad08d fix comment __str__ display
raph
parents: 259
diff changeset
   311
        return '<%d> %s [st_wrp:%s, st_ofs:%s, e_wrp:%s, e_ofs:%s]' % (self.id, self.title,  self.start_wrapper ,  self.start_offset,  self.end_wrapper,  self.end_offset, )    
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   312
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   313
    def is_reply(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   314
        return self.reply_to != None
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   315
    
16
036705244cab FIX: comment visibility takes ownership into account
raph
parents: 12
diff changeset
   316
    def is_thread_full_visible(self, own_user=None):
036705244cab FIX: comment visibility takes ownership into account
raph
parents: 12
diff changeset
   317
        """
036705244cab FIX: comment visibility takes ownership into account
raph
parents: 12
diff changeset
   318
        own_user: comment belonging to this user are also visible 
036705244cab FIX: comment visibility takes ownership into account
raph
parents: 12
diff changeset
   319
        """
036705244cab FIX: comment visibility takes ownership into account
raph
parents: 12
diff changeset
   320
        if self.state == 'approved' or (own_user and self.user == own_user):
036705244cab FIX: comment visibility takes ownership into account
raph
parents: 12
diff changeset
   321
            if self.reply_to==None:
036705244cab FIX: comment visibility takes ownership into account
raph
parents: 12
diff changeset
   322
                return True
036705244cab FIX: comment visibility takes ownership into account
raph
parents: 12
diff changeset
   323
            else:                
036705244cab FIX: comment visibility takes ownership into account
raph
parents: 12
diff changeset
   324
                return self.reply_to.is_thread_full_visible(own_user)
036705244cab FIX: comment visibility takes ownership into account
raph
parents: 12
diff changeset
   325
        return False
036705244cab FIX: comment visibility takes ownership into account
raph
parents: 12
diff changeset
   326
               
103
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   327
    def is_scope_removed(self):
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   328
        #when scope is "removed" we will have 
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   329
        #self.start_wrapper == self.end_wrapper == self.start_offset == self.end_offset == -1
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   330
        return (self.start_wrapper == -1)  
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   331
               
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   332
    def top_comment(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   333
        if self.reply_to == None :
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   334
            return self
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   335
        else : 
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   336
            return self.reply_to.top_comment()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   337
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   338
    def depth(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   339
        if self.reply_to == None :
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   340
            return 0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   341
        else : 
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   342
            return 1 + self.reply_to.depth()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   343
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   344
    def delete(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   345
        PermanentModel.delete(self)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   346
        # delete replies
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   347
        [c.delete() for c in self.comment_set.all()]
103
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   348
        
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   349
    def remove_scope(self):
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   350
        self.start_wrapper = self.end_wrapper = self.start_offset = self.end_offset = -1
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   351
        self.save()
61fd17f9ab78 enh: detached comments
rbernard
parents: 72
diff changeset
   352
        
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   353
# http://docs.djangoproject.com/en/dev/topics/files/#topics-files
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   354
511
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   355
# default conf values
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   356
DEFAULT_CONF = {
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   357
                'workspace_name' : 'Workspace',
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   358
                'site_url' : settings.SITE_URL,
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   359
                'email_from' : settings.DEFAULT_FROM_EMAIL,
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   360
                }
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   361
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   362
from cm.role_models import change_role_model
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   363
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   364
class ConfigurationManager(models.Manager):
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   365
    def set_workspace_name(self, workspace_name):
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   366
        if workspace_name:
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   367
            self.set_key('workspace_name', workspace_name)
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   368
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   369
    def get_key(self, key, default_value=None):
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   370
        try:
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   371
            return self.get(key=key).value
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   372
        except Configuration.DoesNotExist:
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   373
            return DEFAULT_CONF.get(key, default_value)
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   374
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   375
    def del_key(self, key):
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   376
        try:
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   377
            self.get(key=key).delete()
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   378
        except Configuration.DoesNotExist:
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   379
            return None
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   380
        
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   381
    def set_key(self, key, value):
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   382
        conf, created = self.get_or_create(key=key)
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   383
        if created or conf.value != value:
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   384
            conf.value = value
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   385
            conf.save()
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   386
            if key == 'workspace_role_model':
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   387
                change_role_model(value)
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   388
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   389
    def __getitem__(self, key):
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   390
        if not key.startswith('f_'):
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   391
            return self.get_key(key, None)
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   392
        else:
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   393
            return getattr(self,key)()
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   394
    
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   395
    def f_get_logo_url(self):
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   396
        key = self.get_key('workspace_logo_file_key', None)
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   397
        if key:
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   398
            attach = Attachment.objects.get(key=key)
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   399
            return attach.data.url
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   400
        else:
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   401
            return None 
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   402
    
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   403
import base64
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   404
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   405
class Configuration(models.Model):
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   406
    key = models.TextField(blank=False) # , unique=True cannot be added: creates error on mysql (?)
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   407
    raw_value = models.TextField(blank=False)
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   408
    
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   409
    def get_value(self):
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   410
        return pickle.loads(base64.b64decode(self.raw_value.encode('utf8')))
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   411
        
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   412
    def set_value(self, value):        
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   413
        self.raw_value = base64.b64encode(pickle.dumps(value, 0)).encode('utf8')
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   414
                
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   415
    value = property(get_value, set_value)
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   416
                
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   417
    objects = ConfigurationManager()
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   418
    
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   419
    def __unicode__(self):
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   420
        return '%s: %s' % (self.key, self.value)    
54d72b4da009 Reorder ConfigurationManager model, because it broke co-ment lite
Production Moz <dev@sopinspace.com>
parents: 504
diff changeset
   421
    
515
0be906e586e6 Fix south migration problem caused by 0003_update_keys_to_textversion using TextVersion model which is updated only in 0010_auto__add_field_textversion_category_1__add_field_textversion_category, same with 0007_populate_id_key_in_comments using Comment model updated in 0011_auto__add_field_comment_category.
Production Moz <dev@sopinspace.com>
parents: 514
diff changeset
   422
ApplicationConfiguration = Configuration.objects     
0be906e586e6 Fix south migration problem caused by 0003_update_keys_to_textversion using TextVersion model which is updated only in 0010_auto__add_field_textversion_category_1__add_field_textversion_category, same with 0007_populate_id_key_in_comments using Comment model updated in 0011_auto__add_field_comment_category.
Production Moz <dev@sopinspace.com>
parents: 514
diff changeset
   423
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   424
class AttachmentManager(KeyManager):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   425
    def create_attachment(self, text_version, filename, data):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   426
        attach = self.create(text_version=text_version)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   427
        ff = ContentFile(data)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   428
        attach.data.save(filename, ff)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   429
        return attach
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   430
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   431
class Attachment(KeyModel):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   432
    data = models.FileField(upload_to="attachments/%Y/%m/%d/", max_length=1000)
504
b2e0186daa5b Adds a category to comments, painted with colored vertical bar.
gibus
parents: 478
diff changeset
   433
    text_version = models.ForeignKey(TextVersion, null=True, blank=True, default=None)
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   434
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   435
    objects = AttachmentManager()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   436
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   437
class NotificationManager(KeyManager):
12
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   438
    def create_notification(self, text, type, active, email_or_user):
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   439
        notification = self.create(text=text, type=type, active=active)
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   440
        notification.set_email_or_user(email_or_user)
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   441
        return notification
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   442
12
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   443
    def get_notifications(self, text, type, email_or_user):
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   444
        if isinstance(email_or_user,unicode):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   445
            prev_notifications = Notification.objects.filter(text=text, type=type, email=email_or_user)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   446
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   447
            prev_notifications = Notification.objects.filter(text=text, type=type, user=email_or_user)
12
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   448
            
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   449
        if prev_notifications:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   450
            return prev_notifications[0]
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   451
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   452
            return None
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   453
     
12
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   454
    def set_notification(self, text, type, active, email_or_user):
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   455
        notification = self.get_notifications(text, type, email_or_user)
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   456
        if notification == None :
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   457
            self.create_notification(text, type, active, email_or_user)
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   458
        else : 
f69ff46d3240 various notification changes
reno
parents: 5
diff changeset
   459
            notification.active = active
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   460
            notification.save()                
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   461
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   462
class Notification(KeyModel, AuthorModel):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   463
    text = models.ForeignKey(Text, null=True, blank=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   464
    type = models.CharField(max_length=30, null=True, blank=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   465
    active = models.BooleanField(default=True) # active = False means user desactivation
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   466
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   467
    objects = NotificationManager()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   468
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   469
    def desactivate_notification_url(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   470
        return reverse('desactivate-notification', args=[self.adminkey])
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   471
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   472
    def desactivate(self):    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   473
        if self.type=='own':
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   474
            self.active = False
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   475
            self.save()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   476
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   477
            self.delete()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   478
    
266
33eb2ee6b9de add notifications to admin area
raph
parents: 262
diff changeset
   479
    def __unicode__(self):
33eb2ee6b9de add notifications to admin area
raph
parents: 262
diff changeset
   480
        return u"%s: %s %s %s %s" % (self.__class__.__name__, self.user, self.text, self.type, self.active )
33eb2ee6b9de add notifications to admin area
raph
parents: 262
diff changeset
   481
    
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   482
# right management
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   483
class UserRoleManager(models.Manager):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   484
    def create_userroles_text(self, text):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   485
        # make sure every user has a userrole on this text
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   486
        for user in User.objects.all():
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   487
            userrole, _ = self.get_or_create(user=user, text=text)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   488
        # anon user
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   489
        userrole, _ = self.get_or_create(user=None, text=text)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   490
        # anon global user
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   491
        global_userrole, _ = self.get_or_create(user=None, text=None)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   492
            
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   493
class UserRole(models.Model):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   494
    role = models.ForeignKey("Role", null=True, blank=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   495
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   496
    # user == null => anyone
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   497
    user = models.ForeignKey(User, null=True, blank=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   498
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   499
    # text == null => any text (workspace role)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   500
    text = models.ForeignKey(Text, null=True, blank=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   501
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   502
    objects = UserRoleManager()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   503
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   504
    class Meta:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   505
        unique_together = (('role', 'user', 'text',))
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   506
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   507
    def __unicode__(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   508
        if self.role:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   509
            rolename = _(self.role.name)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   510
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   511
            rolename = ''
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   512
            
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   513
        if self.user:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   514
            return u"%s: %s %s %s" % (self.__class__.__name__, self.user.username, self.text, rolename)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   515
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   516
            return u"%s: *ALL* %s %s" % (self.__class__.__name__, self.text, rolename)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   517
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   518
    def __repr__(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   519
        return self.__unicode__()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   520
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   521
from cm.models_base import generate_key
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   522
from cm.utils.misc import update
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   523
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   524
class Role(models.Model):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   525
    """
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   526
    'Static' application roles 
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   527
    """
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   528
    name = models.CharField(ugettext_lazy('name'), max_length=50, unique=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   529
    description = models.TextField(ugettext_lazy('description'))
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   530
    permissions = models.ManyToManyField(Permission, related_name="roles")
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   531
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   532
    global_scope = models.BooleanField('global scope', default=False) # applies to global scope only
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   533
    anon = models.BooleanField('anonymous', default=False) # role possible for anonymous users?
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   534
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   535
    def __unicode__(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   536
        return _(self.name)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   537
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   538
    def __hash__(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   539
        return self.id
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   540
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   541
    def name_i18n(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   542
        return _(self.name)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   543
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   544
from django.utils.safestring import mark_safe
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   545
 
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   546
class RegistrationManager(KeyManager):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   547
    def activate_user(self, activation_key):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   548
        """
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   549
        Validates an activation key and activates the corresponding
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   550
        ``User`` if valid.
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   551
        If the key is valid , returns the ``User`` as second arg
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   552
        First is boolean indicating if user has just been activated
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   553
        """
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   554
        # Make sure the key we're trying conforms to the pattern of a
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   555
        # SHA1 hash; if it doesn't, no point trying to look it up in
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   556
        # the database.
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   557
        try:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   558
            profile = self.get(admin_key=activation_key)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   559
        except self.model.DoesNotExist:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   560
            return False, False
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   561
        user = profile.user
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   562
        activated = False
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   563
        if not user.is_active:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   564
            user.is_active = True
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   565
            user.save()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   566
            activated = True
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   567
        return (activated, user)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   568
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   569
    def _create_manager(self, email, username, password, first_name, last_name):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   570
        if username and email and password and len(User.objects.filter(username=username)) == 0:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   571
            user = User.objects.create(username=username, email=email, first_name=first_name, last_name=last_name, is_active=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   572
            user.set_password(password)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   573
            user.save()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   574
            
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   575
            profile = UserProfile.objects.create(user=user)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   576
                    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   577
            manager = Role.objects.get(name='Manager')
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   578
            UserRole.objects.create(text=None, user=user, role=manager)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   579
            return user
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   580
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   581
            return None
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   582
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   583
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   584
    def create_inactive_user(self, email, send_invitation, **kwargs):
448
054d572a3db4 Avoid warnings for deprecated instructions in djan 1.3
gibus
parents: 407
diff changeset
   585
        if 'postgresql' in settings.DATABASES['default']['ENGINE']:
305
df2c3202fd8e do not lock table with backends other than postgresql (postgresql specific syntax) (fixes #52)
raph
parents: 289
diff changeset
   586
            #prevent concurrent access 
df2c3202fd8e do not lock table with backends other than postgresql (postgresql specific syntax) (fixes #52)
raph
parents: 289
diff changeset
   587
            cursor = connection.cursor()
df2c3202fd8e do not lock table with backends other than postgresql (postgresql specific syntax) (fixes #52)
raph
parents: 289
diff changeset
   588
            sql = "LOCK TABLE auth_user IN EXCLUSIVE MODE"
df2c3202fd8e do not lock table with backends other than postgresql (postgresql specific syntax) (fixes #52)
raph
parents: 289
diff changeset
   589
            cursor.execute(sql)
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   590
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   591
        try:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   592
            user_with_email = User.objects.get(email__iexact=email)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   593
        except User.DoesNotExist:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   594
            user = User.objects.create(username=email, email=email)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   595
            profile = UserProfile.objects.create(user=user)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   596
            update(user, kwargs)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   597
            update(profile, kwargs)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   598
            
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   599
            user.is_active = False
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   600
            user.save()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   601
            profile.save()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   602
            
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   603
            note = kwargs.get('note', None) 
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   604
            if send_invitation:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   605
                profile.send_activation_email(note)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   606
            return user
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   607
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   608
            return user_with_email
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   609
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   610
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   611
from cm.utils.mail import send_mail
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   612
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   613
class UserProfile(KeyModel):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   614
    modified = models.DateTimeField(auto_now=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   615
    created = models.DateTimeField(auto_now_add=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   616
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   617
    user = models.ForeignKey(User, unique=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   618
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   619
    allow_contact = models.BooleanField(ugettext_lazy(u'Allow contact'), default=True, help_text=ugettext_lazy(u"Allow email messages from other users"))    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   620
    preferred_language = models.CharField(ugettext_lazy(u'Preferred language'), max_length=2, default="en")
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   621
    is_temp = models.BooleanField(default=False)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   622
    is_email_error = models.BooleanField(default=False)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   623
    is_suspended = models.BooleanField(ugettext_lazy(u'Suspended access'), default=False) # used to disable access or to wait for approval when registering
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   624
38
be1807d390d4 - add tag support to user
raph
parents: 24
diff changeset
   625
    tags = TagField(ugettext_lazy("Tags"), max_length=1000)
be1807d390d4 - add tag support to user
raph
parents: 24
diff changeset
   626
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   627
    objects = RegistrationManager()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   628
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   629
    class Meta:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   630
        permissions = (
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   631
            ("can_create_user", "Can create user"),
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   632
            ("can_delete_user", "Can delete user"),
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   633
        )
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   634
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   635
    def __unicode__(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   636
        return unicode(self.user)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   637
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   638
    def global_userrole(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   639
        try:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   640
            return UserRole.objects.get(user=self.user, text=None)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   641
        except UserRole.DoesNotExist:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   642
            return None
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   643
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   644
    def admin_print(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   645
        if self.is_suspended:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   646
            if self.user.is_active:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   647
                return mark_safe('%s (%s)' % (self.user.username, _(u'suspended'),))
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   648
            else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   649
                return mark_safe('%s (%s)' % (self.user.username, _(u'waiting approval'),))
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   650
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   651
            if self.user.is_active:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   652
                return mark_safe('%s' % self.user.username) 
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   653
            else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   654
                email_username = self.user.email.split('@')[0]
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   655
                return mark_safe('%s (%s)' % (self.user.username, _(u'pending'),))
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   656
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   657
    def simple_print(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   658
        if self.user.is_active:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   659
            return self.user.username 
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   660
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   661
            return self.user.email
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   662
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   663
    def send_activation_email(self, note=None):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   664
        self._send_act_invit_email(note=note, template='email/activation_email.txt')
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   665
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   666
    def send_invitation_email(self, note=None):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   667
        self._send_act_invit_email(note=note, template='email/invitation_email.txt')
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   668
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   669
    def _send_act_invit_email(self, template, note=None):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   670
        subject = _(u'Invitation')
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   671
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   672
        activate_url = reverse('user-activate', args=[self.adminkey])
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   673
        message = render_to_string(template,
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   674
                                   { 
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   675
                                     'activate_url' : activate_url,
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   676
                                     'note' : note,
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   677
                                     'CONF': ApplicationConfiguration
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   678
                                      })
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   679
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   680
        send_mail(subject, message, ApplicationConfiguration['email_from'], [self.user.email])
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   681
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   682
from django.db.models import signals
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   683
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   684
def delete_profile(sender, **kwargs):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   685
    user_profile = kwargs['instance']
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   686
    user = user_profile.user
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   687
    user.delete()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   688
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   689
signals.post_delete.connect(delete_profile, sender=UserProfile)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   690
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   691
class ActivityManager(models.Manager):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   692
    pass
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   693
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   694
class Activity(models.Model):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   695
    created = models.DateTimeField(auto_now_add=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   696
    originator_user = models.ForeignKey(User, related_name='originator_activity', null=True, blank=True, default=None)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   697
    text = models.ForeignKey(Text, null=True, blank=True, default=None)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   698
    text_version = models.ForeignKey(TextVersion, null=True, blank=True, default=None)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   699
    comment = models.ForeignKey(Comment, null=True, blank=True, default=None)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   700
    user = models.ForeignKey(User, null=True, blank=True, default=None)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   701
    type = models.CharField(max_length=30)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   702
    ip = models.IPAddressField(null=True, blank=True, default=None)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   703
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   704
    objects = ActivityManager()
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   705
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   706
    # viewable activities (i.e. now 'text-view')
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   707
    VIEWABLE_ACTIVITIES = {
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   708
                   'view_comments' : ['comment_created', 'comment_removed'],
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   709
                   'view_users' : ['user_created', 'user_activated', 'user_refused', 'user_enabled', 'user_approved', 'user_suspended'],
460
2fdb7d095d5c Added import from XML file, including text, comments and attachments.
gibus
parents: 448
diff changeset
   710
                   'view_texts' : ['text_created', 'text_imported', 'text_removed', 'text_edited', 'text_edited_new_version'],
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   711
                   }
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   712
    ACTIVITIES_TYPES = reduce(list.__add__, VIEWABLE_ACTIVITIES.values())
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   713
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   714
    IMGS = {
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   715
            'text_created' : u'page_add_small.png',
460
2fdb7d095d5c Added import from XML file, including text, comments and attachments.
gibus
parents: 448
diff changeset
   716
            'text_imported' : u'sop_import_small.png',
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   717
            'text_removed' : u'page_delete_small.png',
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   718
            'text_edited'  : u'page_save_small.png',
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   719
            'text_edited_new_version' : u'page_save_small.png',
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   720
            'user_created' : u'user_add_small.png',
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   721
            'user_enabled' : u'user_add_small.png',
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   722
            'user_refused': u'user_delete_small.png',
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   723
            'user_suspended': u'user_delete_small.png',
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   724
            'user_approved': u'user_add_small.png',
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   725
            'user_activated' : u'user_go.png',
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   726
            'comment_created' : u'note_add_small.png',
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   727
            'comment_removed' : u'note_delete_small.png',
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   728
        }
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   729
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   730
    #type/msg
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   731
    MSGS = {
227
57492cd7e162 add author in activities traces
raph
parents: 175
diff changeset
   732
         'text_edited' : ugettext_lazy(u'Text %(link_to_text)s edited by %(creator)s'),
57492cd7e162 add author in activities traces
raph
parents: 175
diff changeset
   733
         'text_edited_new_version' : ugettext_lazy(u'Text %(link_to_text)s edited (new version created) by %(creator)s'),
57492cd7e162 add author in activities traces
raph
parents: 175
diff changeset
   734
         'text_created' :  ugettext_lazy(u'Text %(link_to_text)s added by %(creator)s'),
460
2fdb7d095d5c Added import from XML file, including text, comments and attachments.
gibus
parents: 448
diff changeset
   735
         'text_imported' :  ugettext_lazy(u'Text %(link_to_text)s imported by %(creator)s'),
5
c3594e4df7c1 lazy strings
raph
parents: 0
diff changeset
   736
         'text_removed' : ugettext_lazy(u'Text %(link_to_text)s removed'),
227
57492cd7e162 add author in activities traces
raph
parents: 175
diff changeset
   737
         'comment_created' : ugettext_lazy(u'Comment %(link_to_comment)s added on text %(link_to_text)s by %(creator)s'),
5
c3594e4df7c1 lazy strings
raph
parents: 0
diff changeset
   738
         'comment_removed' : ugettext_lazy(u'Comment %(link_to_comment)s removed from text %(link_to_text)s'),
c3594e4df7c1 lazy strings
raph
parents: 0
diff changeset
   739
         'user_created' : ugettext_lazy(u'User %(username)s added'),
c3594e4df7c1 lazy strings
raph
parents: 0
diff changeset
   740
         'user_enabled' : ugettext_lazy(u'User %(username)s access to workspace enabled'),
c3594e4df7c1 lazy strings
raph
parents: 0
diff changeset
   741
         'user_refused' : ugettext_lazy(u'User %(username)s access to workspace refused'),
c3594e4df7c1 lazy strings
raph
parents: 0
diff changeset
   742
         'user_suspended' : ugettext_lazy(u'User %(username)s access to workspace suspended'),
c3594e4df7c1 lazy strings
raph
parents: 0
diff changeset
   743
         'user_activated' : ugettext_lazy(u'User %(username)s access to workspace activated'),
c3594e4df7c1 lazy strings
raph
parents: 0
diff changeset
   744
         'user_approved' : ugettext_lazy(u'User %(username)s has activated his account'),
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   745
         }
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   746
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   747
    def is_same_user(self, other_activity):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   748
        if (self.originator_user != None or other_activity.originator_user != None) and self.originator_user != other_activity.originator_user:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   749
            return False
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   750
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   751
            return self.ip != None and self.ip == other_activity.ip
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   752
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   753
    def linkable_text_title(self, html=True, link=True):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   754
        # html: whether or not output sould be html
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   755
        format_args = {'link':absolute_reverse('text-view', args=[self.text.key]), 'title':self.text.title}
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   756
        if html and not self.text.deleted :
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   757
            return mark_safe(u'<a href="%(link)s">%(title)s</a>' % format_args)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   758
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   759
            if link and not self.text.deleted:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   760
                return u'%(title)s (%(link)s)' % format_args
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   761
            else:             
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   762
                return self.text.title ;
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   763
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   764
    def linkable_comment_title(self, html=True, link=True):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   765
        if self.comment:
144
e7e1bc64f613 lpermalink uses id_key
rbernard
parents: 140
diff changeset
   766
            format_args = {'link':absolute_reverse('text-view-show-comment', args=[self.text.key, self.comment.id_key]), 'title':self.comment.title}
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   767
            if html and not self.comment.deleted and not self.text.deleted:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   768
                return mark_safe(u'<a href="%(link)s">%(title)s</a>' % format_args)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   769
            else :
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   770
                if link and not self.comment.deleted and not self.text.deleted:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   771
                    return u'%(title)s (%(link)s)' % format_args
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   772
                else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   773
                    return self.comment.title ;
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   774
        else:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   775
            return u''
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   776
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   777
    def __unicode__(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   778
        return u"%s %s %s %s %s" % (self.type, self.originator_user, self.text, self.comment, self.user)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   779
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   780
    def img_name(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   781
        return self.IMGS.get(self.type)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   782
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   783
    def printable_data_nohtml_link(self):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   784
        return self.printable_data(html=False, link=True)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   785
        
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   786
    def printable_data(self, html=True, link=True):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   787
        msg = self.MSGS.get(self.type, None)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   788
        if msg:
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   789
            return mark_safe(msg % {
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   790
                                     'link_to_text' : self.linkable_text_title(html=html, link=link) if self.text else None,
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   791
                                     'link_to_comment' : self.linkable_comment_title(html=html, link=link) if self.comment else None,
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   792
                                     'username' : self.user.username if self.user else None,
227
57492cd7e162 add author in activities traces
raph
parents: 175
diff changeset
   793
                                     'creator' : self.originator_user.username if self.originator_user else ugettext_lazy(u'anonymous'),
0
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   794
                                    })
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   795
        return ''
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   796
    
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   797
    def printable_metadata(self, html=True):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   798
        ret = []
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   799
        if self.type == 'user_activated':
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   800
            ret.append(_(u'by "%(username)s"') % {'username' : self.originator_user.username})
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   801
            ret.append(' ')
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   802
        ret.append(_(u"%(time_since)s ago") % {'time_since':timesince(self.created)})
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   803
        return ''.join(ret)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   804
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   805
    def printable_metadata_absolute(self, html=True):
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   806
        ret = []
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   807
        if self.type == 'user_activated':
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   808
            ret.append(_(u'by "%(username)s"') % {'username' : self.originator_user.username})
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   809
            ret.append(u' ')
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   810
        ret.append(datetime_to_user_str(self.created))
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   811
        return u''.join(ret)
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   812
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   813
import cm.denorm_engine
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   814
import cm.admin
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   815
import cm.main
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   816
import cm.activity
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   817
import cm.notifications
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   818
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   819
# we fill username with email so we need a bigger value 
40c8f766c9b8 import from internal svn r 4007
raph
parents:
diff changeset
   820
User._meta.get_field('username').max_length = 75
347
0a359382701b prevent uno custom __import__ from messing with south import machinery (to discover south enabled dj apps)
Production Moz <dev@sopinspace.com>
parents: 327
diff changeset
   821
0a359382701b prevent uno custom __import__ from messing with south import machinery (to discover south enabled dj apps)
Production Moz <dev@sopinspace.com>
parents: 327
diff changeset
   822
import monkey_patches