notification.py
281 lines
| 10.3 KiB
| text/x-python
|
PythonLexer
r1702 | # -*- coding: utf-8 -*- | |||
""" | ||||
rhodecode.model.notification | ||||
r1839 | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | |||
r1702 | ||||
Model for notifications | ||||
r1800 | ||||
r1702 | :created_on: Nov 20, 2011 | |||
:author: marcink | ||||
r1824 | :copyright: (C) 2010-2012 Marcin Kuzminski <marcin@python-works.com> | |||
r1702 | :license: GPLv3, see COPYING for more details. | |||
""" | ||||
# This program is free software: you can redistribute it and/or modify | ||||
# it under the terms of the GNU General Public License as published by | ||||
# the Free Software Foundation, either version 3 of the License, or | ||||
# (at your option) any later version. | ||||
# | ||||
# This program is distributed in the hope that it will be useful, | ||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||
# GNU General Public License for more details. | ||||
# | ||||
# You should have received a copy of the GNU General Public License | ||||
# along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||
r1717 | import os | |||
r1702 | import logging | |||
import traceback | ||||
r3423 | from pylons import tmpl_context as c | |||
r1702 | from pylons.i18n.translation import _ | |||
r1723 | import rhodecode | |||
r1717 | from rhodecode.lib import helpers as h | |||
r1702 | from rhodecode.model import BaseModel | |||
from rhodecode.model.db import Notification, User, UserNotification | ||||
r3430 | from rhodecode.model.meta import Session | |||
r1702 | ||||
r1712 | log = logging.getLogger(__name__) | |||
r1702 | ||||
r1716 | ||||
r1702 | class NotificationModel(BaseModel): | |||
r2522 | cls = Notification | |||
r1712 | def __get_notification(self, notification): | |||
if isinstance(notification, Notification): | ||||
return notification | ||||
r2149 | elif isinstance(notification, (int, long)): | |||
r1712 | return Notification.get(notification) | |||
else: | ||||
if notification: | ||||
r2149 | raise Exception('notification must be int, long or Instance' | |||
r1712 | ' of Notification got %s' % type(notification)) | |||
r1731 | def create(self, created_by, subject, body, recipients=None, | |||
type_=Notification.TYPE_MESSAGE, with_email=True, | ||||
r3430 | email_kwargs={}, email_subject=None): | |||
r1703 | """ | |||
r1800 | ||||
r1703 | Creates notification of given type | |||
r1800 | ||||
r1703 | :param created_by: int, str or User instance. User who created this | |||
notification | ||||
:param subject: | ||||
:param body: | ||||
r1818 | :param recipients: list of int, str or User objects, when None | |||
r1731 | is given send to all admins | |||
r1703 | :param type_: type of notification | |||
r1731 | :param with_email: send email with this notification | |||
:param email_kwargs: additional dict to pass as args to email template | ||||
r3430 | :param email_subject: use given subject as email subject | |||
r1703 | """ | |||
r1722 | from rhodecode.lib.celerylib import tasks, run_task | |||
r1702 | ||||
r1731 | if recipients and not getattr(recipients, '__iter__', False): | |||
Mads Kiilerich
|
r3270 | raise Exception('recipients must be a list or iterable') | ||
r1702 | ||||
r2432 | created_by_obj = self._get_user(created_by) | |||
r1702 | ||||
r1731 | if recipients: | |||
recipients_objs = [] | ||||
for u in recipients: | ||||
r2432 | obj = self._get_user(u) | |||
r1731 | if obj: | |||
recipients_objs.append(obj) | ||||
recipients_objs = set(recipients_objs) | ||||
r2077 | log.debug('sending notifications %s to %s' % ( | |||
type_, recipients_objs) | ||||
) | ||||
r1731 | else: | |||
# empty recipients means to all admins | ||||
recipients_objs = User.query().filter(User.admin == True).all() | ||||
r2077 | log.debug('sending notifications %s to admins: %s' % ( | |||
type_, recipients_objs) | ||||
) | ||||
notif = Notification.create( | ||||
created_by=created_by_obj, subject=subject, | ||||
body=body, recipients=recipients_objs, type_=type_ | ||||
) | ||||
r1717 | ||||
Mads Kiilerich
|
r3625 | if not with_email: | ||
r1731 | return notif | |||
r2387 | #don't send email to person who created this comment | |||
rec_objs = set(recipients_objs).difference(set([created_by_obj])) | ||||
# send email with notification to all other participants | ||||
for rec in rec_objs: | ||||
r3430 | if not email_subject: | |||
r3448 | email_subject = NotificationModel()\ | |||
.make_description(notif, show_age=False) | ||||
r1731 | type_ = type_ | |||
r3448 | email_body = None # we set body to none, we just send HTML emails | |||
r2296 | ## this is passed into template | |||
r1800 | kwargs = {'subject': subject, 'body': h.rst_w_mentions(body)} | |||
r1731 | kwargs.update(email_kwargs) | |||
r1717 | email_body_html = EmailNotificationModel()\ | |||
r1731 | .get_email_tmpl(type_, **kwargs) | |||
r2156 | ||||
r1722 | run_task(tasks.send_email, rec.email, email_subject, email_body, | |||
r1717 | email_body_html) | |||
return notif | ||||
r1702 | ||||
r1713 | def delete(self, user, notification): | |||
# we don't want to remove actual notification just the assignment | ||||
r1712 | try: | |||
r1713 | notification = self.__get_notification(notification) | |||
r2432 | user = self._get_user(user) | |||
r1713 | if notification and user: | |||
r1717 | obj = UserNotification.query()\ | |||
.filter(UserNotification.user == user)\ | ||||
.filter(UserNotification.notification | ||||
== notification)\ | ||||
.one() | ||||
r3430 | Session().delete(obj) | |||
r1712 | return True | |||
except Exception: | ||||
log.error(traceback.format_exc()) | ||||
raise | ||||
r1702 | ||||
r2433 | def get_for_user(self, user, filter_=None): | |||
""" | ||||
Get mentions for given user, filter them if filter dict is given | ||||
:param user: | ||||
:param filter: | ||||
""" | ||||
r2432 | user = self._get_user(user) | |||
r2433 | ||||
q = UserNotification.query()\ | ||||
.filter(UserNotification.user == user)\ | ||||
.join((Notification, UserNotification.notification_id == | ||||
Notification.notification_id)) | ||||
if filter_: | ||||
r2503 | q = q.filter(Notification.type_.in_(filter_)) | |||
r1702 | ||||
r2433 | return q.all() | |||
r2610 | def mark_read(self, user, notification): | |||
try: | ||||
notification = self.__get_notification(notification) | ||||
user = self._get_user(user) | ||||
if notification and user: | ||||
obj = UserNotification.query()\ | ||||
.filter(UserNotification.user == user)\ | ||||
.filter(UserNotification.notification | ||||
== notification)\ | ||||
.one() | ||||
obj.read = True | ||||
r3430 | Session().add(obj) | |||
r2610 | return True | |||
except Exception: | ||||
log.error(traceback.format_exc()) | ||||
raise | ||||
r2433 | def mark_all_read_for_user(self, user, filter_=None): | |||
r2432 | user = self._get_user(user) | |||
r2433 | q = UserNotification.query()\ | |||
.filter(UserNotification.user == user)\ | ||||
r2217 | .filter(UserNotification.read == False)\ | |||
r2433 | .join((Notification, UserNotification.notification_id == | |||
Notification.notification_id)) | ||||
if filter_: | ||||
r2503 | q = q.filter(Notification.type_.in_(filter_)) | |||
r2433 | ||||
# this is a little inefficient but sqlalchemy doesn't support | ||||
# update on joined tables :( | ||||
for obj in q.all(): | ||||
obj.read = True | ||||
r3430 | Session().add(obj) | |||
r1791 | ||||
r1713 | def get_unread_cnt_for_user(self, user): | |||
r2432 | user = self._get_user(user) | |||
r1702 | return UserNotification.query()\ | |||
r1712 | .filter(UserNotification.read == False)\ | |||
r1713 | .filter(UserNotification.user == user).count() | |||
r1702 | ||||
r1713 | def get_unread_for_user(self, user): | |||
r2432 | user = self._get_user(user) | |||
r1702 | return [x.notification for x in UserNotification.query()\ | |||
r1712 | .filter(UserNotification.read == False)\ | |||
r1713 | .filter(UserNotification.user == user).all()] | |||
r1712 | ||||
def get_user_notification(self, user, notification): | ||||
r2432 | user = self._get_user(user) | |||
r1712 | notification = self.__get_notification(notification) | |||
return UserNotification.query()\ | ||||
.filter(UserNotification.notification == notification)\ | ||||
.filter(UserNotification.user == user).scalar() | ||||
r1717 | def make_description(self, notification, show_age=True): | |||
r1712 | """ | |||
Creates a human readable description based on properties | ||||
of notification object | ||||
""" | ||||
r2443 | #alias | |||
_n = notification | ||||
r2082 | _map = { | |||
Mads Kiilerich
|
r3654 | _n.TYPE_CHANGESET_COMMENT: _('%(user)s commented on changeset at %(when)s'), | ||
_n.TYPE_MESSAGE: _('%(user)s sent message at %(when)s'), | ||||
_n.TYPE_MENTION: _('%(user)s mentioned you at %(when)s'), | ||||
_n.TYPE_REGISTRATION: _('%(user)s registered in RhodeCode at %(when)s'), | ||||
_n.TYPE_PULL_REQUEST: _('%(user)s opened new pull request at %(when)s'), | ||||
_n.TYPE_PULL_REQUEST_COMMENT: _('%(user)s commented on pull request at %(when)s') | ||||
r2082 | } | |||
Mads Kiilerich
|
r3654 | tmpl = _map[notification.type_] | ||
r1731 | ||||
r1717 | if show_age: | |||
when = h.age(notification.created_on) | ||||
else: | ||||
r2445 | when = h.fmt_date(notification.created_on) | |||
r2156 | ||||
Mads Kiilerich
|
r3654 | return tmpl % dict( | ||
r2082 | user=notification.created_by_user.username, | |||
Mads Kiilerich
|
r3654 | when=when, | ||
) | ||||
r1717 | ||||
class EmailNotificationModel(BaseModel): | ||||
r1732 | TYPE_CHANGESET_COMMENT = Notification.TYPE_CHANGESET_COMMENT | |||
Mads Kiilerich
|
r3270 | TYPE_PASSWORD_RESET = 'password_link' | ||
r1732 | TYPE_REGISTRATION = Notification.TYPE_REGISTRATION | |||
r2434 | TYPE_PULL_REQUEST = Notification.TYPE_PULL_REQUEST | |||
r2802 | TYPE_PULL_REQUEST_COMMENT = Notification.TYPE_PULL_REQUEST_COMMENT | |||
r1717 | TYPE_DEFAULT = 'default' | |||
def __init__(self): | ||||
r1723 | self._template_root = rhodecode.CONFIG['pylons.paths']['templates'][0] | |||
self._tmpl_lookup = rhodecode.CONFIG['pylons.app_globals'].mako_lookup | ||||
r1717 | ||||
self.email_types = { | ||||
r2082 | self.TYPE_CHANGESET_COMMENT: 'email_templates/changeset_comment.html', | |||
self.TYPE_PASSWORD_RESET: 'email_templates/password_reset.html', | ||||
self.TYPE_REGISTRATION: 'email_templates/registration.html', | ||||
r2799 | self.TYPE_DEFAULT: 'email_templates/default.html', | |||
self.TYPE_PULL_REQUEST: 'email_templates/pull_request.html', | ||||
r2802 | self.TYPE_PULL_REQUEST_COMMENT: 'email_templates/pull_request_comment.html', | |||
r1717 | } | |||
def get_email_tmpl(self, type_, **kwargs): | ||||
""" | ||||
return generated template for email based on given type | ||||
r1818 | ||||
r1717 | :param type_: | |||
""" | ||||
r1723 | ||||
r1732 | base = self.email_types.get(type_, self.email_types[self.TYPE_DEFAULT]) | |||
r1723 | email_template = self._tmpl_lookup.get_template(base) | |||
r3121 | # translator and helpers inject | |||
_kwargs = {'_': _, | ||||
r3423 | 'h': h, | |||
'c': c} | ||||
r1717 | _kwargs.update(kwargs) | |||
log.debug('rendering tmpl %s with kwargs %s' % (base, _kwargs)) | ||||
return email_template.render(**_kwargs) | ||||