2016-04-28 15:46:00 +02:00
|
|
|
"""
|
|
|
|
Taiga integration for Zulip.
|
|
|
|
|
|
|
|
Tips for notification output:
|
|
|
|
|
|
|
|
*Emojis*: most of the events have specific emojis e.g.
|
|
|
|
- :notebook: - change of subject/name/description
|
|
|
|
- :chart_with_upwards_trend: - change of status
|
|
|
|
etc. If no there's no meaningful emoji for certain event, the defaults are used:
|
|
|
|
- :thought_balloon: - event connected to commenting
|
|
|
|
- :busts_in_silhouette: - event connected to a certain user
|
|
|
|
- :package: - all other events connected to user story
|
|
|
|
- :calendar: - all other events connected to milestones
|
|
|
|
- :clipboard: - all other events connected to tasks
|
|
|
|
- :bulb: - all other events connected to issues
|
|
|
|
|
|
|
|
*Text formatting*: if there has been a change of a property, the new value should always be in bold; otherwise the
|
|
|
|
subject of US/task should be in bold.
|
|
|
|
"""
|
|
|
|
|
|
|
|
from __future__ import absolute_import
|
2016-11-23 04:53:54 +01:00
|
|
|
from typing import Any, Mapping, Optional, Tuple, Text
|
2016-05-25 15:02:02 +02:00
|
|
|
|
|
|
|
from django.utils.translation import ugettext as _
|
2016-06-06 01:56:35 +02:00
|
|
|
from django.http import HttpRequest, HttpResponse
|
2016-05-25 15:02:02 +02:00
|
|
|
|
2016-04-28 15:46:00 +02:00
|
|
|
from zerver.lib.actions import check_send_message
|
|
|
|
from zerver.lib.response import json_success, json_error
|
|
|
|
from zerver.decorator import REQ, has_request_variables, api_key_only_webhook_view
|
2016-06-06 01:56:35 +02:00
|
|
|
from zerver.models import UserProfile, Client
|
2016-04-28 15:46:00 +02:00
|
|
|
|
|
|
|
import ujson
|
|
|
|
from six.moves import range
|
|
|
|
|
|
|
|
|
2016-05-12 22:49:36 +02:00
|
|
|
@api_key_only_webhook_view('Taiga')
|
2016-04-28 15:46:00 +02:00
|
|
|
@has_request_variables
|
2016-05-12 22:49:36 +02:00
|
|
|
def api_taiga_webhook(request, user_profile, client, message=REQ(argument_type='body'),
|
2016-05-06 21:19:34 +02:00
|
|
|
stream=REQ(default='taiga'), topic=REQ(default='General')):
|
2016-11-23 04:53:54 +01:00
|
|
|
# type: (HttpRequest, UserProfile, Client, Dict[str, Any], Text, Text) -> HttpResponse
|
2016-04-28 15:46:00 +02:00
|
|
|
parsed_events = parse_message(message)
|
|
|
|
|
2016-07-13 21:52:39 +02:00
|
|
|
content_lines = []
|
2016-04-28 15:46:00 +02:00
|
|
|
for event in parsed_events:
|
2016-07-13 21:52:39 +02:00
|
|
|
content_lines.append(generate_content(event) + '\n')
|
|
|
|
content = "".join(sorted(content_lines))
|
2016-04-28 15:46:00 +02:00
|
|
|
|
2016-05-12 22:49:36 +02:00
|
|
|
check_send_message(user_profile, client, 'stream', [stream], topic, content)
|
2016-04-28 15:46:00 +02:00
|
|
|
|
|
|
|
return json_success()
|
|
|
|
|
|
|
|
templates = {
|
|
|
|
'userstory': {
|
|
|
|
'create': u':package: %(user)s created user story **%(subject)s**.',
|
|
|
|
'set_assigned_to': u':busts_in_silhouette: %(user)s assigned user story **%(subject)s** to %(new)s.',
|
2016-07-17 19:53:11 +02:00
|
|
|
'changed_assigned_to': u':busts_in_silhouette: %(user)s reassigned user story **%(subject)s**'
|
2016-12-02 08:15:05 +01:00
|
|
|
' from %(old)s to %(new)s.',
|
2016-04-28 15:46:00 +02:00
|
|
|
'points': u':game_die: %(user)s changed estimation of user story **%(subject)s**.',
|
|
|
|
'blocked': u':lock: %(user)s blocked user story **%(subject)s**.',
|
|
|
|
'unblocked': u':unlock: %(user)s unblocked user story **%(subject)s**.',
|
|
|
|
'set_milestone': u':calendar: %(user)s added user story **%(subject)s** to sprint %(new)s.',
|
2016-07-17 19:53:11 +02:00
|
|
|
'changed_milestone': u':calendar: %(user)s changed sprint of user story **%(subject)s** from %(old)s'
|
2016-12-02 08:15:05 +01:00
|
|
|
' to %(new)s.',
|
2016-07-17 19:53:11 +02:00
|
|
|
'changed_status': u':chart_with_upwards_trend: %(user)s changed status of user story **%(subject)s**'
|
2016-12-02 08:15:05 +01:00
|
|
|
' from %(old)s to %(new)s.',
|
2016-04-28 15:46:00 +02:00
|
|
|
'closed': u':checkered_flag: %(user)s closed user story **%(subject)s**.',
|
|
|
|
'reopened': u':package: %(user)s reopened user story **%(subject)s**.',
|
|
|
|
'renamed': u':notebook: %(user)s renamed user story from %(old)s to **%(new)s**.',
|
|
|
|
'description': u':notebook: %(user)s updated description of user story **%(subject)s**.',
|
|
|
|
'commented': u':thought_balloon: %(user)s commented on user story **%(subject)s**.',
|
|
|
|
'delete': u':x: %(user)s deleted user story **%(subject)s**.'
|
|
|
|
},
|
|
|
|
'milestone': {
|
|
|
|
'create': u':calendar: %(user)s created sprint **%(subject)s**.',
|
|
|
|
'renamed': u':notebook: %(user)s renamed sprint from %(old)s to **%(new)s**.',
|
2016-07-17 19:53:11 +02:00
|
|
|
'estimated_start': u':calendar: %(user)s changed estimated start of sprint **%(subject)s**'
|
2016-12-02 08:15:05 +01:00
|
|
|
' from %(old)s to %(new)s.',
|
2016-07-17 19:53:11 +02:00
|
|
|
'estimated_finish': u':calendar: %(user)s changed estimated finish of sprint **%(subject)s**'
|
2016-12-02 08:15:05 +01:00
|
|
|
' from %(old)s to %(new)s.',
|
2016-04-28 15:46:00 +02:00
|
|
|
'delete': u':x: %(user)s deleted sprint **%(subject)s**.'
|
|
|
|
},
|
|
|
|
'task': {
|
|
|
|
'create': u':clipboard: %(user)s created task **%(subject)s**.',
|
|
|
|
'set_assigned_to': u':busts_in_silhouette: %(user)s assigned task **%(subject)s** to %(new)s.',
|
2016-07-17 19:53:11 +02:00
|
|
|
'changed_assigned_to': u':busts_in_silhouette: %(user)s reassigned task **%(subject)s**'
|
2016-12-02 08:15:05 +01:00
|
|
|
' from %(old)s to %(new)s.',
|
2016-04-28 15:46:00 +02:00
|
|
|
'blocked': u':lock: %(user)s blocked task **%(subject)s**.',
|
|
|
|
'unblocked': u':unlock: %(user)s unblocked task **%(subject)s**.',
|
|
|
|
'set_milestone': u':calendar: %(user)s added task **%(subject)s** to sprint %(new)s.',
|
|
|
|
'changed_milestone': u':calendar: %(user)s changed sprint of task **%(subject)s** from %(old)s to %(new)s.',
|
2016-07-17 19:53:11 +02:00
|
|
|
'changed_status': u':chart_with_upwards_trend: %(user)s changed status of task **%(subject)s**'
|
2016-12-02 08:15:05 +01:00
|
|
|
' from %(old)s to %(new)s.',
|
2016-04-28 15:46:00 +02:00
|
|
|
'renamed': u':notebook: %(user)s renamed task %(old)s to **%(new)s**.',
|
|
|
|
'description': u':notebook: %(user)s updated description of task **%(subject)s**.',
|
|
|
|
'commented': u':thought_balloon: %(user)s commented on task **%(subject)s**.',
|
|
|
|
'delete': u':x: %(user)s deleted task **%(subject)s**.',
|
|
|
|
'changed_us': u':clipboard: %(user)s moved task **%(subject)s** from user story %(old)s to %(new)s.'
|
|
|
|
},
|
|
|
|
'issue': {
|
|
|
|
'create': u':bulb: %(user)s created issue **%(subject)s**.',
|
|
|
|
'set_assigned_to': u':busts_in_silhouette: %(user)s assigned issue **%(subject)s** to %(new)s.', #
|
2016-07-17 19:53:11 +02:00
|
|
|
'changed_assigned_to': u':busts_in_silhouette: %(user)s reassigned issue **%(subject)s**'
|
2016-12-02 08:15:05 +01:00
|
|
|
' from %(old)s to %(new)s.',
|
2016-04-28 15:46:00 +02:00
|
|
|
'changed_priority': u':rocket: %(user)s changed priority of issue **%(subject)s** from %(old)s to %(new)s.',
|
|
|
|
'changed_severity': u':warning: %(user)s changed severity of issue **%(subject)s** from %(old)s to %(new)s.',
|
2016-07-17 19:53:11 +02:00
|
|
|
'changed_status': u':chart_with_upwards_trend: %(user)s changed status of issue **%(subject)s**'
|
|
|
|
' from %(old)s to %(new)s.',
|
2016-04-28 15:46:00 +02:00
|
|
|
'changed_type': u':bulb: %(user)s changed type of issue **%(subject)s** from %(old)s to %(new)s.',
|
|
|
|
'renamed': u':notebook: %(user)s renamed issue %(old)s to **%(new)s**.',
|
|
|
|
'description': u':notebook: %(user)s updated description of issue **%(subject)s**.',
|
|
|
|
'commented': u':thought_balloon: %(user)s commented on issue **%(subject)s**.',
|
|
|
|
'delete': u':x: %(user)s deleted issue **%(subject)s**.'
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def get_old_and_new_values(change_type, message):
|
2016-06-06 01:56:35 +02:00
|
|
|
# type: (str, Mapping[str, Any]) -> Tuple[Optional[Dict[str, Any]], Optional[Dict[str, Any]]]
|
2016-04-28 15:46:00 +02:00
|
|
|
""" Parses the payload and finds previous and current value of change_type."""
|
|
|
|
values_map = {
|
|
|
|
'assigned_to': 'users',
|
|
|
|
'status': 'status',
|
|
|
|
'severity': 'severity',
|
|
|
|
'priority': 'priority',
|
|
|
|
'milestone': 'milestone',
|
|
|
|
'type': 'type',
|
|
|
|
'user_story': 'user_story'
|
|
|
|
}
|
|
|
|
|
|
|
|
if change_type in ['subject', 'name', 'estimated_finish', 'estimated_start']:
|
|
|
|
old = message["change"]["diff"][change_type]["from"]
|
|
|
|
new = message["change"]["diff"][change_type]["to"]
|
|
|
|
return old, new
|
|
|
|
|
|
|
|
try:
|
|
|
|
old_id = message["change"]["diff"][change_type]["from"]
|
|
|
|
old = message["change"]["values"][values_map[change_type]][str(old_id)]
|
|
|
|
except KeyError:
|
|
|
|
old = None
|
|
|
|
|
|
|
|
try:
|
|
|
|
new_id = message["change"]["diff"][change_type]["to"]
|
|
|
|
new = message["change"]["values"][values_map[change_type]][str(new_id)]
|
|
|
|
except KeyError:
|
|
|
|
new = None
|
|
|
|
|
|
|
|
return old, new
|
|
|
|
|
|
|
|
|
|
|
|
def parse_comment(message):
|
2016-06-06 01:56:35 +02:00
|
|
|
# type: (Mapping[str, Any]) -> Dict[str, Any]
|
2016-04-28 15:46:00 +02:00
|
|
|
""" Parses the comment to issue, task or US. """
|
|
|
|
return {
|
|
|
|
'event': 'commented',
|
|
|
|
'type': message["type"],
|
|
|
|
'values': {
|
|
|
|
'user': message["change"]["user"]["name"],
|
2016-07-17 19:53:11 +02:00
|
|
|
'subject': message["data"]["subject"] if "subject" in list(message["data"].keys()) else
|
2016-12-02 08:15:05 +01:00
|
|
|
(message["data"]["name"])
|
2016-04-28 15:46:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
def parse_create_or_delete(message):
|
2016-06-06 01:56:35 +02:00
|
|
|
# type: (Mapping[str, Any]) -> Dict[str, Any]
|
2016-04-28 15:46:00 +02:00
|
|
|
""" Parses create or delete event. """
|
|
|
|
return {
|
|
|
|
'type': message["type"],
|
|
|
|
'event': message["action"],
|
|
|
|
'values':
|
|
|
|
{
|
|
|
|
'user': message["data"]["owner"]["name"],
|
2016-07-17 19:53:11 +02:00
|
|
|
'subject': message["data"]["subject"] if "subject" in list(message["data"].keys()) else
|
2016-12-02 08:15:05 +01:00
|
|
|
(message["data"]["name"])
|
2016-04-28 15:46:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def parse_change_event(change_type, message):
|
2016-06-06 01:56:35 +02:00
|
|
|
# type: (str, Mapping[str, Any]) -> Dict[str, Any]
|
2016-04-28 15:46:00 +02:00
|
|
|
""" Parses change event. """
|
|
|
|
evt = {}
|
|
|
|
values = {
|
|
|
|
'user': message["change"]["user"]["name"],
|
|
|
|
'subject': message["data"]["subject"] if "subject" in list(message["data"].keys()) else message["data"]["name"]
|
|
|
|
}
|
|
|
|
|
|
|
|
if change_type in ["description", "points"]:
|
|
|
|
event_type = change_type
|
|
|
|
|
|
|
|
elif change_type in ["milestone", "assigned_to"]:
|
|
|
|
old, new = get_old_and_new_values(change_type, message)
|
|
|
|
if not old:
|
|
|
|
event_type = "set_" + change_type
|
|
|
|
values["new"] = new
|
|
|
|
else:
|
|
|
|
event_type = "changed_" + change_type
|
|
|
|
values.update({'old': old, 'new': new})
|
|
|
|
|
|
|
|
elif change_type == "is_blocked":
|
|
|
|
if message["change"]["diff"]["is_blocked"]["to"]:
|
|
|
|
event_type = "blocked"
|
|
|
|
else:
|
|
|
|
event_type = "unblocked"
|
|
|
|
|
|
|
|
elif change_type == "is_closed":
|
|
|
|
if message["change"]["diff"]["is_closed"]["to"]:
|
|
|
|
event_type = "closed"
|
|
|
|
else:
|
|
|
|
event_type = "reopened"
|
|
|
|
|
|
|
|
elif change_type == "user_story":
|
|
|
|
old, new = get_old_and_new_values(change_type, message)
|
|
|
|
event_type = "changed_us"
|
|
|
|
values.update({'old': old, 'new': new})
|
|
|
|
|
|
|
|
elif change_type in ["subject", 'name']:
|
|
|
|
event_type = 'renamed'
|
|
|
|
old, new = get_old_and_new_values(change_type, message)
|
|
|
|
values.update({'old': old, 'new': new})
|
|
|
|
|
|
|
|
elif change_type in ["estimated_finish", "estimated_start"]:
|
|
|
|
old, new = get_old_and_new_values(change_type, message)
|
|
|
|
if not old == new:
|
|
|
|
event_type = change_type
|
|
|
|
values.update({'old': old, 'new': new})
|
|
|
|
else:
|
|
|
|
# date hasn't changed
|
|
|
|
return None
|
|
|
|
|
|
|
|
elif change_type in ["priority", "severity", "type", "status"]:
|
|
|
|
event_type = 'changed_' + change_type
|
|
|
|
old, new = get_old_and_new_values(change_type, message)
|
|
|
|
values.update({'old': old, 'new': new})
|
|
|
|
|
|
|
|
else:
|
|
|
|
# we are not supporting this type of event
|
|
|
|
return None
|
|
|
|
|
|
|
|
evt.update({"type": message["type"], "event": event_type, "values": values})
|
|
|
|
return evt
|
|
|
|
|
|
|
|
|
|
|
|
def parse_message(message):
|
2016-06-06 01:56:35 +02:00
|
|
|
# type: (Mapping[str, Any]) -> List[Dict[str, Any]]
|
2016-04-28 15:46:00 +02:00
|
|
|
""" Parses the payload by delegating to specialized functions. """
|
|
|
|
events = []
|
|
|
|
if message["action"] in ['create', 'delete']:
|
|
|
|
events.append(parse_create_or_delete(message))
|
|
|
|
elif message["action"] == 'change':
|
|
|
|
if message["change"]["diff"]:
|
|
|
|
for value in message["change"]["diff"]:
|
|
|
|
parsed_event = parse_change_event(value, message)
|
2016-11-30 21:45:02 +01:00
|
|
|
if parsed_event:
|
|
|
|
events.append(parsed_event)
|
2016-04-28 15:46:00 +02:00
|
|
|
if message["change"]["comment"]:
|
|
|
|
events.append(parse_comment(message))
|
|
|
|
|
|
|
|
return events
|
|
|
|
|
|
|
|
def generate_content(data):
|
2016-06-06 01:56:35 +02:00
|
|
|
# type: (Mapping[str, Any]) -> str
|
2016-04-28 15:46:00 +02:00
|
|
|
""" Gets the template string and formats it with parsed data. """
|
|
|
|
try:
|
|
|
|
return templates[data['type']][data['event']] % data['values']
|
|
|
|
except KeyError:
|
2016-05-25 15:02:02 +02:00
|
|
|
return json_error(_("Unknown message"))
|