2016-03-13 12:38:57 +01:00
|
|
|
# Webhooks for external integrations.
|
|
|
|
|
|
|
|
from __future__ import absolute_import
|
2016-06-05 19:29:21 +02:00
|
|
|
from django.http import HttpRequest, HttpResponse
|
|
|
|
from zerver.models import get_client, UserProfile
|
2016-03-13 12:38:57 +01:00
|
|
|
from zerver.lib.actions import check_send_message
|
|
|
|
from zerver.lib.response import json_success
|
|
|
|
from zerver.lib.validator import check_dict
|
|
|
|
from zerver.decorator import REQ, has_request_variables, authenticated_rest_api_view
|
|
|
|
|
|
|
|
import base64
|
|
|
|
from functools import wraps
|
|
|
|
|
2016-11-23 20:15:23 +01:00
|
|
|
from zerver.webhooks.github.view import build_message_from_gitlog
|
2016-03-13 12:38:57 +01:00
|
|
|
|
2017-04-22 02:34:53 +02:00
|
|
|
from typing import Any, Callable, Dict, TypeVar, Optional, Text
|
2016-07-05 05:48:40 +02:00
|
|
|
from zerver.lib.str_utils import force_str, force_bytes
|
2016-03-13 12:38:57 +01:00
|
|
|
|
2016-07-22 15:10:19 +02:00
|
|
|
ViewFuncT = TypeVar('ViewFuncT', bound=Callable[..., HttpResponse])
|
2016-03-13 12:38:57 +01:00
|
|
|
|
|
|
|
# Beanstalk's web hook UI rejects url with a @ in the username section of a url
|
|
|
|
# So we ask the user to replace them with %40
|
|
|
|
# We manually fix the username here before passing it along to @authenticated_rest_api_view
|
|
|
|
def beanstalk_decoder(view_func):
|
2016-07-22 15:10:19 +02:00
|
|
|
# type: (ViewFuncT) -> ViewFuncT
|
2016-03-13 12:38:57 +01:00
|
|
|
@wraps(view_func)
|
|
|
|
def _wrapped_view_func(request, *args, **kwargs):
|
2016-06-05 19:29:21 +02:00
|
|
|
# type: (HttpRequest, *Any, **Any) -> HttpResponse
|
2016-03-13 12:38:57 +01:00
|
|
|
try:
|
2017-05-07 20:05:35 +02:00
|
|
|
auth_type, encoded_value = request.META['HTTP_AUTHORIZATION'].split() # type: str, str
|
2016-03-13 12:38:57 +01:00
|
|
|
if auth_type.lower() == "basic":
|
2016-07-05 05:48:40 +02:00
|
|
|
email, api_key = base64.b64decode(force_bytes(encoded_value)).decode('utf-8').split(":")
|
2016-03-13 12:38:57 +01:00
|
|
|
email = email.replace('%40', '@')
|
2016-07-05 05:48:40 +02:00
|
|
|
credentials = u"%s:%s" % (email, api_key)
|
2016-07-17 19:43:36 +02:00
|
|
|
encoded_credentials = force_str(base64.b64encode(credentials.encode('utf-8')))
|
|
|
|
request.META['HTTP_AUTHORIZATION'] = "Basic " + encoded_credentials
|
2016-07-05 05:48:40 +02:00
|
|
|
except Exception:
|
2016-03-13 12:38:57 +01:00
|
|
|
pass
|
|
|
|
|
|
|
|
return view_func(request, *args, **kwargs)
|
|
|
|
|
2017-05-07 20:05:35 +02:00
|
|
|
return _wrapped_view_func # type: ignore # https://github.com/python/mypy/issues/1927
|
2016-03-13 12:38:57 +01:00
|
|
|
|
|
|
|
@beanstalk_decoder
|
2016-05-18 20:35:35 +02:00
|
|
|
@authenticated_rest_api_view(is_webhook=True)
|
2016-03-13 12:38:57 +01:00
|
|
|
@has_request_variables
|
|
|
|
def api_beanstalk_webhook(request, user_profile,
|
2017-04-22 02:34:53 +02:00
|
|
|
payload=REQ(validator=check_dict([])),
|
|
|
|
branches=REQ(default=None)):
|
|
|
|
# type: (HttpRequest, UserProfile, Dict[str, Any], Optional[Text]) -> HttpResponse
|
2016-03-13 12:38:57 +01:00
|
|
|
# Beanstalk supports both SVN and git repositories
|
|
|
|
# We distinguish between the two by checking for a
|
|
|
|
# 'uri' key that is only present for git repos
|
|
|
|
git_repo = 'uri' in payload
|
|
|
|
if git_repo:
|
2017-04-22 02:34:53 +02:00
|
|
|
if branches is not None and branches.find(payload['branch']) == -1:
|
|
|
|
return json_success()
|
2016-03-13 12:38:57 +01:00
|
|
|
# To get a linkable url,
|
2017-04-05 09:12:19 +02:00
|
|
|
for commit in payload['commits']:
|
2017-04-25 22:34:07 +02:00
|
|
|
commit['author'] = {'username': commit['author']['name']}
|
2017-04-05 09:12:19 +02:00
|
|
|
|
2016-03-13 12:38:57 +01:00
|
|
|
subject, content = build_message_from_gitlog(user_profile, payload['repository']['name'],
|
|
|
|
payload['ref'], payload['commits'],
|
|
|
|
payload['before'], payload['after'],
|
|
|
|
payload['repository']['url'],
|
|
|
|
payload['pusher_name'])
|
|
|
|
else:
|
|
|
|
author = payload.get('author_full_name')
|
|
|
|
url = payload.get('changeset_url')
|
|
|
|
revision = payload.get('revision')
|
|
|
|
(short_commit_msg, _, _) = payload.get('message').partition("\n")
|
|
|
|
|
|
|
|
subject = "svn r%s" % (revision,)
|
|
|
|
content = "%s pushed [revision %s](%s):\n\n> %s" % (author, revision, url, short_commit_msg)
|
|
|
|
|
|
|
|
check_send_message(user_profile, get_client("ZulipBeanstalkWebhook"), "stream",
|
|
|
|
["commits"], subject, content)
|
|
|
|
return json_success()
|