notification.py
257 lines
| 9.1 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 | ||||
r1717 | import datetime | |||
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 | ||||
r2433 | from sqlalchemy.orm import joinedload | |||
r1702 | ||||
r1712 | log = logging.getLogger(__name__) | |||
r1702 | ||||
r1716 | ||||
r1702 | class NotificationModel(BaseModel): | |||
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, | ||||
email_kwargs={}): | ||||
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 | ||||
r1703 | """ | |||
r1722 | from rhodecode.lib.celerylib import tasks, run_task | |||
r1702 | ||||
r1731 | if recipients and not getattr(recipients, '__iter__', False): | |||
r1702 | raise Exception('recipients must be a list of iterable') | |||
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 | ||||
r1731 | if with_email is False: | |||
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: | ||||
r1717 | email_subject = NotificationModel().make_description(notif, False) | |||
r1731 | type_ = type_ | |||
r1717 | email_body = body | |||
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() | ||||
r1713 | self.sa.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: | ||||
:type 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_: | ||||
q = q.filter(Notification.type_ == filter_.get('type')) | ||||
r1702 | ||||
r2433 | return q.all() | |||
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_: | ||||
q = q.filter(Notification.type_ == filter_.get('type')) | ||||
# this is a little inefficient but sqlalchemy doesn't support | ||||
# update on joined tables :( | ||||
for obj in q.all(): | ||||
obj.read = True | ||||
self.sa.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 = { | |||
r2443 | _n.TYPE_CHANGESET_COMMENT: _('commented on commit'), | |||
_n.TYPE_MESSAGE: _('sent message'), | ||||
_n.TYPE_MENTION: _('mentioned you'), | ||||
_n.TYPE_REGISTRATION: _('registered in RhodeCode'), | ||||
_n.TYPE_PULL_REQUEST: _('opened new pull request'), | ||||
_n.TYPE_PULL_REQUEST_COMMENT: _('commented on pull request') | ||||
r2082 | } | |||
r1731 | ||||
r2445 | # action == _map string | |||
tmpl = "%(user)s %(action)s at %(when)s" | ||||
r1717 | if show_age: | |||
when = h.age(notification.created_on) | ||||
else: | ||||
r2445 | when = h.fmt_date(notification.created_on) | |||
r2156 | ||||
r2082 | data = dict( | |||
user=notification.created_by_user.username, | ||||
action=_map[notification.type_], when=when, | ||||
) | ||||
r1712 | return tmpl % data | |||
r1717 | ||||
class EmailNotificationModel(BaseModel): | ||||
r1732 | TYPE_CHANGESET_COMMENT = Notification.TYPE_CHANGESET_COMMENT | |||
r1717 | TYPE_PASSWORD_RESET = 'passoword_link' | |||
r1732 | TYPE_REGISTRATION = Notification.TYPE_REGISTRATION | |||
r2434 | TYPE_PULL_REQUEST = Notification.TYPE_PULL_REQUEST | |||
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', | ||||
self.TYPE_DEFAULT: 'email_templates/default.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) | |||
r1717 | # translator inject | |||
r2082 | _kwargs = {'_': _} | |||
r1717 | _kwargs.update(kwargs) | |||
log.debug('rendering tmpl %s with kwargs %s' % (base, _kwargs)) | ||||
return email_template.render(**_kwargs) | ||||