##// END OF EJS Templates
Use aggregation to get tags popularity
neko259 -
r607:9b64cf8c default
parent child Browse files
Show More
@@ -1,99 +1,103 b''
1 from boards.models import Thread, Post
1 from boards.models import Thread, Post
2 from django.db import models
2 from django.db import models
3 from django.db.models import Count
3 from django.db.models import Count, Sum
4
4
5 __author__ = 'neko259'
5 __author__ = 'neko259'
6
6
7 MAX_TAG_FONT = 1
7 MAX_TAG_FONT = 1
8 MIN_TAG_FONT = 0.2
8 MIN_TAG_FONT = 0.2
9
9
10 TAG_POPULARITY_MULTIPLIER = 20
10 TAG_POPULARITY_MULTIPLIER = 20
11
11
12 ARCHIVE_POPULARITY_MODIFIER = 0.5
12 ARCHIVE_POPULARITY_MODIFIER = 0.5
13
13
14
14
15 class TagManager(models.Manager):
15 class TagManager(models.Manager):
16
16
17 def get_not_empty_tags(self):
17 def get_not_empty_tags(self):
18 tags = self.annotate(Count('threads')) \
18 tags = self.annotate(Count('threads')) \
19 .filter(threads__count__gt=0).order_by('name')
19 .filter(threads__count__gt=0).order_by('name')
20
20
21 return tags
21 return tags
22
22
23
23
24 class Tag(models.Model):
24 class Tag(models.Model):
25 """
25 """
26 A tag is a text node assigned to the thread. The tag serves as a board
26 A tag is a text node assigned to the thread. The tag serves as a board
27 section. There can be multiple tags for each thread
27 section. There can be multiple tags for each thread
28 """
28 """
29
29
30 objects = TagManager()
30 objects = TagManager()
31
31
32 class Meta:
32 class Meta:
33 app_label = 'boards'
33 app_label = 'boards'
34
34
35 name = models.CharField(max_length=100)
35 name = models.CharField(max_length=100)
36 threads = models.ManyToManyField(Thread, null=True,
36 threads = models.ManyToManyField(Thread, null=True,
37 blank=True, related_name='tag+')
37 blank=True, related_name='tag+')
38 linked = models.ForeignKey('Tag', null=True, blank=True)
38 linked = models.ForeignKey('Tag', null=True, blank=True)
39
39
40 def __unicode__(self):
40 def __unicode__(self):
41 return self.name
41 return self.name
42
42
43 def is_empty(self):
43 def is_empty(self):
44 return self.get_thread_count() == 0
44 return self.get_thread_count() == 0
45
45
46 def get_thread_count(self):
46 def get_thread_count(self):
47 return self.threads.count()
47 return self.threads.count()
48
48
49 def get_popularity(self):
49 def get_popularity(self):
50 all_post_count = Post.objects.all().count()
50 all_post_count = Post.objects.all().count()
51
51
52 tag_reply_count = 0.0
52 tag_reply_count = 0.0
53
53
54 tag_reply_count += self.get_post_count()
54 tag_reply_count += self.get_post_count()
55 tag_reply_count +=\
55 tag_reply_count +=\
56 self.get_post_count(archived=True) * ARCHIVE_POPULARITY_MODIFIER
56 self.get_post_count(archived=True) * ARCHIVE_POPULARITY_MODIFIER
57
57
58 popularity = tag_reply_count / all_post_count
58 popularity = tag_reply_count / all_post_count
59
59
60 return popularity
60 return popularity
61
61
62 def get_linked_tags(self):
62 def get_linked_tags(self):
63 tag_list = []
63 tag_list = []
64 self.get_linked_tags_list(tag_list)
64 self.get_linked_tags_list(tag_list)
65
65
66 return tag_list
66 return tag_list
67
67
68 def get_linked_tags_list(self, tag_list=[]):
68 def get_linked_tags_list(self, tag_list=[]):
69 """
69 """
70 Returns the list of tags linked to current. The list can be got
70 Returns the list of tags linked to current. The list can be got
71 through returned value or tag_list parameter
71 through returned value or tag_list parameter
72 """
72 """
73
73
74 linked_tag = self.linked
74 linked_tag = self.linked
75
75
76 if linked_tag and not (linked_tag in tag_list):
76 if linked_tag and not (linked_tag in tag_list):
77 tag_list.append(linked_tag)
77 tag_list.append(linked_tag)
78
78
79 linked_tag.get_linked_tags_list(tag_list)
79 linked_tag.get_linked_tags_list(tag_list)
80
80
81 def get_font_value(self):
81 def get_font_value(self):
82 """Get tag font value to differ most popular tags in the list"""
82 """Get tag font value to differ most popular tags in the list"""
83
83
84 popularity = self.get_popularity()
84 popularity = self.get_popularity()
85
85
86 font_value = popularity * Tag.objects.get_not_empty_tags().count()
86 font_value = popularity * Tag.objects.get_not_empty_tags().count()
87 font_value = max(font_value, MIN_TAG_FONT)
87 font_value = max(font_value, MIN_TAG_FONT)
88 font_value = min(font_value, MAX_TAG_FONT)
88 font_value = min(font_value, MAX_TAG_FONT)
89
89
90 return str(font_value)
90 return str(font_value)
91
91
92 def get_post_count(self, archived=False):
92 def get_post_count(self, archived=False):
93 posts_count = 0
93 posts_count = 0
94
94
95 for thread in self.threads.annotate(Count('replies')).filter(
95 threads = self.threads.filter(archived=archived)
96 archived=archived):
96 if threads.exists():
97 posts_count += thread.replies__count
97 posts_count = threads.annotate(posts_count=Count('replies')).aggregate(
98 posts_sum=Sum('posts_count'))['posts_sum']
99
100 if not posts_count:
101 posts_count = 0
98
102
99 return posts_count
103 return posts_count
General Comments 0
You need to be logged in to leave comments. Login now