bitbot-3.11-fork/modules/github.py

346 lines
14 KiB
Python
Raw Normal View History

import itertools, json, urllib.parse
from src import ModuleManager, utils
FORM_ENCODED = "application/x-www-form-urlencoded"
COMMIT_URL = "https://github.com/%s/commit/%s"
COMMIT_RANGE_URL = "https://github.com/%s/compare/%s...%s"
CREATE_URL = "https://github.com/%s/tree/%s"
API_ISSUE_URL = "https://api.github.com/repos/%s/%s/issues/%s"
API_PULL_URL = "https://api.github.com/repos/%s/%s/pulls/%s"
2018-11-17 22:32:44 +00:00
DEFAULT_EVENTS = [
"push",
"commit_comment",
"pull_request",
"pull_request_review",
"pull_request_review_comment",
"issue_comment",
"issues",
"create",
"delete",
2018-12-10 18:47:54 +00:00
"release",
"fork"
2018-11-17 22:32:44 +00:00
]
COMMENT_ACTIONS = {
"created": "commented",
"edited": "edited a comment",
"deleted": "deleted a comment"
}
@utils.export("channelset", {"setting": "github-hook",
"help": ("Disable/Enable showing BitBot's github commits in the "
"current channel"), "array": True})
@utils.export("channelset", {"setting": "github-hide-prefix",
"help": "Hide/show command-like prefix on Github hook outputs",
"validate": utils.bool_or_none})
@utils.export("channelset", {"setting": "github-default-repo",
"help": "Set the default github repo for the current channel"})
class Module(ModuleManager.BaseModule):
def _parse_ref(self, channel, ref):
repo, _, number = event["args_split"][0].partition("#")
if not repo:
repo = channel.get_setting("github-default-repo", None)
username, _, repository = repo.partition("/")
if not username or not repository or not number:
raise utils.EventError("Please provide username/repo#number")
if not number.isdigit():
raise utils.EventError("Issue number must be a number")
return username, repository, number
@utils.hook("received.command.ghissue", min_args=1)
def github_issue(self, event):
username, repository, number = self._parse_ref(
event["target"], event["args_split"][0])
page = utils.http.request(
API_ISSUE_URL % (username, repository, number),
json=True)
if page:
labels = [label["name"] for label in page.data["labels"]]
url = self._short_url(page.data["html_url"])
event["stdout"].write("(%s/%s issue#%s) %s [%s] %s" % (
username, repository, number, page.data["title"],
", ".join(labels), url))
@utils.hook("received.command.ghpull", min_args=1)
def github_pull(self, event):
username, repository, number = self._parse_ref(
event["target"], event["args_split"][0])
page = utils.http.request(
API_PULL_URL % (username, repository, number),
json=True)
if page:
repo_from = page.data["head"]["label"]
2019-01-09 22:50:21 +00:00
repo_to = page.data["base"]["label"]
added = self._added(page.data["additions"])
removed = self._removed(page.data["deletions"])
url = self._short_url(page.data["html_url"])
event["stdout"].write(
"(%s/%s pull#%s) [%s/%s] %s%s - %s %s" % (
username, repository, number, added, removed,
repo_from, repo_to, page.data["title"], url))
@utils.hook("api.post.github")
def webhook(self, event):
payload = event["data"].decode("utf8")
if event["headers"]["Content-Type"] == FORM_ENCODED:
payload = urllib.parse.unquote(urllib.parse.parse_qs(payload)[
"payload"][0])
data = json.loads(payload)
github_event = event["headers"]["X-GitHub-Event"]
2018-11-06 16:08:02 +00:00
if github_event == "ping":
return ""
2018-11-06 16:08:02 +00:00
full_name = data["repository"]["full_name"]
repo_username, repo_name = full_name.split("/", 1)
hooks = self.bot.database.channel_settings.find_by_setting(
"github-hook")
2018-11-17 22:32:44 +00:00
targets = []
repo_hooked = False
2018-11-17 22:32:44 +00:00
for i, (server_id, channel_name, hooked_repos) in list(
2018-11-06 14:01:30 +00:00
enumerate(hooks))[::-1]:
if repo_username in hooked_repos or full_name in hooked_repos:
repo_hooked = True
2018-11-17 22:32:44 +00:00
server = self.bot.get_server(server_id)
if server and channel_name in server.channels:
2018-11-17 22:34:51 +00:00
channel = server.channels.get(channel_name)
2018-11-17 22:32:44 +00:00
github_events = channel.get_setting("github-events",
DEFAULT_EVENTS)
if github_event in github_events:
targets.append([server, channel])
2018-11-17 22:32:44 +00:00
if not targets:
return "" if repo_hooked else None
outputs = None
if github_event == "push":
outputs = self.push(event, full_name, data)
elif github_event == "commit_comment":
outputs = self.commit_comment(event, full_name, data)
elif github_event == "pull_request":
outputs = self.pull_request(event, full_name, data)
elif github_event == "pull_request_review":
outputs = self.pull_request_review(event, full_name, data)
elif github_event == "pull_request_review_comment":
outputs = self.pull_request_review_comment(event, full_name, data)
elif github_event == "issue_comment":
outputs = self.issue_comment(event, full_name, data)
elif github_event == "issues":
outputs = self.issues(event, full_name, data)
elif github_event == "create":
outputs = self.create(event, full_name, data)
2018-11-17 21:10:03 +00:00
elif github_event == "delete":
outputs = self.delete(event, full_name, data)
elif github_event == "release":
outputs = self.release(event, full_name, data)
elif github_event == "status":
outputs = self.status(event, full_name, data)
2018-12-10 18:47:54 +00:00
elif github_event == "fork":
outputs = self.fork(event, full_name, data)
if outputs:
for server, channel in targets:
for output in outputs:
output = "(%s) %s" % (full_name, output)
self.events.on("send.stdout").call(target=channel,
2018-11-27 15:07:22 +00:00
module_name="Github", server=server, message=output,
hide_prefix=channel.get_setting(
"github-hide-prefix", False))
return ""
2018-12-11 22:27:04 +00:00
def _short_url(self, url):
try:
page = utils.http.request("https://git.io", method="POST",
post_data={"url": url})
return page.headers["Location"]
except utils.http.HTTPTimeoutException:
return url
2018-12-11 22:27:04 +00:00
def _change_count(self, n, symbol, color):
return utils.irc.color("%s%d" % (symbol, n), color)+utils.irc.bold("")
def _added(self, n):
return self._change_count(n, "+", utils.consts.GREEN)
def _removed(self, n):
return self._change_count(n, "-", utils.consts.RED)
def _modified(self, n):
return self._change_count(n, "~", utils.consts.PURPLE)
def _short_hash(self, hash):
return hash[:8]
def _flat_unique(self, commits, key):
return set(itertools.chain(*(commit[key] for commit in commits)))
def push(self, event, full_name, data):
outputs = []
branch = data["ref"].split("/", 2)[2]
branch = utils.irc.color(branch, utils.consts.LIGHTBLUE)
if len(data["commits"]) <= 3:
for commit in data["commits"]:
id = self._short_hash(commit["id"])
2018-11-17 20:33:23 +00:00
message = commit["message"].split("\n")[0].strip()
author = commit["author"]["name"] or commit["author"]["login"]
author = utils.irc.bold(author)
2018-12-11 22:27:04 +00:00
url = self._short_url(COMMIT_URL % (full_name, id))
added = self._added(len(commit["added"]))
removed = self._removed(len(commit["removed"]))
2018-11-17 08:22:39 +00:00
modified = self._modified(len(commit["modified"]))
outputs.append("[%s/%s/%s files] commit by %s to %s: %s - %s"
% (added, removed, modified, author, branch, message, url))
else:
first_id = self._short_hash(data["before"])
last_id = self._short_hash(data["commits"][-1]["id"])
2018-11-17 09:40:44 +00:00
pusher = utils.irc.bold(data["pusher"]["name"])
2018-12-11 22:27:04 +00:00
url = self._short_url(
COMMIT_RANGE_URL % (full_name, first_id, last_id))
commits = data["commits"]
added = self._added(len(self._flat_unique(commits, "added")))
removed = self._removed(len(self._flat_unique(commits, "removed")))
modified = self._modified(len(self._flat_unique(commits,
"modified")))
outputs.append("[%s/%s/%s files] %s pushed %d commits to %s - %s"
% (added, removed, modified, pusher, len(data["commits"]),
branch, url))
return outputs
def commit_comment(self, event, full_name, data):
action = data["action"]
commit = data["commit_id"][:8]
2018-11-12 17:16:17 +00:00
commenter = utils.irc.bold(data["comment"]["user"]["login"])
2018-12-11 22:27:04 +00:00
url = self._short_url(data["comment"]["html_url"])
return ["[commit/%s] %s commented" % (commit, commenter, action)]
def pull_request(self, event, full_name, data):
2018-12-29 19:05:14 +00:00
number = data["pull_request"]["number"]
action = data["action"]
action_desc = action
branch = data["pull_request"]["base"]["ref"]
colored_branch = utils.irc.color(branch, utils.consts.LIGHTBLUE)
if action == "opened":
action_desc = "requested merge into %s" % colored_branch
elif action == "closed":
if data["pull_request"]["merged"]:
action_desc = "%s into %s" % (
utils.irc.color("merged", utils.consts.GREEN),
colored_branch)
else:
action_desc = utils.irc.color("closed without merging",
2018-11-13 16:02:26 +00:00
utils.consts.RED)
elif action == "synchronize":
action_desc = "committed to"
elif action == "labeled":
action_desc = "labeled as '%s'" % data["label"]["name"]
elif action == "unlabeled":
action_desc = "unlabeled as '%s'" % data["label"]["name"]
pr_title = data["pull_request"]["title"]
2018-11-12 17:16:17 +00:00
author = utils.irc.bold(data["sender"]["login"])
2018-12-11 22:27:04 +00:00
url = self._short_url(data["pull_request"]["html_url"])
2018-12-29 19:05:14 +00:00
return ["[pr #%d] %s %s: %s - %s" % (
number, author, action_desc, pr_title, url)]
def pull_request_review(self, event, full_name, data):
if data["review"]["state"] == "commented":
return []
2018-12-29 19:05:14 +00:00
number = data["pull_request"]["number"]
action = data["action"]
pr_title = data["pull_request"]["title"]
reviewer = utils.irc.bold(data["sender"]["login"])
2018-12-11 22:27:04 +00:00
url = self._short_url(data["review"]["html_url"])
2018-12-29 19:05:14 +00:00
return ["[pr #%d] %s %s a review on: %s - %s" % (
number, reviewer, action, pr_title, url)]
def pull_request_review_comment(self, event, full_name, data):
2018-12-29 19:05:14 +00:00
number = data["pull_request"]["number"]
action = data["action"]
pr_title = data["pull_request"]["title"]
sender = utils.irc.bold(data["sender"]["login"])
2018-12-11 22:27:04 +00:00
url = self._short_url(data["comment"]["html_url"])
2018-12-29 19:05:14 +00:00
return ["[pr #%d] %s %s on a review: %s - %s" %
(number, sender, COMMENT_ACTIONS[action], pr_title, url)]
def issues(self, event, full_name, data):
2018-12-29 19:05:14 +00:00
number = data["issue"]["number"]
action = data["action"]
action_desc = action
if action == "labeled":
action_desc = "labeled as '%s'" % data["label"]["name"]
elif action == "unlabeled":
action_desc = "unlabeled as '%s'" % data["label"]["name"]
issue_title = data["issue"]["title"]
2018-11-12 17:16:17 +00:00
author = utils.irc.bold(data["sender"]["login"])
2018-12-11 22:27:04 +00:00
url = self._short_url(data["issue"]["html_url"])
2018-12-29 19:05:14 +00:00
return ["[issue #%d] %s %s: %s - %s" %
(number, author, action_desc, issue_title, url)]
def issue_comment(self, event, full_name, data):
2018-12-29 19:05:14 +00:00
number = data["issue"]["number"]
action = data["action"]
issue_title = data["issue"]["title"]
type = "pr" if "pull_request" in data["issue"] else "issue"
2018-11-12 17:16:17 +00:00
commenter = utils.irc.bold(data["comment"]["user"]["login"])
2018-12-11 22:27:04 +00:00
url = self._short_url(data["comment"]["html_url"])
2018-12-29 19:05:14 +00:00
return ["[%s #%d] %s %s on: %s - %s" %
(type, number, commenter, COMMENT_ACTIONS[action], issue_title,
url)]
def create(self, event, full_name, data):
ref = data["ref"]
ref_color = utils.irc.color(ref, utils.consts.LIGHTBLUE)
type = data["ref_type"]
2018-11-17 21:06:27 +00:00
sender = utils.irc.bold(data["sender"]["login"])
2018-12-11 22:27:04 +00:00
url = self._short_url(CREATE_URL % (full_name, ref))
return ["%s created a %s: %s - %s" % (sender, type, ref_color, url)]
2018-11-17 21:10:03 +00:00
def delete(self, event, full_name, data):
ref = data["ref"]
type = data["ref_type"]
sender = utils.irc.bold(data["sender"]["login"])
return ["%s deleted a %s: %s" % (sender, type, ref)]
def release(self, event, full_name, data):
action = data["action"]
2018-11-17 22:17:14 +00:00
tag = data["release"]["tag_name"]
name = data["release"]["name"] or ""
if name:
name = ": %s"
author = utils.irc.bold(data["release"]["author"]["login"])
2018-12-11 22:27:04 +00:00
url = self._short_url(data["release"]["html_url"])
return ["%s %s a release%s - %s" % (author, action, name, url)]
def status(self, event, full_name, data):
context = data["context"]
state = data["state"]
url = data["target_url"]
commit = self._short_id(data["sha"])
return ["[%s status] %s is '%s' - %s" %
(commit, context, state, url)]
2018-12-10 18:47:54 +00:00
def fork(self, event, full_name, data):
forker = utils.irc.bold(data["sender"]["login"])
fork_full_name = utils.irc.color(data["forkee"]["full_name"],
utils.consts.LIGHTBLUE)
2018-12-11 22:27:04 +00:00
url = self._short_url(data["forkee"]["html_url"])
2018-12-10 18:47:54 +00:00
return ["%s forked into %s - %s" %
(forker, fork_full_name, url)]