2016-04-14 23:39:37 +02:00
|
|
|
import os
|
2017-11-16 00:43:27 +01:00
|
|
|
import ujson
|
2019-03-09 11:01:20 +01:00
|
|
|
from typing import Union, Dict
|
|
|
|
|
2016-04-14 23:39:37 +02:00
|
|
|
from django.conf import settings
|
2017-11-02 10:19:24 +01:00
|
|
|
from django.core.management.base import CommandParser
|
2017-11-16 00:43:27 +01:00
|
|
|
from django.test import Client
|
2016-04-14 23:39:37 +02:00
|
|
|
|
2019-05-03 23:20:39 +02:00
|
|
|
from zerver.lib.management import ZulipBaseCommand, CommandError
|
2017-10-09 05:12:55 +02:00
|
|
|
from zerver.models import get_realm
|
2016-04-14 23:39:37 +02:00
|
|
|
|
2019-05-16 08:24:23 +02:00
|
|
|
def parse_headers(custom_headers: Union[None, str]) -> Union[None, Dict[str, str]]:
|
|
|
|
""" The main aim of this method is be to convert regular HTTP headers into a format that
|
|
|
|
Django prefers. Note: This function throws a ValueError and thus it should be used in a
|
|
|
|
try/except block. """
|
|
|
|
headers = {}
|
|
|
|
if not custom_headers:
|
|
|
|
return None
|
|
|
|
custom_headers_dict = ujson.loads(custom_headers)
|
|
|
|
for header in custom_headers_dict:
|
|
|
|
if len(header.split(" ")) > 1:
|
|
|
|
raise ValueError("custom header '%s' contains a space." % (header,))
|
2019-06-08 16:46:56 +02:00
|
|
|
new_header = header.upper().replace("-", "_")
|
|
|
|
|
|
|
|
if new_header not in ["CONTENT_TYPE", "CONTENT_LENGTH"]:
|
|
|
|
# See https://docs.djangoproject.com/en/2.2/ref/request-response/
|
|
|
|
# for how Django formats HTTP headers.
|
|
|
|
new_header = "HTTP_" + new_header
|
|
|
|
headers[new_header] = str(custom_headers_dict[header])
|
2019-05-16 08:24:23 +02:00
|
|
|
return headers
|
|
|
|
|
2017-10-09 05:12:55 +02:00
|
|
|
class Command(ZulipBaseCommand):
|
2016-04-14 23:39:37 +02:00
|
|
|
help = """
|
|
|
|
Create webhook message based on given fixture
|
|
|
|
Example:
|
|
|
|
./manage.py send_webhook_fixture_message \
|
2017-10-09 05:12:55 +02:00
|
|
|
[--realm=zulip] \
|
|
|
|
--fixture=zerver/webhooks/integration/fixtures/name.json \
|
2016-04-14 23:39:37 +02:00
|
|
|
'--url=/api/v1/external/integration?stream=stream_name&api_key=api_key'
|
|
|
|
|
2019-03-09 11:01:20 +01:00
|
|
|
To pass custom headers along with the webhook message use the --custom-headers
|
|
|
|
command line option.
|
|
|
|
Example:
|
|
|
|
--custom-headers='{"X-Custom-Header": "value"}'
|
|
|
|
|
|
|
|
The format is a JSON dictionary, so make sure that the header names do
|
|
|
|
not contain any spaces in them and that you use the precise quoting
|
|
|
|
approach shown above.
|
2016-04-14 23:39:37 +02:00
|
|
|
"""
|
|
|
|
|
2017-10-26 11:35:57 +02:00
|
|
|
def add_arguments(self, parser: CommandParser) -> None:
|
2016-11-03 10:22:19 +01:00
|
|
|
parser.add_argument('-f', '--fixture',
|
|
|
|
dest='fixture',
|
|
|
|
type=str,
|
|
|
|
help='The path to the fixture you\'d like to send '
|
|
|
|
'into Zulip')
|
|
|
|
|
|
|
|
parser.add_argument('-u', '--url',
|
|
|
|
dest='url',
|
|
|
|
type=str,
|
|
|
|
help='The url on your Zulip server that you want '
|
|
|
|
'to post the fixture to')
|
2016-04-14 23:39:37 +02:00
|
|
|
|
2019-03-09 11:01:20 +01:00
|
|
|
parser.add_argument('-H', '--custom-headers',
|
|
|
|
dest='custom-headers',
|
|
|
|
type=str,
|
|
|
|
help='The headers you want to provide along with '
|
|
|
|
'your mock request to Zulip.')
|
|
|
|
|
2017-10-09 05:12:55 +02:00
|
|
|
self.add_realm_args(parser, help="Specify which realm/subdomain to connect to; default is zulip")
|
|
|
|
|
2019-03-09 11:01:20 +01:00
|
|
|
def parse_headers(self, custom_headers: Union[None, str]) -> Union[None, Dict[str, str]]:
|
|
|
|
try:
|
2019-05-16 08:24:23 +02:00
|
|
|
return parse_headers(custom_headers)
|
2019-03-09 11:01:20 +01:00
|
|
|
except ValueError as ve:
|
2019-05-03 23:20:39 +02:00
|
|
|
raise CommandError('Encountered an error while attempting to parse custom headers: {}\n'
|
|
|
|
'Note: all strings must be enclosed within "" instead of \'\''.format(ve))
|
2019-03-09 11:01:20 +01:00
|
|
|
|
2017-10-26 11:35:57 +02:00
|
|
|
def handle(self, **options: str) -> None:
|
2016-04-14 23:39:37 +02:00
|
|
|
if options['fixture'] is None or options['url'] is None:
|
2016-11-22 01:44:16 +01:00
|
|
|
self.print_help('./manage.py', 'send_webhook_fixture_message')
|
2019-05-03 23:20:39 +02:00
|
|
|
raise CommandError
|
2016-04-14 23:39:37 +02:00
|
|
|
|
|
|
|
full_fixture_path = os.path.join(settings.DEPLOY_ROOT, options['fixture'])
|
|
|
|
|
|
|
|
if not self._does_fixture_path_exist(full_fixture_path):
|
2019-05-03 23:20:39 +02:00
|
|
|
raise CommandError('Fixture {} does not exist'.format(options['fixture']))
|
2016-04-14 23:39:37 +02:00
|
|
|
|
2019-03-09 11:01:20 +01:00
|
|
|
headers = self.parse_headers(options['custom-headers'])
|
2016-04-14 23:39:37 +02:00
|
|
|
json = self._get_fixture_as_json(full_fixture_path)
|
2017-10-09 05:12:55 +02:00
|
|
|
realm = self.get_realm(options)
|
|
|
|
if realm is None:
|
|
|
|
realm = get_realm("zulip")
|
|
|
|
|
2016-04-14 23:39:37 +02:00
|
|
|
client = Client()
|
2019-03-09 11:01:20 +01:00
|
|
|
if headers:
|
|
|
|
result = client.post(options['url'], json, content_type="application/json",
|
|
|
|
HTTP_HOST=realm.host, **headers)
|
|
|
|
else:
|
|
|
|
result = client.post(options['url'], json, content_type="application/json",
|
|
|
|
HTTP_HOST=realm.host)
|
2017-10-20 18:02:32 +02:00
|
|
|
if result.status_code != 200:
|
2019-05-03 23:20:39 +02:00
|
|
|
raise CommandError('Error status %s: %s' % (result.status_code, result.content))
|
2016-04-14 23:39:37 +02:00
|
|
|
|
2017-10-26 11:35:57 +02:00
|
|
|
def _does_fixture_path_exist(self, fixture_path: str) -> bool:
|
2016-04-14 23:39:37 +02:00
|
|
|
return os.path.exists(fixture_path)
|
|
|
|
|
2017-10-26 11:35:57 +02:00
|
|
|
def _get_fixture_as_json(self, fixture_path: str) -> str:
|
2016-04-14 23:39:37 +02:00
|
|
|
return ujson.dumps(ujson.loads(open(fixture_path).read()))
|