mirror of
https://github.com/zulip/zulip.git
synced 2025-11-05 06:23:38 +00:00
This add support to event filtering system for most webhooks that require trivial changes to adapt this feature.
325 lines
10 KiB
Python
325 lines
10 KiB
Python
from functools import partial
|
|
from html.parser import HTMLParser
|
|
from typing import Any, Callable, Dict, List, Tuple
|
|
|
|
from django.http import HttpRequest, HttpResponse
|
|
|
|
from zerver.decorator import webhook_view
|
|
from zerver.lib.exceptions import UnsupportedWebhookEventType
|
|
from zerver.lib.request import REQ, has_request_variables
|
|
from zerver.lib.response import json_success
|
|
from zerver.lib.webhooks.common import check_send_webhook_message
|
|
from zerver.models import UserProfile
|
|
|
|
COMPANY_CREATED = """
|
|
New company **{name}** created:
|
|
* **User count**: {user_count}
|
|
* **Monthly spending**: {monthly_spend}
|
|
""".strip()
|
|
|
|
CONTACT_EMAIL_ADDED = "New email {email} added to contact."
|
|
|
|
CONTACT_CREATED = """
|
|
New contact created:
|
|
* **Name (or pseudonym)**: {name}
|
|
* **Email**: {email}
|
|
* **Location**: {location_info}
|
|
""".strip()
|
|
|
|
CONTACT_SIGNED_UP = """
|
|
Contact signed up:
|
|
* **Email**: {email}
|
|
* **Location**: {location_info}
|
|
""".strip()
|
|
|
|
CONTACT_TAG_CREATED = "Contact tagged with the `{name}` tag."
|
|
|
|
CONTACT_TAG_DELETED = "The tag `{name}` was removed from the contact."
|
|
|
|
CONVERSATION_ADMIN_ASSIGNED = "{name} assigned to conversation."
|
|
|
|
CONVERSATION_ADMIN_TEMPLATE = "{admin_name} {action} the conversation."
|
|
|
|
CONVERSATION_ADMIN_REPLY_TEMPLATE = """
|
|
{admin_name} {action} the conversation:
|
|
|
|
``` quote
|
|
{content}
|
|
```
|
|
""".strip()
|
|
|
|
CONVERSATION_ADMIN_INITIATED_CONVERSATION = """
|
|
{admin_name} initiated a conversation:
|
|
|
|
``` quote
|
|
{content}
|
|
```
|
|
""".strip()
|
|
|
|
EVENT_CREATED = "New event **{event_name}** created."
|
|
|
|
USER_CREATED = """
|
|
New user created:
|
|
* **Name**: {name}
|
|
* **Email**: {email}
|
|
""".strip()
|
|
|
|
|
|
class MLStripper(HTMLParser):
|
|
def __init__(self) -> None:
|
|
self.reset()
|
|
self.strict = False
|
|
self.convert_charrefs = True
|
|
self.fed: List[str] = []
|
|
|
|
def handle_data(self, d: str) -> None:
|
|
self.fed.append(d)
|
|
|
|
def get_data(self) -> str:
|
|
return "".join(self.fed)
|
|
|
|
|
|
def strip_tags(html: str) -> str:
|
|
s = MLStripper()
|
|
s.feed(html)
|
|
return s.get_data()
|
|
|
|
|
|
def get_topic_for_contacts(user: Dict[str, Any]) -> str:
|
|
topic = "{type}: {name}".format(
|
|
type=user["type"].capitalize(),
|
|
name=user.get("name") or user.get("pseudonym") or user.get("email"),
|
|
)
|
|
|
|
return topic
|
|
|
|
|
|
def get_company_created_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
body = COMPANY_CREATED.format(**payload["data"]["item"])
|
|
return ("Companies", body)
|
|
|
|
|
|
def get_contact_added_email_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
user = payload["data"]["item"]
|
|
body = CONTACT_EMAIL_ADDED.format(email=user["email"])
|
|
topic = get_topic_for_contacts(user)
|
|
return (topic, body)
|
|
|
|
|
|
def get_contact_created_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
contact = payload["data"]["item"]
|
|
body = CONTACT_CREATED.format(
|
|
name=contact.get("name") or contact.get("pseudonym"),
|
|
email=contact["email"],
|
|
location_info="{city_name}, {region_name}, {country_name}".format(
|
|
**contact["location_data"],
|
|
),
|
|
)
|
|
topic = get_topic_for_contacts(contact)
|
|
return (topic, body)
|
|
|
|
|
|
def get_contact_signed_up_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
contact = payload["data"]["item"]
|
|
body = CONTACT_SIGNED_UP.format(
|
|
email=contact["email"],
|
|
location_info="{city_name}, {region_name}, {country_name}".format(
|
|
**contact["location_data"],
|
|
),
|
|
)
|
|
topic = get_topic_for_contacts(contact)
|
|
return (topic, body)
|
|
|
|
|
|
def get_contact_tag_created_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
body = CONTACT_TAG_CREATED.format(**payload["data"]["item"]["tag"])
|
|
contact = payload["data"]["item"]["contact"]
|
|
topic = get_topic_for_contacts(contact)
|
|
return (topic, body)
|
|
|
|
|
|
def get_contact_tag_deleted_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
body = CONTACT_TAG_DELETED.format(**payload["data"]["item"]["tag"])
|
|
contact = payload["data"]["item"]["contact"]
|
|
topic = get_topic_for_contacts(contact)
|
|
return (topic, body)
|
|
|
|
|
|
def get_conversation_admin_assigned_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
body = CONVERSATION_ADMIN_ASSIGNED.format(**payload["data"]["item"]["assignee"])
|
|
user = payload["data"]["item"]["user"]
|
|
topic = get_topic_for_contacts(user)
|
|
return (topic, body)
|
|
|
|
|
|
def get_conversation_admin_message(
|
|
payload: Dict[str, Any],
|
|
action: str,
|
|
) -> Tuple[str, str]:
|
|
assignee = payload["data"]["item"]["assignee"]
|
|
user = payload["data"]["item"]["user"]
|
|
body = CONVERSATION_ADMIN_TEMPLATE.format(
|
|
admin_name=assignee.get("name"),
|
|
action=action,
|
|
)
|
|
topic = get_topic_for_contacts(user)
|
|
return (topic, body)
|
|
|
|
|
|
def get_conversation_admin_reply_message(
|
|
payload: Dict[str, Any],
|
|
action: str,
|
|
) -> Tuple[str, str]:
|
|
assignee = payload["data"]["item"]["assignee"]
|
|
user = payload["data"]["item"]["user"]
|
|
note = payload["data"]["item"]["conversation_parts"]["conversation_parts"][0]
|
|
content = strip_tags(note["body"])
|
|
body = CONVERSATION_ADMIN_REPLY_TEMPLATE.format(
|
|
admin_name=assignee.get("name"),
|
|
action=action,
|
|
content=content,
|
|
)
|
|
topic = get_topic_for_contacts(user)
|
|
return (topic, body)
|
|
|
|
|
|
def get_conversation_admin_single_created_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
assignee = payload["data"]["item"]["assignee"]
|
|
user = payload["data"]["item"]["user"]
|
|
conversation_body = payload["data"]["item"]["conversation_message"]["body"]
|
|
content = strip_tags(conversation_body)
|
|
body = CONVERSATION_ADMIN_INITIATED_CONVERSATION.format(
|
|
admin_name=assignee.get("name"),
|
|
content=content,
|
|
)
|
|
topic = get_topic_for_contacts(user)
|
|
return (topic, body)
|
|
|
|
|
|
def get_conversation_user_created_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
user = payload["data"]["item"]["user"]
|
|
conversation_body = payload["data"]["item"]["conversation_message"]["body"]
|
|
content = strip_tags(conversation_body)
|
|
body = CONVERSATION_ADMIN_INITIATED_CONVERSATION.format(
|
|
admin_name=user.get("name"),
|
|
content=content,
|
|
)
|
|
topic = get_topic_for_contacts(user)
|
|
return (topic, body)
|
|
|
|
|
|
def get_conversation_user_replied_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
user = payload["data"]["item"]["user"]
|
|
note = payload["data"]["item"]["conversation_parts"]["conversation_parts"][0]
|
|
content = strip_tags(note["body"])
|
|
body = CONVERSATION_ADMIN_REPLY_TEMPLATE.format(
|
|
admin_name=user.get("name"),
|
|
action="replied to",
|
|
content=content,
|
|
)
|
|
topic = get_topic_for_contacts(user)
|
|
return (topic, body)
|
|
|
|
|
|
def get_event_created_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
event = payload["data"]["item"]
|
|
body = EVENT_CREATED.format(**event)
|
|
return ("Events", body)
|
|
|
|
|
|
def get_user_created_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
user = payload["data"]["item"]
|
|
body = USER_CREATED.format(**user)
|
|
topic = get_topic_for_contacts(user)
|
|
return (topic, body)
|
|
|
|
|
|
def get_user_deleted_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
user = payload["data"]["item"]
|
|
topic = get_topic_for_contacts(user)
|
|
return (topic, "User deleted.")
|
|
|
|
|
|
def get_user_email_updated_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
user = payload["data"]["item"]
|
|
body = "User's email was updated to {}.".format(user["email"])
|
|
topic = get_topic_for_contacts(user)
|
|
return (topic, body)
|
|
|
|
|
|
def get_user_tagged_message(
|
|
payload: Dict[str, Any],
|
|
action: str,
|
|
) -> Tuple[str, str]:
|
|
user = payload["data"]["item"]["user"]
|
|
tag = payload["data"]["item"]["tag"]
|
|
topic = get_topic_for_contacts(user)
|
|
body = "The tag `{tag_name}` was {action} the user.".format(
|
|
tag_name=tag["name"],
|
|
action=action,
|
|
)
|
|
return (topic, body)
|
|
|
|
|
|
def get_user_unsubscribed_message(payload: Dict[str, Any]) -> Tuple[str, str]:
|
|
user = payload["data"]["item"]
|
|
body = "User unsubscribed from emails."
|
|
topic = get_topic_for_contacts(user)
|
|
return (topic, body)
|
|
|
|
|
|
EVENT_TO_FUNCTION_MAPPER: Dict[str, Callable[[Dict[str, Any]], Tuple[str, str]]] = {
|
|
"company.created": get_company_created_message,
|
|
"contact.added_email": get_contact_added_email_message,
|
|
"contact.created": get_contact_created_message,
|
|
"contact.signed_up": get_contact_signed_up_message,
|
|
"contact.tag.created": get_contact_tag_created_message,
|
|
"contact.tag.deleted": get_contact_tag_deleted_message,
|
|
"conversation.admin.assigned": get_conversation_admin_assigned_message,
|
|
"conversation.admin.closed": partial(get_conversation_admin_message, action="closed"),
|
|
"conversation.admin.opened": partial(get_conversation_admin_message, action="opened"),
|
|
"conversation.admin.snoozed": partial(get_conversation_admin_message, action="snoozed"),
|
|
"conversation.admin.unsnoozed": partial(get_conversation_admin_message, action="unsnoozed"),
|
|
"conversation.admin.replied": partial(
|
|
get_conversation_admin_reply_message, action="replied to"
|
|
),
|
|
"conversation.admin.noted": partial(
|
|
get_conversation_admin_reply_message, action="added a note to"
|
|
),
|
|
"conversation.admin.single.created": get_conversation_admin_single_created_message,
|
|
"conversation.user.created": get_conversation_user_created_message,
|
|
"conversation.user.replied": get_conversation_user_replied_message,
|
|
"event.created": get_event_created_message,
|
|
"user.created": get_user_created_message,
|
|
"user.deleted": get_user_deleted_message,
|
|
"user.email.updated": get_user_email_updated_message,
|
|
"user.tag.created": partial(get_user_tagged_message, action="added to"),
|
|
"user.tag.deleted": partial(get_user_tagged_message, action="removed from"),
|
|
"user.unsubscribed": get_user_unsubscribed_message,
|
|
# Note that we do not have a payload for visitor.signed_up
|
|
# but it should be identical to contact.signed_up
|
|
"visitor.signed_up": get_contact_signed_up_message,
|
|
}
|
|
|
|
ALL_EVENT_TYPES = list(EVENT_TO_FUNCTION_MAPPER.keys())
|
|
|
|
|
|
@webhook_view("Intercom", all_event_types=ALL_EVENT_TYPES)
|
|
@has_request_variables
|
|
def api_intercom_webhook(
|
|
request: HttpRequest,
|
|
user_profile: UserProfile,
|
|
payload: Dict[str, Any] = REQ(argument_type="body"),
|
|
) -> HttpResponse:
|
|
event_type = payload["topic"]
|
|
if event_type == "ping":
|
|
return json_success()
|
|
|
|
handler = EVENT_TO_FUNCTION_MAPPER.get(event_type)
|
|
if handler is None:
|
|
raise UnsupportedWebhookEventType(event_type)
|
|
topic, body = handler(payload)
|
|
|
|
check_send_webhook_message(request, user_profile, topic, body, event_type)
|
|
return json_success()
|