thread.py
324 lines
| 10.3 KiB
| text/x-python
|
PythonLexer
neko259
|
r691 | import logging | ||
neko259
|
r1616 | from datetime import timedelta | ||
neko259
|
r1760 | from django.db import models, transaction | ||
neko259
|
r1345 | from django.db.models import Count, Sum, QuerySet, Q | ||
neko259
|
r691 | from django.utils import timezone | ||
neko259
|
r957 | |||
neko259
|
r1760 | import boards | ||
from boards import settings | ||||
neko259
|
r1416 | from boards.models import STATUS_BUMPLIMIT, STATUS_ACTIVE, STATUS_ARCHIVE | ||
neko259
|
r1760 | from boards.models.attachment import FILE_TYPES_IMAGE | ||
neko259
|
r958 | from boards.models.post import Post | ||
neko259
|
r2004 | from boards.models.tag import Tag, TagAlias | ||
from boards.settings import SECTION_VIEW | ||||
neko259
|
r1760 | from boards.utils import cached_result, datetime_to_epoch | ||
neko259
|
r957 | |||
neko259
|
r1346 | FAV_THREAD_NO_UPDATES = -1 | ||
neko259
|
r691 | |||
__author__ = 'neko259' | ||||
logger = logging.getLogger(__name__) | ||||
neko259
|
r1088 | WS_NOTIFICATION_TYPE_NEW_POST = 'new_post' | ||
WS_NOTIFICATION_TYPE = 'notification_type' | ||||
WS_CHANNEL_THREAD = "thread:" | ||||
neko259
|
r1428 | STATUS_CHOICES = ( | ||
(STATUS_ACTIVE, STATUS_ACTIVE), | ||||
(STATUS_BUMPLIMIT, STATUS_BUMPLIMIT), | ||||
(STATUS_ARCHIVE, STATUS_ARCHIVE), | ||||
) | ||||
neko259
|
r1088 | |||
neko259
|
r715 | class ThreadManager(models.Manager): | ||
neko259
|
r1616 | def process_old_threads(self): | ||
neko259
|
r715 | """ | ||
Preserves maximum thread count. If there are too many threads, | ||||
neko259
|
r716 | archive or delete the old ones. | ||
neko259
|
r715 | """ | ||
neko259
|
r1616 | old_time_delta = settings.get_int('Messages', 'ThreadArchiveDays') | ||
old_time = timezone.now() - timedelta(days=old_time_delta) | ||||
old_ops = Post.objects.filter(opening=True, pub_time__lte=old_time).exclude(thread__status=STATUS_ARCHIVE) | ||||
neko259
|
r715 | |||
neko259
|
r1616 | for op in old_ops: | ||
thread = op.get_thread() | ||||
if settings.get_bool('Storage', 'ArchiveThreads'): | ||||
self._archive_thread(thread) | ||||
else: | ||||
thread.delete() | ||||
logger.info('Processed old thread {}'.format(thread)) | ||||
neko259
|
r715 | |||
neko259
|
r716 | def _archive_thread(self, thread): | ||
neko259
|
r1414 | thread.status = STATUS_ARCHIVE | ||
neko259
|
r716 | thread.last_edit_time = timezone.now() | ||
neko259
|
r1029 | thread.update_posts_time() | ||
neko259
|
r1414 | thread.save(update_fields=['last_edit_time', 'status']) | ||
neko259
|
r716 | |||
neko259
|
r2044 | def get_new_posts(self, last_posts): | ||
neko259
|
r1345 | query = None | ||
neko259
|
r2044 | for post in last_posts: | ||
if not post.get_thread().is_archived(): | ||||
q = Q(id=post.thread_id) & Q(replies__id__gt=post.id) | ||||
neko259
|
r1346 | if query is None: | ||
query = q | ||||
else: | ||||
query = query | q | ||||
if query is not None: | ||||
return self.filter(query).annotate( | ||||
neko259
|
r1705 | new_post_count=Count('replies')) | ||
neko259
|
r1345 | |||
neko259
|
r2044 | def get_new_post_count(self, last_posts): | ||
new_posts = self.get_new_posts(last_posts) | ||||
neko259
|
r1705 | return new_posts.aggregate(total_count=Count('replies'))\ | ||
neko259
|
r1347 | ['total_count'] if new_posts else 0 | ||
neko259
|
r1345 | |||
neko259
|
r715 | |||
neko259
|
r1136 | def get_thread_max_posts(): | ||
neko259
|
r1153 | return settings.get_int('Messages', 'MaxPostsPerThread') | ||
neko259
|
r1136 | |||
neko259
|
r691 | class Thread(models.Model): | ||
neko259
|
r715 | objects = ThreadManager() | ||
neko259
|
r691 | |||
class Meta: | ||||
app_label = 'boards' | ||||
neko259
|
r1269 | tags = models.ManyToManyField('Tag', related_name='thread_tags') | ||
neko259
|
r966 | bump_time = models.DateTimeField(db_index=True) | ||
neko259
|
r691 | last_edit_time = models.DateTimeField() | ||
neko259
|
r1136 | max_posts = models.IntegerField(default=get_thread_max_posts) | ||
neko259
|
r1428 | status = models.CharField(max_length=50, default=STATUS_ACTIVE, | ||
neko259
|
r1667 | choices=STATUS_CHOICES, db_index=True) | ||
neko259
|
r1434 | monochrome = models.BooleanField(default=False) | ||
neko259
|
r1951 | stickerpack = models.BooleanField(default=False) | ||
neko259
|
r691 | |||
neko259
|
r1186 | def get_tags(self) -> QuerySet: | ||
neko259
|
r691 | """ | ||
Gets a sorted tag list. | ||||
""" | ||||
neko259
|
r1891 | return self.tags.filter(aliases__in=TagAlias.objects.filter_localized(parent__thread_tags=self)).order_by('aliases__name') | ||
neko259
|
r691 | |||
def bump(self): | ||||
""" | ||||
Bumps (moves to up) thread if possible. | ||||
""" | ||||
if self.can_bump(): | ||||
neko259
|
r1029 | self.bump_time = self.last_edit_time | ||
neko259
|
r691 | |||
neko259
|
r1046 | self.update_bump_status() | ||
neko259
|
r885 | |||
neko259
|
r691 | logger.info('Bumped thread %d' % self.id) | ||
neko259
|
r1083 | def has_post_limit(self) -> bool: | ||
neko259
|
r1055 | return self.max_posts > 0 | ||
neko259
|
r1134 | def update_bump_status(self, exclude_posts=None): | ||
neko259
|
r1055 | if self.has_post_limit() and self.get_reply_count() >= self.max_posts: | ||
neko259
|
r1414 | self.status = STATUS_BUMPLIMIT | ||
neko259
|
r1134 | self.update_posts_time(exclude_posts=exclude_posts) | ||
neko259
|
r1046 | |||
neko259
|
r1106 | def _get_cache_key(self): | ||
return [datetime_to_epoch(self.last_edit_time)] | ||||
@cached_result(key_method=_get_cache_key) | ||||
neko259
|
r1083 | def get_reply_count(self) -> int: | ||
neko259
|
r958 | return self.get_replies().count() | ||
neko259
|
r691 | |||
neko259
|
r1106 | @cached_result(key_method=_get_cache_key) | ||
neko259
|
r1083 | def get_images_count(self) -> int: | ||
neko259
|
r1590 | return self.get_replies().filter( | ||
attachments__mimetype__in=FILE_TYPES_IMAGE)\ | ||||
.annotate(images_count=Count( | ||||
neko259
|
r1591 | 'attachments')).aggregate(Sum('images_count'))['images_count__sum'] or 0 | ||
neko259
|
r691 | |||
neko259
|
r1960 | @cached_result(key_method=_get_cache_key) | ||
def get_attachment_count(self) -> int: | ||||
return self.get_replies().annotate(attachment_count=Count('attachments'))\ | ||||
.aggregate(Sum('attachment_count'))['attachment_count__sum'] or 0 | ||||
neko259
|
r1083 | def can_bump(self) -> bool: | ||
neko259
|
r691 | """ | ||
Checks if the thread can be bumped by replying to it. | ||||
""" | ||||
neko259
|
r1414 | return self.get_status() == STATUS_ACTIVE | ||
neko259
|
r691 | |||
neko259
|
r1186 | def get_last_replies(self) -> QuerySet: | ||
neko259
|
r691 | """ | ||
Gets several last replies, not including opening post | ||||
""" | ||||
neko259
|
r2004 | last_replies_count = settings.get_int(SECTION_VIEW, 'LastRepliesCount') | ||
neko259
|
r1153 | |||
if last_replies_count > 0: | ||||
neko259
|
r691 | reply_count = self.get_reply_count() | ||
if reply_count > 0: | ||||
neko259
|
r1153 | reply_count_to_show = min(last_replies_count, | ||
neko259
|
r691 | reply_count - 1) | ||
neko259
|
r694 | replies = self.get_replies() | ||
neko259
|
r987 | last_replies = replies[reply_count - reply_count_to_show:] | ||
neko259
|
r691 | |||
return last_replies | ||||
neko259
|
r1083 | def get_skipped_replies_count(self) -> int: | ||
neko259
|
r691 | """ | ||
Gets number of posts between opening post and last replies. | ||||
""" | ||||
reply_count = self.get_reply_count() | ||||
neko259
|
r2004 | last_replies_count = min(settings.get_int(SECTION_VIEW, 'LastRepliesCount'), | ||
neko259
|
r691 | reply_count - 1) | ||
return reply_count - last_replies_count - 1 | ||||
neko259
|
r1614 | # TODO Remove argument, it is not used | ||
def get_replies(self, view_fields_only=True) -> QuerySet: | ||||
neko259
|
r691 | """ | ||
Gets sorted thread posts | ||||
""" | ||||
neko259
|
r1704 | query = self.replies.order_by('pub_time').prefetch_related( | ||
'attachments') | ||||
neko259
|
r1530 | return query | ||
neko259
|
r691 | |||
neko259
|
r1614 | def get_viewable_replies(self) -> QuerySet: | ||
""" | ||||
Gets replies with only fields that are used for viewing. | ||||
""" | ||||
neko259
|
r1928 | return self.get_replies().defer('text', 'last_edit_time') | ||
neko259
|
r1614 | |||
neko259
|
r1186 | def get_top_level_replies(self) -> QuerySet: | ||
neko259
|
r1180 | return self.get_replies().exclude(refposts__threads__in=[self]) | ||
neko259
|
r1186 | def get_replies_with_images(self, view_fields_only=False) -> QuerySet: | ||
neko259
|
r950 | """ | ||
Gets replies that have at least one image attached | ||||
""" | ||||
neko259
|
r1590 | return self.get_replies(view_fields_only).filter( | ||
attachments__mimetype__in=FILE_TYPES_IMAGE).annotate(images_count=Count( | ||||
'attachments')).filter(images_count__gt=0) | ||||
neko259
|
r693 | |||
neko259
|
r1083 | def get_opening_post(self, only_id=False) -> Post: | ||
neko259
|
r691 | """ | ||
Gets the first post of the thread | ||||
""" | ||||
neko259
|
r1381 | query = self.get_replies().filter(opening=True) | ||
neko259
|
r949 | if only_id: | ||
query = query.only('id') | ||||
opening_post = query.first() | ||||
return opening_post | ||||
neko259
|
r691 | |||
neko259
|
r1106 | @cached_result() | ||
neko259
|
r1083 | def get_opening_post_id(self) -> int: | ||
neko259
|
r691 | """ | ||
Gets ID of the first thread post. | ||||
""" | ||||
neko259
|
r957 | return self.get_opening_post(only_id=True).id | ||
neko259
|
r691 | |||
def get_pub_time(self): | ||||
""" | ||||
Gets opening post's pub time because thread does not have its own one. | ||||
""" | ||||
neko259
|
r949 | return self.get_opening_post().pub_time | ||
neko259
|
r718 | |||
neko259
|
r875 | def __str__(self): | ||
neko259
|
r1968 | return 'T#{}/{}'.format(self.id, self.get_opening_post()) | ||
neko259
|
r1027 | |||
neko259
|
r1083 | def get_tag_url_list(self) -> list: | ||
neko259
|
r1874 | return boards.models.Tag.objects.get_tag_url_list(self.get_tags().all()) | ||
neko259
|
r1029 | |||
neko259
|
r1134 | def update_posts_time(self, exclude_posts=None): | ||
neko259
|
r1221 | last_edit_time = self.last_edit_time | ||
neko259
|
r1708 | for post in self.replies.all(): | ||
neko259
|
r1219 | if exclude_posts is None or post not in exclude_posts: | ||
neko259
|
r1134 | # Manual update is required because uids are generated on save | ||
neko259
|
r1221 | post.last_edit_time = last_edit_time | ||
neko259
|
r1134 | post.save(update_fields=['last_edit_time']) | ||
neko259
|
r1120 | |||
neko259
|
r1149 | def get_absolute_url(self): | ||
return self.get_opening_post().get_absolute_url() | ||||
neko259
|
r1257 | |||
def get_required_tags(self): | ||||
return self.get_tags().filter(required=True) | ||||
neko259
|
r1323 | |||
neko259
|
r1739 | def get_sections_str(self): | ||
return Tag.objects.get_tag_url_list(self.get_required_tags()) | ||||
neko259
|
r1323 | def get_replies_newer(self, post_id): | ||
return self.get_replies().filter(id__gt=post_id) | ||||
neko259
|
r1344 | def is_archived(self): | ||
neko259
|
r1414 | return self.get_status() == STATUS_ARCHIVE | ||
neko259
|
r1968 | def is_bumplimit(self): | ||
return self.get_status() == STATUS_BUMPLIMIT | ||||
neko259
|
r1414 | def get_status(self): | ||
return self.status | ||||
neko259
|
r1434 | |||
def is_monochrome(self): | ||||
return self.monochrome | ||||
neko259
|
r1471 | |||
neko259
|
r1951 | def is_stickerpack(self): | ||
return self.stickerpack | ||||
neko259
|
r1471 | # If tags have parent, add them to the tag list | ||
@transaction.atomic | ||||
def refresh_tags(self): | ||||
for tag in self.get_tags().all(): | ||||
parents = tag.get_all_parents() | ||||
if len(parents) > 0: | ||||
self.tags.add(*parents) | ||||
neko259
|
r1473 | def get_reply_tree(self): | ||
replies = self.get_replies().prefetch_related('refposts') | ||||
tree = [] | ||||
for reply in replies: | ||||
parents = reply.refposts.all() | ||||
neko259
|
r1489 | |||
neko259
|
r1473 | found_parent = False | ||
neko259
|
r1489 | searching_for_index = False | ||
neko259
|
r1473 | if len(parents) > 0: | ||
index = 0 | ||||
neko259
|
r1489 | parent_depth = 0 | ||
indexes_to_insert = [] | ||||
neko259
|
r1473 | for depth, element in tree: | ||
index += 1 | ||||
neko259
|
r1489 | |||
# If this element is next after parent on the same level, | ||||
# insert child before it | ||||
if searching_for_index and depth <= parent_depth: | ||||
indexes_to_insert.append((index - 1, parent_depth)) | ||||
searching_for_index = False | ||||
neko259
|
r1473 | if element in parents: | ||
neko259
|
r1488 | found_parent = True | ||
neko259
|
r1489 | searching_for_index = True | ||
parent_depth = depth | ||||
neko259
|
r1487 | |||
neko259
|
r2028 | if not found_parent: | ||
tree.append((0, reply)) | ||||
else: | ||||
if searching_for_index: | ||||
tree.append((parent_depth + 1, reply)) | ||||
neko259
|
r1489 | |||
neko259
|
r2028 | offset = 0 | ||
for last_index, parent_depth in indexes_to_insert: | ||||
tree.insert(last_index + offset, (parent_depth + 1, reply)) | ||||
offset += 1 | ||||
neko259
|
r1473 | |||
return tree | ||||