mirror of
https://github.com/zulip/zulip.git
synced 2025-10-24 08:33:43 +00:00
Outgoing Webhook System: Add DoRestCall and helper functions
This commit is contained in:
@@ -893,6 +893,7 @@ def do_send_messages(messages_maybe_none):
|
||||
"message": message_to_dict(message['message'], apply_markdown=False),
|
||||
"trigger": outgoing_webhook_event['trigger'],
|
||||
"user_profile_id": outgoing_webhook_event["user_profile"].id,
|
||||
"failed_tries": 0,
|
||||
},
|
||||
lambda x: None
|
||||
)
|
||||
|
96
zerver/lib/outgoing_webhook.py
Normal file
96
zerver/lib/outgoing_webhook.py
Normal file
@@ -0,0 +1,96 @@
|
||||
from __future__ import absolute_import
|
||||
from typing import Any, Iterable, Dict, Tuple, Callable, Text, Mapping
|
||||
|
||||
import requests
|
||||
import json
|
||||
import sys
|
||||
import inspect
|
||||
import logging
|
||||
from six.moves import urllib
|
||||
from functools import reduce
|
||||
|
||||
from django.utils.translation import ugettext as _
|
||||
|
||||
from zerver.models import Realm, get_realm_by_email_domain, get_user_profile_by_id, get_client
|
||||
from zerver.lib.actions import check_send_message
|
||||
from zerver.lib.queue import queue_json_publish
|
||||
from zerver.lib.validator import check_dict, check_string
|
||||
from zerver.decorator import JsonableError
|
||||
|
||||
MAX_REQUEST_RETRIES = 3
|
||||
|
||||
def send_response_message(bot_id, message, response_message_content):
|
||||
# type: (str, Dict[str, Any], Text) -> None
|
||||
recipient_type_name = message['type']
|
||||
bot_user = get_user_profile_by_id(bot_id)
|
||||
realm = get_realm_by_email_domain(message['sender_email'])
|
||||
|
||||
if recipient_type_name == 'stream':
|
||||
recipients = [message['display_recipient']]
|
||||
check_send_message(bot_user, get_client("OutgoingWebhookResponse"), recipient_type_name, recipients,
|
||||
message['subject'], response_message_content, realm, forwarder_user_profile=bot_user)
|
||||
else:
|
||||
# Private message; only send if the bot is there in the recipients
|
||||
recipients = [recipient['email'] for recipient in message['display_recipient']]
|
||||
if bot_user.email in recipients:
|
||||
check_send_message(bot_user, get_client("OutgoingWebhookResponse"), recipient_type_name, recipients,
|
||||
message['subject'], response_message_content, realm, forwarder_user_profile=bot_user)
|
||||
|
||||
def succeed_with_message(event, success_message):
|
||||
# type: (Dict[str, Any], Text) -> None
|
||||
success_message = "Success! " + success_message
|
||||
send_response_message(event['user_profile_id'], event['message'], success_message)
|
||||
|
||||
def fail_with_message(event, failure_message):
|
||||
# type: (Dict[str, Any], Text) -> None
|
||||
failure_message = "Failure! " + failure_message
|
||||
send_response_message(event['user_profile_id'], event['message'], failure_message)
|
||||
|
||||
def request_retry(event, failure_message):
|
||||
# type: (Dict[str, Any], Text) -> None
|
||||
event['failed_tries'] += 1
|
||||
if event['failed_tries'] > MAX_REQUEST_RETRIES:
|
||||
bot_user = get_user_profile_by_id(event['user_profile_id'])
|
||||
failure_message = "Maximum retries exceeded! " + failure_message
|
||||
fail_with_message(event, failure_message)
|
||||
logging.warning("Maximum retries exceeded for trigger:%s event:%s" % (bot_user.email, event['command']))
|
||||
else:
|
||||
queue_json_publish("outgoing_webhooks", event, lambda x: None)
|
||||
|
||||
def do_rest_call(rest_operation, event, timeout=None):
|
||||
# type: (Dict[str, Any], Dict[str, Any], Any) -> None
|
||||
rest_operation_validator = check_dict([
|
||||
('method', check_string),
|
||||
('relative_url_path', check_string),
|
||||
('request_kwargs', check_dict([])),
|
||||
('base_url', check_string),
|
||||
])
|
||||
|
||||
error = rest_operation_validator('rest_operation', rest_operation)
|
||||
if error:
|
||||
raise JsonableError(_("%s") % (error,))
|
||||
|
||||
http_method = rest_operation['method']
|
||||
final_url = urllib.parse.urljoin(rest_operation['base_url'], rest_operation['relative_url_path'])
|
||||
request_kwargs = rest_operation['request_kwargs']
|
||||
request_kwargs['timeout'] = timeout
|
||||
|
||||
try:
|
||||
response = requests.request(http_method, final_url, data=json.dumps(event), **request_kwargs)
|
||||
if str(response.status_code).startswith('2'):
|
||||
succeed_with_message(event, "received response: `" + str(response.content) + "`.")
|
||||
|
||||
# On 50x errors, try retry
|
||||
elif str(response.status_code).startswith('5'):
|
||||
request_retry(event, "unable to connect with the third party.")
|
||||
else:
|
||||
fail_with_message(event, "unable to communicate with the third party.")
|
||||
|
||||
except requests.exceptions.Timeout:
|
||||
logging.info("Trigger event %s on %s timed out. Retrying" % (event["command"], event['service_name']))
|
||||
request_retry(event, 'unable to connect with the third party.')
|
||||
|
||||
except requests.exceptions.RequestException as e:
|
||||
response_message = "An exception occured for message `%s`! See the logs for more information." % (event["command"],)
|
||||
logging.exception("Outhook trigger failed:\n %s" % (e,))
|
||||
fail_with_message(event, response_message)
|
@@ -1737,9 +1737,9 @@ def get_realm_outgoing_webhook_services_name(realm):
|
||||
return list(Service.objects.filter(user_profile__realm=realm, user_profile__is_bot=True,
|
||||
user_profile__bot_type=UserProfile.OUTGOING_WEBHOOK_BOT).values('name'))
|
||||
|
||||
def get_realm_bot_services(email, realm):
|
||||
# type: (str, Realm) -> List[Any]
|
||||
return list(Service.objects.filter(user_profile__email=email, user_profile__realm=realm).values())
|
||||
def get_bot_services(user_profile_id):
|
||||
# type: (str) -> List[Service]
|
||||
return list(Service.objects.filter(user_profile__id=user_profile_id))
|
||||
|
||||
def get_service_profile(email, realm, service_name):
|
||||
# type: (str, Realm, str) -> Service
|
||||
|
74
zerver/tests/test_outgoing_webhook_system.py
Normal file
74
zerver/tests/test_outgoing_webhook_system.py
Normal file
@@ -0,0 +1,74 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
from __future__ import absolute_import
|
||||
from __future__ import print_function
|
||||
|
||||
import mock
|
||||
from typing import Any
|
||||
|
||||
from zerver.lib.test_helpers import get_user_profile_by_email
|
||||
from zerver.lib.test_classes import ZulipTestCase
|
||||
from zerver.models import Service
|
||||
from zerver.lib.outgoing_webhook import do_rest_call
|
||||
|
||||
import requests
|
||||
|
||||
rest_operation = {'method': "POST",
|
||||
'relative_url_path': "",
|
||||
'request_kwargs': {},
|
||||
'base_url': ""}
|
||||
|
||||
class ResponseMock(object):
|
||||
def __init__(self, status_code, data, content):
|
||||
# type: (int, Any, str) -> None
|
||||
self.status_code = status_code
|
||||
self.data = data
|
||||
self.content = content
|
||||
|
||||
def request_exception_error(http_method, final_url, data, **request_kwargs):
|
||||
# type: (Any, Any, Any, Any) -> Any
|
||||
raise requests.exceptions.RequestException
|
||||
|
||||
def timeout_error(http_method, final_url, data, **request_kwargs):
|
||||
# type: (Any, Any, Any, Any) -> Any
|
||||
raise requests.exceptions.Timeout
|
||||
|
||||
class DoRestCallTests(ZulipTestCase):
|
||||
@mock.patch('zerver.lib.outgoing_webhook.succeed_with_message')
|
||||
def test_successful_request(self, mock_succeed_with_message):
|
||||
# type: (mock.Mock) -> None
|
||||
response = ResponseMock(200, {"message": "testing"}, '')
|
||||
with mock.patch('requests.request', return_value=response):
|
||||
do_rest_call(rest_operation, None, None)
|
||||
self.assertTrue(mock_succeed_with_message.called)
|
||||
|
||||
@mock.patch('zerver.lib.outgoing_webhook.request_retry')
|
||||
def test_retry_request(self, mock_request_retry):
|
||||
# type: (mock.Mock) -> None
|
||||
response = ResponseMock(500, {"message": "testing"}, '')
|
||||
with mock.patch('requests.request', return_value=response):
|
||||
do_rest_call(rest_operation, None, None)
|
||||
self.assertTrue(mock_request_retry.called)
|
||||
|
||||
@mock.patch('zerver.lib.outgoing_webhook.fail_with_message')
|
||||
def test_fail_request(self, mock_fail_with_message):
|
||||
# type: (mock.Mock) -> None
|
||||
response = ResponseMock(400, {"message": "testing"}, '')
|
||||
with mock.patch('requests.request', return_value=response):
|
||||
do_rest_call(rest_operation, None, None)
|
||||
self.assertTrue(mock_fail_with_message.called)
|
||||
|
||||
@mock.patch('logging.info')
|
||||
@mock.patch('requests.request', side_effect=timeout_error)
|
||||
@mock.patch('zerver.lib.outgoing_webhook.request_retry')
|
||||
def test_timeout_request(self, mock_request_retry, mock_requests_request, mock_logger):
|
||||
# type: (mock.Mock, mock.Mock, mock.Mock) -> None
|
||||
do_rest_call(rest_operation, {"command": "", "service_name": ""}, None)
|
||||
self.assertTrue(mock_request_retry.called)
|
||||
|
||||
@mock.patch('logging.exception')
|
||||
@mock.patch('requests.request', side_effect=request_exception_error)
|
||||
@mock.patch('zerver.lib.outgoing_webhook.fail_with_message')
|
||||
def test_request_exception(self, mock_fail_with_message, mock_requests_request, mock_logger):
|
||||
# type: (mock.Mock, mock.Mock, mock.Mock) -> None
|
||||
do_rest_call(rest_operation, {"command": ""}, None)
|
||||
self.assertTrue(mock_fail_with_message.called)
|
Reference in New Issue
Block a user