[MERGE] merging stats on forum
[odoo/odoo.git] / addons / website_forum / models / forum.py
1 # -*- coding: utf-8 -*-
2
3 from datetime import datetime
4 import logging
5 import math
6 import uuid
7 from werkzeug.exceptions import Forbidden
8
9 from openerp import _
10 from openerp import api, fields, models
11 from openerp import modules
12 from openerp import tools
13 from openerp import SUPERUSER_ID
14 from openerp.addons.website.models.website import slug
15 from openerp.exceptions import Warning
16
17 _logger = logging.getLogger(__name__)
18
19
20 class KarmaError(Forbidden):
21     """ Karma-related error, used for forum and posts. """
22     pass
23
24
25 class Forum(models.Model):
26     _name = 'forum.forum'
27     _description = 'Forum'
28     _inherit = ['mail.thread', 'website.seo.metadata']
29
30     def init(self, cr):
31         """ Add forum uuid for user email validation.
32
33         TDE TODO: move me somewhere else, auto_init ? """
34         forum_uuids = self.pool['ir.config_parameter'].search(cr, SUPERUSER_ID, [('key', '=', 'website_forum.uuid')])
35         if not forum_uuids:
36             self.pool['ir.config_parameter'].set_param(cr, SUPERUSER_ID, 'website_forum.uuid', str(uuid.uuid4()), ['base.group_system'])
37
38     @api.model
39     def _get_default_faq(self):
40         fname = modules.get_module_resource('website_forum', 'data', 'forum_default_faq.html')
41         with open(fname, 'r') as f:
42             return f.read()
43         return False
44
45     # description and use
46     name = fields.Char('Forum Name', required=True, translate=True)
47     faq = fields.Html('Guidelines', default=_get_default_faq, translate=True)
48     description = fields.Html(
49         'Description',
50         default='<p> This community is for professionals and enthusiasts of our products and services.'
51                 'Share and discuss the best content and new marketing ideas,'
52                 'build your professional profile and become a better marketer together.</p>')
53     default_order = fields.Selection([
54         ('create_date desc', 'Newest'),
55         ('write_date desc', 'Last Updated'),
56         ('vote_count desc', 'Most Voted'),
57         ('relevancy desc', 'Relevancy'),
58         ('child_count desc', 'Answered')],
59         string='Default Order', required=True, default='write_date desc')
60     relevancy_post_vote = fields.Float('First Relevancy Parameter', default=0.8)
61     relevancy_time_decay = fields.Float('Second Relevancy Parameter', default=1.8)
62     default_post_type = fields.Selection([
63         ('question', 'Question'),
64         ('discussion', 'Discussion'),
65         ('link', 'Link')],
66         string='Default Post', required=True, default='question')
67     allow_question = fields.Boolean('Questions', help="Users can answer only once per question. Contributors can edit answers and mark the right ones.", default=True)
68     allow_discussion = fields.Boolean('Discussions', default=True)
69     allow_link = fields.Boolean('Links', help="When clicking on the post, it redirects to an external link", default=True)
70     # karma generation
71     karma_gen_question_new = fields.Integer(string='Asking a question', default=2)
72     karma_gen_question_upvote = fields.Integer(string='Question upvoted', default=5)
73     karma_gen_question_downvote = fields.Integer(string='Question downvoted', default=-2)
74     karma_gen_answer_upvote = fields.Integer(string='Answer upvoted', default=10)
75     karma_gen_answer_downvote = fields.Integer(string='Answer downvoted', default=-2)
76     karma_gen_answer_accept = fields.Integer(string='Accepting an answer', default=2)
77     karma_gen_answer_accepted = fields.Integer(string='Answer accepted', default=15)
78     karma_gen_answer_flagged = fields.Integer(string='Answer flagged', default=-100)
79     # karma-based actions
80     karma_ask = fields.Integer(string='Ask a new question', default=3)
81     karma_answer = fields.Integer(string='Answer a question', default=3)
82     karma_edit_own = fields.Integer(string='Edit its own posts', default=1)
83     karma_edit_all = fields.Integer(string='Edit all posts', default=300)
84     karma_close_own = fields.Integer(string='Close its own posts', default=100)
85     karma_close_all = fields.Integer(string='Close all posts', default=500)
86     karma_unlink_own = fields.Integer(string='Delete its own posts', default=500)
87     karma_unlink_all = fields.Integer(string='Delete all posts', default=1000)
88     karma_upvote = fields.Integer(string='Upvote', default=5)
89     karma_downvote = fields.Integer(string='Downvote', default=50)
90     karma_answer_accept_own = fields.Integer(string='Accept an answer on its own questions', default=20)
91     karma_answer_accept_all = fields.Integer(string='Accept an answers to all questions', default=500)
92     karma_editor_link_files = fields.Integer(string='Linking files (Editor)', default=20)
93     karma_editor_clickable_link = fields.Integer(string='Add clickable links (Editor)', default=20)
94     karma_comment_own = fields.Integer(string='Comment its own posts', default=1)
95     karma_comment_all = fields.Integer(string='Comment all posts', default=1)
96     karma_comment_convert_own = fields.Integer(string='Convert its own answers to comments and vice versa', default=50)
97     karma_comment_convert_all = fields.Integer(string='Convert all answers to answers and vice versa', default=500)
98     karma_comment_unlink_own = fields.Integer(string='Unlink its own comments', default=50)
99     karma_comment_unlink_all = fields.Integer(string='Unlinnk all comments', default=500)
100     karma_retag = fields.Integer(string='Change question tags', default=75)
101     karma_flag = fields.Integer(string='Flag a post as offensive', default=500)
102     karma_dofollow = fields.Integer(string='Disabled links', help='If the author has not enough karma, a nofollow attribute is added to links', default=500)
103
104     @api.model
105     def create(self, values):
106         return super(Forum, self.with_context(mail_create_nolog=True)).create(values)
107
108     @api.model
109     def _tag_to_write_vals(self, tags=''):
110         User = self.env['res.users']
111         Tag = self.env['forum.tag']
112         post_tags = []
113         existing_keep = []
114         for tag in filter(None, tags.split(',')):
115             if tag.startswith('_'):  # it's a new tag
116                 # check that not arleady created meanwhile or maybe excluded by the limit on the search
117                 tag_ids = Tag.search([('name', '=', tag[1:])])
118                 if tag_ids:
119                     existing_keep.append(int(tag_ids[0]))
120                 else:
121                     # check if user have Karma needed to create need tag
122                     user = User.sudo().browse(self._uid)
123                     if user.exists() and user.karma >= self.karma_retag:
124                         post_tags.append((0, 0, {'name': tag[1:], 'forum_id': self.id}))
125             else:
126                 existing_keep.append(int(tag))
127         post_tags.insert(0, [6, 0, existing_keep])
128         return post_tags
129
130
131 class Post(models.Model):
132     _name = 'forum.post'
133     _description = 'Forum Post'
134     _inherit = ['mail.thread', 'website.seo.metadata']
135     _order = "is_correct DESC, vote_count DESC, write_date DESC"
136
137     name = fields.Char('Title')
138     forum_id = fields.Many2one('forum.forum', string='Forum', required=True)
139     content = fields.Html('Content')
140     content_link = fields.Char('URL', help="URL of Link Articles")
141     tag_ids = fields.Many2many('forum.tag', 'forum_tag_rel', 'forum_id', 'forum_tag_id', string='Tags')
142     state = fields.Selection([('active', 'Active'), ('close', 'Close'), ('offensive', 'Offensive')], string='Status', default='active')
143     views = fields.Integer('Number of Views', default=0)
144     active = fields.Boolean('Active', default=True)
145     post_type = fields.Selection([
146         ('question', 'Question'),
147         ('link', 'Article'),
148         ('discussion', 'Discussion')],
149         string='Type', default='question')
150     website_message_ids = fields.One2many(
151         'mail.message', 'res_id',
152         domain=lambda self: ['&', ('model', '=', self._name), ('type', 'in', ['email', 'comment'])],
153         string='Post Messages', help="Comments on forum post",
154     )
155     # history
156     create_date = fields.Datetime('Asked on', select=True, readonly=True)
157     create_uid = fields.Many2one('res.users', string='Created by', select=True, readonly=True)
158     write_date = fields.Datetime('Update on', select=True, readonly=True)
159     write_uid = fields.Many2one('res.users', string='Updated by', select=True, readonly=True)
160     relevancy = fields.Float('Relevancy', compute="_compute_relevancy", store=True)
161
162     @api.one
163     @api.depends('vote_count', 'forum_id.relevancy_post_vote', 'forum_id.relevancy_time_decay')
164     def _compute_relevancy(self):
165         days = (datetime.today() - datetime.strptime(self.create_date, tools.DEFAULT_SERVER_DATETIME_FORMAT)).days
166         self.relevancy = math.copysign(1, self.vote_count) * (abs(self.vote_count - 1) ** self.forum_id.relevancy_post_vote / (days + 2) ** self.forum_id.relevancy_time_decay)
167
168     # vote
169     vote_ids = fields.One2many('forum.post.vote', 'post_id', string='Votes')
170     user_vote = fields.Integer('My Vote', compute='_get_user_vote')
171     vote_count = fields.Integer('Votes', compute='_get_vote_count', store=True)
172
173     @api.multi
174     def _get_user_vote(self):
175         votes = self.env['forum.post.vote'].search_read([('post_id', 'in', self._ids), ('user_id', '=', self._uid)], ['vote', 'post_id'])
176         mapped_vote = dict([(v['post_id'][0], v['vote']) for v in votes])
177         for vote in self:
178             vote.user_vote = mapped_vote.get(vote.id, 0)
179
180     @api.multi
181     @api.depends('vote_ids')
182     def _get_vote_count(self):
183         read_group_res = self.env['forum.post.vote'].read_group([('post_id', 'in', self._ids)], ['post_id', 'vote'], ['post_id', 'vote'], lazy=False)
184         result = dict.fromkeys(self._ids, 0)
185         for data in read_group_res:
186             result[data['post_id'][0]] += data['__count'] * int(data['vote'])
187         for post in self:
188             post.vote_count = result[post.id]
189
190     # favorite
191     favourite_ids = fields.Many2many('res.users', string='Favourite')
192     user_favourite = fields.Boolean('Is Favourite', compute='_get_user_favourite')
193     favourite_count = fields.Integer('Favorite Count', compute='_get_favorite_count', store=True)
194
195     @api.one
196     def _get_user_favourite(self):
197         self.user_favourite = self._uid in self.favourite_ids.ids
198
199     @api.one
200     @api.depends('favourite_ids')
201     def _get_favorite_count(self):
202         self.favourite_count = len(self.favourite_ids)
203
204     # hierarchy
205     is_correct = fields.Boolean('Correct', help='Correct answer or answer accepted')
206     parent_id = fields.Many2one('forum.post', string='Question', ondelete='cascade')
207     self_reply = fields.Boolean('Reply to own question', compute='_is_self_reply', store=True)
208     child_ids = fields.One2many('forum.post', 'parent_id', string='Answers')
209     child_count = fields.Integer('Number of answers', compute='_get_child_count', store=True)
210     uid_has_answered = fields.Boolean('Has Answered', compute='_get_uid_has_answered')
211     has_validated_answer = fields.Boolean('Is answered', compute='_get_has_validated_answer', store=True)
212
213     @api.one
214     @api.depends('create_uid', 'parent_id')
215     def _is_self_reply(self):
216         self.self_reply = self.parent_id.create_uid.id == self._uid
217
218     @api.one
219     @api.depends('child_ids.create_uid', 'website_message_ids')
220     def _get_child_count(self):
221         def process(node):
222             total = len(node.website_message_ids) + len(node.child_ids)
223             for child in node.child_ids:
224                 total += process(child)
225             return total
226         self.child_count = process(self)
227
228     @api.one
229     def _get_uid_has_answered(self):
230         self.uid_has_answered = any(answer.create_uid.id == self._uid for answer in self.child_ids)
231
232     @api.one
233     @api.depends('child_ids.is_correct')
234     def _get_has_validated_answer(self):
235         self.has_validated_answer = any(answer.is_correct for answer in self.child_ids)
236
237     # closing
238     closed_reason_id = fields.Many2one('forum.post.reason', string='Reason')
239     closed_uid = fields.Many2one('res.users', string='Closed by', select=1)
240     closed_date = fields.Datetime('Closed on', readonly=True)
241     # karma calculation and access
242     karma_accept = fields.Integer('Convert comment to answer', compute='_get_post_karma_rights')
243     karma_edit = fields.Integer('Karma to edit', compute='_get_post_karma_rights')
244     karma_close = fields.Integer('Karma to close', compute='_get_post_karma_rights')
245     karma_unlink = fields.Integer('Karma to unlink', compute='_get_post_karma_rights')
246     karma_comment = fields.Integer('Karma to comment', compute='_get_post_karma_rights')
247     karma_comment_convert = fields.Integer('Karma to convert comment to answer', compute='_get_post_karma_rights')
248     can_ask = fields.Boolean('Can Ask', compute='_get_post_karma_rights')
249     can_answer = fields.Boolean('Can Answer', compute='_get_post_karma_rights')
250     can_accept = fields.Boolean('Can Accept', compute='_get_post_karma_rights')
251     can_edit = fields.Boolean('Can Edit', compute='_get_post_karma_rights')
252     can_close = fields.Boolean('Can Close', compute='_get_post_karma_rights')
253     can_unlink = fields.Boolean('Can Unlink', compute='_get_post_karma_rights')
254     can_upvote = fields.Boolean('Can Upvote', compute='_get_post_karma_rights')
255     can_downvote = fields.Boolean('Can Downvote', compute='_get_post_karma_rights')
256     can_comment = fields.Boolean('Can Comment', compute='_get_post_karma_rights')
257     can_comment_convert = fields.Boolean('Can Convert to Comment', compute='_get_post_karma_rights')
258
259     @api.one
260     def _get_post_karma_rights(self):
261         user = self.env.user
262
263         self.karma_accept = self.parent_id and self.parent_id.create_uid.id == self._uid and self.forum_id.karma_answer_accept_own or self.forum_id.karma_answer_accept_all
264         self.karma_edit = self.create_uid.id == self._uid and self.forum_id.karma_edit_own or self.forum_id.karma_edit_all
265         self.karma_close = self.create_uid.id == self._uid and self.forum_id.karma_close_own or self.forum_id.karma_close_all
266         self.karma_unlink = self.create_uid.id == self._uid and self.forum_id.karma_unlink_own or self.forum_id.karma_unlink_all
267         self.karma_comment = self.create_uid.id == self._uid and self.forum_id.karma_comment_own or self.forum_id.karma_comment_all
268         self.karma_comment_convert = self.create_uid.id == self._uid and self.forum_id.karma_comment_convert_own or self.forum_id.karma_comment_convert_all
269
270         self.can_ask = user.karma >= self.forum_id.karma_ask
271         self.can_answer = user.karma >= self.forum_id.karma_answer
272         self.can_accept = user.karma >= self.karma_accept
273         self.can_edit = user.karma >= self.karma_edit
274         self.can_close = user.karma >= self.karma_close
275         self.can_unlink = user.karma >= self.karma_unlink
276         self.can_upvote = user.karma >= self.forum_id.karma_upvote
277         self.can_downvote = user.karma >= self.forum_id.karma_downvote
278         self.can_comment = user.karma >= self.karma_comment
279         self.can_comment_convert = user.karma >= self.karma_comment_convert
280
281     @api.model
282     def create(self, vals):
283         post = super(Post, self.with_context(mail_create_nolog=True)).create(vals)
284         # deleted or closed questions
285         if post.parent_id and (post.parent_id.state == 'close' or post.parent_id.active is False):
286             raise Warning(_('Posting answer on a [Deleted] or [Closed] question is not possible'))
287         # karma-based access
288         if not post.parent_id and not post.can_ask:
289             raise KarmaError('Not enough karma to create a new question')
290         elif post.parent_id and not post.can_answer:
291             raise KarmaError('Not enough karma to answer to a question')
292         # messaging and chatter
293         base_url = self.env['ir.config_parameter'].get_param('web.base.url')
294         if post.parent_id:
295             body = _(
296                 '<p>A new answer for <i>%s</i> has been posted. <a href="%s/forum/%s/question/%s">Click here to access the post.</a></p>' %
297                 (post.parent_id.name, base_url, slug(post.parent_id.forum_id), slug(post.parent_id))
298             )
299             post.parent_id.message_post(subject=_('Re: %s') % post.parent_id.name, body=body, subtype='website_forum.mt_answer_new')
300         else:
301             body = _(
302                 '<p>A new question <i>%s</i> has been asked on %s. <a href="%s/forum/%s/question/%s">Click here to access the question.</a></p>' %
303                 (post.name, post.forum_id.name, base_url, slug(post.forum_id), slug(post))
304             )
305             post.message_post(subject=post.name, body=body, subtype='website_forum.mt_question_new')
306             self.env.user.sudo().add_karma(post.forum_id.karma_gen_question_new)
307         return post
308
309     @api.multi
310     def write(self, vals):
311         if 'state' in vals:
312             if vals['state'] in ['active', 'close'] and any(not post.can_close for post in self):
313                 raise KarmaError('Not enough karma to close or reopen a post.')
314         if 'active' in vals:
315             if any(not post.can_unlink for post in self):
316                 raise KarmaError('Not enough karma to delete or reactivate a post')
317         if 'is_correct' in vals:
318             if any(not post.can_accept for post in self):
319                 raise KarmaError('Not enough karma to accept or refuse an answer')
320             # update karma except for self-acceptance
321             mult = 1 if vals['is_correct'] else -1
322             for post in self:
323                 if vals['is_correct'] != post.is_correct and post.create_uid.id != self._uid:
324                     post.create_uid.sudo().add_karma(post.forum_id.karma_gen_answer_accepted * mult)
325                     self.env.user.sudo().add_karma(post.forum_id.karma_gen_answer_accept * mult)
326         if any(key not in ['state', 'active', 'is_correct', 'closed_uid', 'closed_date', 'closed_reason_id'] for key in vals.keys()) and any(not post.can_edit for post in self):
327             raise KarmaError('Not enough karma to edit a post.')
328
329         res = super(Post, self).write(vals)
330         # if post content modify, notify followers
331         if 'content' in vals or 'name' in vals:
332             for post in self:
333                 if post.parent_id:
334                     body, subtype = _('Answer Edited'), 'website_forum.mt_answer_edit'
335                     obj_id = post.parent_id
336                 else:
337                     body, subtype = _('Question Edited'), 'website_forum.mt_question_edit'
338                     obj_id = post
339                 obj_id.message_post(body=body, subtype=subtype)
340         return res
341
342     @api.multi
343     def reopen(self):
344         if any(post.parent_id or post.state != 'close' for post in self):
345             return False
346
347         reason_offensive = self.env.ref('website_forum.reason_7')
348         reason_spam = self.env.ref('website_forum.reason_8')
349         for post in self:
350             if post.closed_reason_id in (reason_offensive, reason_spam):
351                 _logger.info('Upvoting user <%s>, reopening spam/offensive question',
352                              post.create_uid)
353                 # TODO: in master, consider making this a tunable karma parameter
354                 post.create_uid.sudo().add_karma(post.forum_id.karma_gen_question_downvote * -5)
355
356         self.sudo().write({'state': 'active'})
357
358     @api.multi
359     def close(self, reason_id):
360         if any(post.parent_id for post in self):
361             return False
362
363         reason_offensive = self.env.ref('website_forum.reason_7').id
364         reason_spam = self.env.ref('website_forum.reason_8').id
365         if reason_id in (reason_offensive, reason_spam):
366             for post in self:
367                 _logger.info('Downvoting user <%s> for posting spam/offensive contents',
368                              post.create_uid)
369                 # TODO: in master, consider making this a tunable karma parameter
370                 post.create_uid.sudo().add_karma(post.forum_id.karma_gen_question_downvote * 5)
371
372         self.write({
373             'state': 'close',
374             'closed_uid': self._uid,
375             'closed_date': datetime.today().strftime(tools.DEFAULT_SERVER_DATETIME_FORMAT),
376             'closed_reason_id': reason_id,
377         })
378         return True
379
380     @api.multi
381     def unlink(self):
382         if any(not post.can_unlink for post in self):
383             raise KarmaError('Not enough karma to unlink a post')
384         # if unlinking an answer with accepted answer: remove provided karma
385         for post in self:
386             if post.is_correct:
387                 post.create_uid.sudo().add_karma(post.forum_id.karma_gen_answer_accepted * -1)
388                 self.env.user.sudo().add_karma(post.forum_id.karma_gen_answer_accepted * -1)
389         return super(Post, self).unlink()
390
391     @api.multi
392     def vote(self, upvote=True):
393         Vote = self.env['forum.post.vote']
394         vote_ids = Vote.search([('post_id', 'in', self._ids), ('user_id', '=', self._uid)])
395         new_vote = '1' if upvote else '-1'
396         voted_forum_ids = set()
397         if vote_ids:
398             for vote in vote_ids:
399                 if upvote:
400                     new_vote = '0' if vote.vote == '-1' else '1'
401                 else:
402                     new_vote = '0' if vote.vote == '1' else '-1'
403                 vote.vote = new_vote
404                 voted_forum_ids.add(vote.post_id.id)
405         for post_id in set(self._ids) - voted_forum_ids:
406             for post_id in self._ids:
407                 Vote.create({'post_id': post_id, 'vote': new_vote})
408         return {'vote_count': self.vote_count, 'user_vote': new_vote}
409
410     @api.one
411     def convert_answer_to_comment(self):
412         """ Tools to convert an answer (forum.post) to a comment (mail.message).
413         The original post is unlinked and a new comment is posted on the question
414         using the post create_uid as the comment's author. """
415         if not self.parent_id:
416             return False
417
418         # karma-based action check: use the post field that computed own/all value
419         if not self.can_comment_convert:
420             raise KarmaError('Not enough karma to convert an answer to a comment')
421
422         # post the message
423         question = self.parent_id
424         values = {
425             'author_id': self.create_uid.partner_id.id,
426             'body': tools.html2plaintext(self.content),
427             'type': 'comment',
428             'subtype': 'mail.mt_comment',
429             'date': self.create_date,
430         }
431         new_message = self.browse(question.id).with_context(mail_create_nosubcribe=True).message_post(**values)
432
433         # unlink the original answer, using SUPERUSER_ID to avoid karma issues
434         self.sudo().unlink()
435
436         return new_message
437
438     @api.model
439     def convert_comment_to_answer(self, message_id, default=None):
440         """ Tool to convert a comment (mail.message) into an answer (forum.post).
441         The original comment is unlinked and a new answer from the comment's author
442         is created. Nothing is done if the comment's author already answered the
443         question. """
444         comment = self.env['mail.message'].sudo().browse(message_id)
445         post = self.browse(comment.res_id)
446         if not comment.author_id or not comment.author_id.user_ids:  # only comment posted by users can be converted
447             return False
448
449         # karma-based action check: must check the message's author to know if own / all
450         karma_convert = comment.author_id.id == self.env.user.partner_id.id and post.forum_id.karma_comment_convert_own or post.forum_id.karma_comment_convert_all
451         can_convert = self.env.user.karma >= karma_convert
452         if not can_convert:
453             raise KarmaError('Not enough karma to convert a comment to an answer')
454
455         # check the message's author has not already an answer
456         question = post.parent_id if post.parent_id else post
457         post_create_uid = comment.author_id.user_ids[0]
458         if any(answer.create_uid.id == post_create_uid.id for answer in question.child_ids):
459             return False
460
461         # create the new post
462         post_values = {
463             'forum_id': question.forum_id.id,
464             'content': comment.body,
465             'parent_id': question.id,
466         }
467         # done with the author user to have create_uid correctly set
468         new_post = self.sudo(post_create_uid.id).create(post_values)
469
470         # delete comment
471         comment.unlink()
472
473         return new_post
474
475     @api.one
476     def unlink_comment(self, message_id):
477         user = self.env.user
478         comment = self.env['mail.message'].sudo().browse(message_id)
479         if not comment.model == 'forum.post' or not comment.res_id == self.id:
480             return False
481         # karma-based action check: must check the message's author to know if own or all
482         karma_unlink = comment.author_id.id == user.partner_id.id and self.forum_id.karma_comment_unlink_own or self.forum_id.karma_comment_unlink_all
483         can_unlink = user.karma >= karma_unlink
484         if not can_unlink:
485             raise KarmaError('Not enough karma to unlink a comment')
486         return comment.unlink()
487
488     @api.multi
489     def set_viewed(self):
490         self._cr.execute("""UPDATE forum_post SET views = views+1 WHERE id IN %s""", (self._ids,))
491         return True
492
493     @api.model
494     def _get_access_link(self, mail, partner):
495         post = self.browse(mail.res_id)
496         res_id = post.parent_id and "%s#answer-%s" % (post.parent_id.id, post.id) or post.id
497         return "/forum/%s/question/%s" % (post.forum_id.id, res_id)
498
499     @api.cr_uid_ids_context
500     def message_post(self, cr, uid, thread_id, type='notification', subtype=None, context=None, **kwargs):
501         if thread_id and type == 'comment':  # user comments have a restriction on karma
502             if isinstance(thread_id, (list, tuple)):
503                 post_id = thread_id[0]
504             else:
505                 post_id = thread_id
506             post = self.browse(cr, uid, post_id, context=context)
507             # TDE FIXME: trigger browse because otherwise the function field is not compted - check with RCO
508             tmp1, tmp2 = post.karma_comment, post.can_comment
509             user = self.pool['res.users'].browse(cr, uid, uid)
510             tmp3 = user.karma
511             # TDE END FIXME
512             if not post.can_comment:
513                 raise KarmaError('Not enough karma to comment')
514         return super(Post, self).message_post(cr, uid, thread_id, type=type, subtype=subtype, context=context, **kwargs)
515
516
517 class PostReason(models.Model):
518     _name = "forum.post.reason"
519     _description = "Post Closing Reason"
520     _order = 'name'
521
522     name = fields.Char(string='Closing Reason', required=True, translate=True)
523
524
525 class Vote(models.Model):
526     _name = 'forum.post.vote'
527     _description = 'Vote'
528
529     post_id = fields.Many2one('forum.post', string='Post', ondelete='cascade', required=True)
530     user_id = fields.Many2one('res.users', string='User', required=True, default=lambda self: self._uid)
531     vote = fields.Selection([('1', '1'), ('-1', '-1'), ('0', '0')], string='Vote', required=True, default='1')
532     create_date = fields.Datetime('Create Date', select=True, readonly=True)
533     forum_id = fields.Many2one('forum.forum', string='Forum', related="post_id.forum_id", store=True)
534     recipient_id = fields.Many2one('res.users', string='To', related="post_id.create_uid", store=True)
535
536     def _get_karma_value(self, old_vote, new_vote, up_karma, down_karma):
537         _karma_upd = {
538             '-1': {'-1': 0, '0': -1 * down_karma, '1': -1 * down_karma + up_karma},
539             '0': {'-1': 1 * down_karma, '0': 0, '1': up_karma},
540             '1': {'-1': -1 * up_karma + down_karma, '0': -1 * up_karma, '1': 0}
541         }
542         return _karma_upd[old_vote][new_vote]
543
544     @api.model
545     def create(self, vals):
546         vote = super(Vote, self).create(vals)
547
548         # own post check
549         if vote.user_id.id == vote.post_id.create_uid.id:
550             raise Warning('Not allowed to vote for its own post')
551         # karma check
552         if vote.vote == '1' and not vote.post_id.can_upvote:
553             raise KarmaError('Not enough karma to upvote.')
554         elif vote.vote == '-1' and not vote.post_id.can_downvote:
555             raise KarmaError('Not enough karma to downvote.')
556
557         if vote.post_id.parent_id:
558             karma_value = self._get_karma_value('0', vote.vote, vote.forum_id.karma_gen_answer_upvote, vote.forum_id.karma_gen_answer_downvote)
559         else:
560             karma_value = self._get_karma_value('0', vote.vote, vote.forum_id.karma_gen_question_upvote, vote.forum_id.karma_gen_question_downvote)
561         vote.recipient_id.sudo().add_karma(karma_value)
562         return vote
563
564     @api.multi
565     def write(self, values):
566         if 'vote' in values:
567             for vote in self:
568                 # own post check
569                 if vote.user_id.id == vote.post_id.create_uid.id:
570                     raise Warning('Not allowed to vote for its own post')
571                 # karma check
572                 if (values['vote'] == '1' or vote.vote == '-1' and values['vote'] == '0') and not vote.post_id.can_upvote:
573                     raise KarmaError('Not enough karma to upvote.')
574                 elif (values['vote'] == '-1' or vote.vote == '1' and values['vote'] == '0') and not vote.post_id.can_downvote:
575                     raise KarmaError('Not enough karma to downvote.')
576
577                 # karma update
578                 if vote.post_id.parent_id:
579                     karma_value = self._get_karma_value(vote.vote, values['vote'], vote.forum_id.karma_gen_answer_upvote, vote.forum_id.karma_gen_answer_downvote)
580                 else:
581                     karma_value = self._get_karma_value(vote.vote, values['vote'], vote.forum_id.karma_gen_question_upvote, vote.forum_id.karma_gen_question_downvote)
582                 vote.recipient_id.sudo().add_karma(karma_value)
583         res = super(Vote, self).write(values)
584         return res
585
586
587 class Tags(models.Model):
588     _name = "forum.tag"
589     _description = "Forum Tag"
590     _inherit = ['website.seo.metadata']
591
592     name = fields.Char('Name', required=True)
593     create_uid = fields.Many2one('res.users', string='Created by', readonly=True)
594     forum_id = fields.Many2one('forum.forum', string='Forum', required=True)
595     post_ids = fields.Many2many('forum.post', 'forum_tag_rel', 'forum_tag_id', 'forum_id', string='Posts')
596     posts_count = fields.Integer('Number of Posts', compute='_get_posts_count', store=True)
597
598     @api.multi
599     @api.depends("post_ids.tag_ids")
600     def _get_posts_count(self):
601         for tag in self:
602             tag.posts_count = len(tag.post_ids)