2018-11-06 17:07:04 +01:00
|
|
|
from urllib.parse import unquote
|
|
|
|
|
2018-04-24 20:22:38 +02:00
|
|
|
from django.conf import settings
|
2018-03-13 23:36:11 +01:00
|
|
|
from django.http import HttpRequest
|
2018-04-24 20:22:38 +02:00
|
|
|
from django.utils.translation import ugettext as _
|
2018-05-11 01:40:23 +02:00
|
|
|
from typing import Optional
|
2018-03-13 23:36:11 +01:00
|
|
|
|
|
|
|
from zerver.lib.actions import check_send_stream_message, \
|
2018-04-24 20:22:38 +02:00
|
|
|
check_send_private_message, send_rate_limited_pm_notification_to_bot_owner
|
|
|
|
from zerver.lib.exceptions import StreamDoesNotExistError, JsonableError, \
|
|
|
|
ErrorCode
|
2018-03-13 23:36:11 +01:00
|
|
|
from zerver.lib.request import REQ, has_request_variables
|
2018-04-24 20:22:38 +02:00
|
|
|
from zerver.lib.send_email import FromAddress
|
|
|
|
from zerver.models import UserProfile, get_system_bot
|
2018-03-13 23:36:11 +01:00
|
|
|
|
2018-03-22 21:43:28 +01:00
|
|
|
|
2018-04-24 20:22:38 +02:00
|
|
|
MISSING_EVENT_HEADER_MESSAGE = """
|
|
|
|
Hi there! Your bot {bot_name} just sent an HTTP request to {request_path} that
|
|
|
|
is missing the HTTP {header_name} header. Because this header is how
|
|
|
|
{integration_name} indicates the event type, this usually indicates a configuration
|
|
|
|
issue, where you either entered the URL for a different integration, or are running
|
|
|
|
an older version of the third-party service that doesn't provide that header.
|
|
|
|
Contact {support_email} if you need help debugging!
|
|
|
|
"""
|
|
|
|
|
2018-11-15 05:31:34 +01:00
|
|
|
INVALID_JSON_MESSAGE = """
|
|
|
|
Hi there! It looks like you tried to setup the Zulip {webhook_name} integration,
|
|
|
|
but didn't correctly configure the webhook to send data in the JSON format
|
|
|
|
that this integration expects!
|
|
|
|
"""
|
|
|
|
|
2018-04-24 20:22:38 +02:00
|
|
|
# Django prefixes all custom HTTP headers with `HTTP_`
|
|
|
|
DJANGO_HTTP_PREFIX = "HTTP_"
|
|
|
|
|
2018-11-15 05:31:34 +01:00
|
|
|
def notify_bot_owner_about_invalid_json(user_profile: UserProfile,
|
|
|
|
webhook_client_name: str) -> None:
|
|
|
|
send_rate_limited_pm_notification_to_bot_owner(
|
|
|
|
user_profile, user_profile.realm,
|
|
|
|
INVALID_JSON_MESSAGE.format(webhook_name=webhook_client_name).strip()
|
|
|
|
)
|
|
|
|
|
2018-05-22 16:45:21 +02:00
|
|
|
class UnexpectedWebhookEventType(JsonableError):
|
|
|
|
code = ErrorCode.UNEXPECTED_WEBHOOK_EVENT_TYPE
|
|
|
|
data_fields = ['webhook_name', 'event_type']
|
|
|
|
|
|
|
|
def __init__(self, webhook_name: str, event_type: Optional[str]) -> None:
|
|
|
|
self.webhook_name = webhook_name
|
|
|
|
self.event_type = event_type
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def msg_format() -> str:
|
|
|
|
return _("The '{event_type}' event isn't currently supported by the {webhook_name} webhook")
|
|
|
|
|
2018-04-24 20:22:38 +02:00
|
|
|
class MissingHTTPEventHeader(JsonableError):
|
|
|
|
code = ErrorCode.MISSING_HTTP_EVENT_HEADER
|
|
|
|
data_fields = ['header']
|
|
|
|
|
2018-05-11 01:40:23 +02:00
|
|
|
def __init__(self, header: str) -> None:
|
2018-04-24 20:22:38 +02:00
|
|
|
self.header = header
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def msg_format() -> str:
|
|
|
|
return _("Missing the HTTP event header '{header}'")
|
|
|
|
|
2018-03-13 23:36:11 +01:00
|
|
|
@has_request_variables
|
|
|
|
def check_send_webhook_message(
|
|
|
|
request: HttpRequest, user_profile: UserProfile,
|
2018-05-11 01:40:23 +02:00
|
|
|
topic: str, body: str, stream: Optional[str]=REQ(default=None),
|
2018-11-06 17:07:04 +01:00
|
|
|
user_specified_topic: Optional[str]=REQ("topic", default=None),
|
2018-12-04 01:59:39 +01:00
|
|
|
unquote_url_parameters: Optional[bool]=False
|
2018-03-13 23:36:11 +01:00
|
|
|
) -> None:
|
|
|
|
|
|
|
|
if stream is None:
|
|
|
|
assert user_profile.bot_owner is not None
|
|
|
|
check_send_private_message(user_profile, request.client,
|
|
|
|
user_profile.bot_owner, body)
|
|
|
|
else:
|
2018-11-06 17:07:04 +01:00
|
|
|
# Some third-party websites (such as Atlassian's JIRA), tend to
|
|
|
|
# double escape their URLs in a manner that escaped space characters
|
|
|
|
# (%20) are never properly decoded. We work around that by making sure
|
2018-12-04 01:59:39 +01:00
|
|
|
# that the URL parameters are decoded on our end.
|
|
|
|
if unquote_url_parameters:
|
2018-11-06 17:07:04 +01:00
|
|
|
stream = unquote(stream)
|
|
|
|
|
2018-03-13 23:36:11 +01:00
|
|
|
if user_specified_topic is not None:
|
|
|
|
topic = user_specified_topic
|
2018-12-04 01:59:39 +01:00
|
|
|
if unquote_url_parameters:
|
|
|
|
topic = unquote(topic)
|
2018-03-22 21:43:28 +01:00
|
|
|
|
|
|
|
try:
|
|
|
|
check_send_stream_message(user_profile, request.client,
|
|
|
|
stream, topic, body)
|
|
|
|
except StreamDoesNotExistError:
|
|
|
|
# A PM will be sent to the bot_owner by check_message, notifying
|
|
|
|
# that the webhook bot just tried to send a message to a non-existent
|
|
|
|
# stream, so we don't need to re-raise it since it clutters up
|
|
|
|
# webhook-errors.log
|
|
|
|
pass
|
2018-04-24 20:22:38 +02:00
|
|
|
|
2018-05-11 01:40:23 +02:00
|
|
|
def validate_extract_webhook_http_header(request: HttpRequest, header: str,
|
|
|
|
integration_name: str) -> str:
|
2018-04-24 20:22:38 +02:00
|
|
|
extracted_header = request.META.get(DJANGO_HTTP_PREFIX + header)
|
|
|
|
if extracted_header is None:
|
|
|
|
message_body = MISSING_EVENT_HEADER_MESSAGE.format(
|
|
|
|
bot_name=request.user.full_name,
|
|
|
|
request_path=request.path,
|
|
|
|
header_name=header,
|
|
|
|
integration_name=integration_name,
|
|
|
|
support_email=FromAddress.SUPPORT,
|
|
|
|
)
|
|
|
|
send_rate_limited_pm_notification_to_bot_owner(
|
|
|
|
request.user, request.user.realm, message_body)
|
|
|
|
|
|
|
|
raise MissingHTTPEventHeader(header)
|
|
|
|
|
|
|
|
return extracted_header
|