2017-04-05 09:12:19 +02:00
|
|
|
from typing import Optional, Any, Dict, List, Text, Tuple
|
|
|
|
from collections import defaultdict
|
2016-10-05 19:21:47 +02:00
|
|
|
SUBJECT_WITH_BRANCH_TEMPLATE = u'{repo} / {branch}'
|
2016-10-19 16:00:40 +02:00
|
|
|
SUBJECT_WITH_PR_OR_ISSUE_INFO_TEMPLATE = u'{repo} / {type} #{id} {title}'
|
2016-10-05 19:21:47 +02:00
|
|
|
|
2016-10-06 17:13:22 +02:00
|
|
|
EMPTY_SHA = '0000000000000000000000000000000000000000'
|
|
|
|
|
2017-03-09 21:25:21 +01:00
|
|
|
COMMITS_LIMIT = 20
|
2017-03-15 10:15:03 +01:00
|
|
|
COMMIT_ROW_TEMPLATE = u'* {commit_msg} ([{commit_short_sha}]({commit_url}))\n'
|
2016-10-12 23:27:24 +02:00
|
|
|
COMMITS_MORE_THAN_LIMIT_TEMPLATE = u"[and {commits_number} more commit(s)]"
|
2017-04-05 09:12:19 +02:00
|
|
|
COMMIT_OR_COMMITS = u"commit{}"
|
2016-10-12 23:27:24 +02:00
|
|
|
|
2017-04-05 09:12:19 +02:00
|
|
|
PUSH_PUSHED_TEXT_WITH_URL = u"[pushed]({compare_url}) {number_of_commits} {commit_or_commits}"
|
|
|
|
PUSH_PUSHED_TEXT_WITHOUT_URL = u"pushed {number_of_commits} {commit_or_commits}"
|
2017-04-26 02:57:47 +02:00
|
|
|
PUSH_COMMITS_MESSAGE_TEMPLATE_WITH_COMMITTERS = u"""{user_name} {pushed_text} to branch {branch_name}. {committers_details}.
|
|
|
|
|
|
|
|
{commits_data}
|
|
|
|
"""
|
|
|
|
PUSH_COMMITS_MESSAGE_TEMPLATE_WITHOUT_COMMITTERS = u"""{user_name} {pushed_text} to branch {branch_name}.
|
2016-09-26 16:32:35 +02:00
|
|
|
|
2016-10-12 23:38:49 +02:00
|
|
|
{commits_data}
|
2016-09-26 16:32:35 +02:00
|
|
|
"""
|
2017-05-16 04:38:25 +02:00
|
|
|
PUSH_DELETE_BRANCH_MESSAGE_TEMPLATE = u"{user_name} [deleted]({compare_url}) the branch {branch_name}."
|
2017-05-17 02:19:33 +02:00
|
|
|
PUSH_LOCAL_BRANCH_WITHOUT_COMMITS_MESSAGE_TEMPLATE = u"{user_name} [pushed]({compare_url}) the branch {branch_name}."
|
2017-04-05 09:12:19 +02:00
|
|
|
PUSH_COMMITS_MESSAGE_EXTENSION = u"Commits by {}"
|
|
|
|
PUSH_COMMITTERS_LIMIT_INFO = 3
|
2016-09-26 16:32:35 +02:00
|
|
|
|
2016-10-05 20:36:17 +02:00
|
|
|
FORCE_PUSH_COMMITS_MESSAGE_TEMPLATE = u"{user_name} [force pushed]({url}) to branch {branch_name}. Head is now {head}"
|
2017-03-16 08:28:13 +01:00
|
|
|
CREATE_BRANCH_MESSAGE_TEMPLATE = u"{user_name} created [{branch_name}]({url}) branch"
|
2016-10-06 15:26:47 +02:00
|
|
|
REMOVE_BRANCH_MESSAGE_TEMPLATE = u"{user_name} deleted branch {branch_name}"
|
2016-10-05 20:36:17 +02:00
|
|
|
|
2016-10-26 21:13:00 +02:00
|
|
|
PULL_REQUEST_OR_ISSUE_MESSAGE_TEMPLATE = u"{user_name} {action} [{type}{id}]({url})"
|
2016-10-19 15:57:57 +02:00
|
|
|
PULL_REQUEST_OR_ISSUE_ASSIGNEE_INFO_TEMPLATE = u"(assigned to {assignee})"
|
2016-10-11 18:53:20 +02:00
|
|
|
PULL_REQUEST_BRANCH_INFO_TEMPLATE = u"\nfrom `{target}` to `{base}`"
|
2016-10-27 21:43:15 +02:00
|
|
|
|
2017-03-17 00:15:57 +01:00
|
|
|
SETUP_MESSAGE_TEMPLATE = u"{integration} webhook has been successfully configured"
|
|
|
|
SETUP_MESSAGE_USER_PART = u" by {user_name}"
|
|
|
|
|
2016-10-27 21:43:15 +02:00
|
|
|
CONTENT_MESSAGE_TEMPLATE = u"\n~~~ quote\n{message}\n~~~"
|
|
|
|
|
|
|
|
COMMITS_COMMENT_MESSAGE_TEMPLATE = u"{user_name} {action} on [{sha}]({url})"
|
2016-10-11 18:53:20 +02:00
|
|
|
|
2016-11-14 20:32:37 +01:00
|
|
|
PUSH_TAGS_MESSAGE_TEMPLATE = u"""{user_name} {action} tag {tag}"""
|
2016-11-08 22:12:22 +01:00
|
|
|
TAG_WITH_URL_TEMPLATE = u"[{tag_name}]({tag_url})"
|
|
|
|
TAG_WITHOUT_URL_TEMPLATE = u"{tag_name}"
|
|
|
|
|
2017-04-05 09:12:19 +02:00
|
|
|
|
2017-05-16 04:38:25 +02:00
|
|
|
def get_push_commits_event_message(user_name, compare_url, branch_name, commits_data, is_truncated=False, deleted=False):
|
|
|
|
# type: (Text, Optional[Text], Text, List[Dict[str, Any]], Optional[bool], Optional[bool]) -> Text
|
|
|
|
if not commits_data and deleted:
|
|
|
|
return PUSH_DELETE_BRANCH_MESSAGE_TEMPLATE.format(
|
|
|
|
user_name=user_name,
|
|
|
|
compare_url=compare_url,
|
|
|
|
branch_name=branch_name
|
|
|
|
)
|
|
|
|
|
2017-05-17 02:19:33 +02:00
|
|
|
if not commits_data and not deleted:
|
|
|
|
return PUSH_LOCAL_BRANCH_WITHOUT_COMMITS_MESSAGE_TEMPLATE.format(
|
|
|
|
user_name=user_name,
|
|
|
|
compare_url=compare_url,
|
|
|
|
branch_name=branch_name
|
|
|
|
)
|
|
|
|
|
2017-04-05 09:12:19 +02:00
|
|
|
pushed_message_template = PUSH_PUSHED_TEXT_WITH_URL if compare_url else PUSH_PUSHED_TEXT_WITHOUT_URL
|
|
|
|
|
|
|
|
pushed_text_message = pushed_message_template.format(
|
|
|
|
compare_url=compare_url,
|
|
|
|
number_of_commits=len(commits_data),
|
|
|
|
commit_or_commits=COMMIT_OR_COMMITS.format(u's' if len(commits_data) > 1 else u''))
|
2016-09-26 16:32:35 +02:00
|
|
|
|
2017-04-26 02:57:47 +02:00
|
|
|
committers_items = get_all_committers(commits_data) # type: List[Tuple[str, int]]
|
|
|
|
if len(committers_items) == 1 and user_name == committers_items[0][0]:
|
|
|
|
return PUSH_COMMITS_MESSAGE_TEMPLATE_WITHOUT_COMMITTERS.format(
|
|
|
|
user_name=user_name,
|
|
|
|
pushed_text=pushed_text_message,
|
|
|
|
branch_name=branch_name,
|
|
|
|
commits_data=get_commits_content(commits_data, is_truncated),
|
|
|
|
).rstrip()
|
|
|
|
else:
|
|
|
|
committers_details = "{} ({})".format(*committers_items[0])
|
|
|
|
|
|
|
|
for name, number_of_commits in committers_items[1:-1]:
|
|
|
|
committers_details = "{}, {} ({})".format(committers_details, name, number_of_commits)
|
|
|
|
|
|
|
|
if len(committers_items) > 1:
|
|
|
|
committers_details = "{} and {} ({})".format(committers_details, *committers_items[-1])
|
|
|
|
|
|
|
|
return PUSH_COMMITS_MESSAGE_TEMPLATE_WITH_COMMITTERS.format(
|
|
|
|
user_name=user_name,
|
|
|
|
pushed_text=pushed_text_message,
|
|
|
|
branch_name=branch_name,
|
|
|
|
committers_details=PUSH_COMMITS_MESSAGE_EXTENSION.format(committers_details),
|
|
|
|
commits_data=get_commits_content(commits_data, is_truncated),
|
|
|
|
).rstrip()
|
2016-10-05 20:36:17 +02:00
|
|
|
|
|
|
|
def get_force_push_commits_event_message(user_name, url, branch_name, head):
|
2016-12-05 07:05:33 +01:00
|
|
|
# type: (Text, Text, Text, Text) -> Text
|
2016-10-05 20:36:17 +02:00
|
|
|
return FORCE_PUSH_COMMITS_MESSAGE_TEMPLATE.format(
|
|
|
|
user_name=user_name,
|
|
|
|
url=url,
|
|
|
|
branch_name=branch_name,
|
|
|
|
head=head
|
|
|
|
)
|
2016-10-06 15:26:47 +02:00
|
|
|
|
2017-03-16 08:28:13 +01:00
|
|
|
def get_create_branch_event_message(user_name, url, branch_name):
|
|
|
|
# type: (Text, Text, Text) -> Text
|
|
|
|
return CREATE_BRANCH_MESSAGE_TEMPLATE.format(
|
|
|
|
user_name=user_name,
|
|
|
|
url=url,
|
|
|
|
branch_name=branch_name,
|
|
|
|
)
|
|
|
|
|
2016-10-06 15:26:47 +02:00
|
|
|
def get_remove_branch_event_message(user_name, branch_name):
|
2016-12-05 07:05:33 +01:00
|
|
|
# type: (Text, Text) -> Text
|
2016-10-06 15:26:47 +02:00
|
|
|
return REMOVE_BRANCH_MESSAGE_TEMPLATE.format(
|
|
|
|
user_name=user_name,
|
|
|
|
branch_name=branch_name,
|
|
|
|
)
|
2016-10-11 18:53:20 +02:00
|
|
|
|
|
|
|
def get_pull_request_event_message(
|
2016-10-26 21:13:00 +02:00
|
|
|
user_name, action, url, number=None,
|
2016-10-11 18:53:20 +02:00
|
|
|
target_branch=None, base_branch=None,
|
2016-10-19 15:57:57 +02:00
|
|
|
message=None, assignee=None, type='PR'
|
2016-10-11 18:53:20 +02:00
|
|
|
):
|
2016-12-05 07:05:33 +01:00
|
|
|
# type: (Text, Text, Text, Optional[int], Optional[Text], Optional[Text], Optional[Text], Optional[Text], Optional[Text]) -> Text
|
2016-10-19 15:57:57 +02:00
|
|
|
main_message = PULL_REQUEST_OR_ISSUE_MESSAGE_TEMPLATE.format(
|
2016-10-11 18:53:20 +02:00
|
|
|
user_name=user_name,
|
|
|
|
action=action,
|
|
|
|
type=type,
|
2016-10-26 21:13:00 +02:00
|
|
|
url=url,
|
|
|
|
id=" #{}".format(number) if number is not None else ''
|
2016-10-11 18:53:20 +02:00
|
|
|
)
|
|
|
|
if assignee:
|
2016-10-19 15:57:57 +02:00
|
|
|
main_message += PULL_REQUEST_OR_ISSUE_ASSIGNEE_INFO_TEMPLATE.format(assignee=assignee)
|
2016-10-11 18:53:20 +02:00
|
|
|
|
|
|
|
if target_branch and base_branch:
|
|
|
|
main_message += PULL_REQUEST_BRANCH_INFO_TEMPLATE.format(
|
|
|
|
target=target_branch,
|
|
|
|
base=base_branch
|
|
|
|
)
|
2016-10-19 15:57:57 +02:00
|
|
|
if message:
|
2016-10-27 21:43:15 +02:00
|
|
|
main_message += '\n' + CONTENT_MESSAGE_TEMPLATE.format(message=message)
|
2016-10-11 18:53:20 +02:00
|
|
|
return main_message.rstrip()
|
2016-10-12 23:38:49 +02:00
|
|
|
|
2017-03-17 00:15:57 +01:00
|
|
|
def get_setup_webhook_message(integration, user_name=None):
|
|
|
|
# type: (Text, Optional[Text]) -> Text
|
|
|
|
content = SETUP_MESSAGE_TEMPLATE.format(integration=integration)
|
|
|
|
if user_name:
|
|
|
|
content += SETUP_MESSAGE_USER_PART.format(user_name=user_name)
|
|
|
|
return content
|
|
|
|
|
2016-10-26 21:13:00 +02:00
|
|
|
def get_issue_event_message(user_name, action, url, number=None, message=None, assignee=None):
|
2016-12-05 07:05:33 +01:00
|
|
|
# type: (Text, Text, Text, Optional[int], Optional[Text], Optional[Text]) -> Text
|
2016-10-19 15:57:57 +02:00
|
|
|
return get_pull_request_event_message(
|
|
|
|
user_name,
|
|
|
|
action,
|
|
|
|
url,
|
2016-10-26 21:13:00 +02:00
|
|
|
number,
|
2016-10-19 15:57:57 +02:00
|
|
|
message=message,
|
|
|
|
assignee=assignee,
|
|
|
|
type='Issue'
|
|
|
|
)
|
|
|
|
|
2016-11-08 22:12:22 +01:00
|
|
|
def get_push_tag_event_message(user_name, tag_name, tag_url=None, action='pushed'):
|
2016-12-05 07:05:33 +01:00
|
|
|
# type: (Text, Text, Optional[Text], Optional[Text]) -> Text
|
2016-11-08 22:12:22 +01:00
|
|
|
if tag_url:
|
|
|
|
tag_part = TAG_WITH_URL_TEMPLATE.format(tag_name=tag_name, tag_url=tag_url)
|
|
|
|
else:
|
|
|
|
tag_part = TAG_WITHOUT_URL_TEMPLATE.format(tag_name=tag_name)
|
|
|
|
return PUSH_TAGS_MESSAGE_TEMPLATE.format(
|
|
|
|
user_name=user_name,
|
|
|
|
action=action,
|
|
|
|
tag=tag_part
|
|
|
|
)
|
|
|
|
|
2016-10-27 21:43:15 +02:00
|
|
|
def get_commits_comment_action_message(user_name, action, commit_url, sha, message=None):
|
2016-12-05 07:05:33 +01:00
|
|
|
# type: (Text, Text, Text, Text, Optional[Text]) -> Text
|
2016-10-27 21:43:15 +02:00
|
|
|
content = COMMITS_COMMENT_MESSAGE_TEMPLATE.format(
|
|
|
|
user_name=user_name,
|
|
|
|
action=action,
|
|
|
|
sha=get_short_sha(sha),
|
|
|
|
url=commit_url
|
|
|
|
)
|
|
|
|
if message is not None:
|
|
|
|
content += CONTENT_MESSAGE_TEMPLATE.format(
|
|
|
|
message=message
|
|
|
|
)
|
|
|
|
return content
|
|
|
|
|
2016-11-09 18:50:10 +01:00
|
|
|
def get_commits_content(commits_data, is_truncated=False):
|
2016-12-05 07:05:33 +01:00
|
|
|
# type: (List[Dict[str, Any]], Optional[bool]) -> Text
|
2016-10-12 23:38:49 +02:00
|
|
|
commits_content = u''
|
|
|
|
for commit in commits_data[:COMMITS_LIMIT]:
|
|
|
|
commits_content += COMMIT_ROW_TEMPLATE.format(
|
2017-05-24 21:28:26 +02:00
|
|
|
commit_short_sha=get_short_sha(commit['sha']),
|
2016-10-12 23:38:49 +02:00
|
|
|
commit_url=commit.get('url'),
|
2017-05-24 21:28:26 +02:00
|
|
|
commit_msg=commit['message'].partition('\n')[0]
|
2016-10-12 23:38:49 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
if len(commits_data) > COMMITS_LIMIT:
|
|
|
|
commits_content += COMMITS_MORE_THAN_LIMIT_TEMPLATE.format(
|
2016-11-09 18:50:10 +01:00
|
|
|
commits_number=len(commits_data) - COMMITS_LIMIT
|
|
|
|
)
|
|
|
|
elif is_truncated:
|
|
|
|
commits_content += COMMITS_MORE_THAN_LIMIT_TEMPLATE.format(
|
|
|
|
commits_number=''
|
|
|
|
).replace(' ', ' ')
|
2016-10-12 23:38:49 +02:00
|
|
|
return commits_content.rstrip()
|
2016-10-27 21:43:15 +02:00
|
|
|
|
|
|
|
def get_short_sha(sha):
|
2016-12-05 07:05:33 +01:00
|
|
|
# type: (Text) -> Text
|
2016-10-27 21:43:15 +02:00
|
|
|
return sha[:7]
|
2017-04-05 09:12:19 +02:00
|
|
|
|
|
|
|
def get_all_committers(commits_data):
|
|
|
|
# type: (List[Dict[str, Any]]) -> List[Tuple[str, int]]
|
|
|
|
committers = defaultdict(int) # type: Dict[str, int]
|
|
|
|
|
|
|
|
for commit in commits_data:
|
|
|
|
committers[commit['name']] += 1
|
|
|
|
|
2017-04-21 20:44:37 +02:00
|
|
|
# Sort by commit count, breaking ties alphabetically.
|
|
|
|
committers_items = sorted(list(committers.items()),
|
|
|
|
key=lambda item: (-item[1], item[0])) # type: List[Tuple[str, int]]
|
2017-04-05 09:12:19 +02:00
|
|
|
committers_values = [c_i[1] for c_i in committers_items] # type: List[int]
|
|
|
|
|
|
|
|
if len(committers) > PUSH_COMMITTERS_LIMIT_INFO:
|
|
|
|
others_number_of_commits = sum(committers_values[PUSH_COMMITTERS_LIMIT_INFO:])
|
|
|
|
committers_items = committers_items[:PUSH_COMMITTERS_LIMIT_INFO]
|
|
|
|
committers_items.append(('others', others_number_of_commits))
|
|
|
|
|
|
|
|
return committers_items
|