diff options
| -rw-r--r-- | modules/git_webhooks/__init__.py | 5 | ||||
| -rw-r--r-- | modules/git_webhooks/gitlab.py | 169 |
2 files changed, 174 insertions, 0 deletions
diff --git a/modules/git_webhooks/__init__.py b/modules/git_webhooks/__init__.py index f2dd1f82..a3c3317b 100644 --- a/modules/git_webhooks/__init__.py +++ b/modules/git_webhooks/__init__.py @@ -38,6 +38,11 @@ class Module(ModuleManager.BaseModule): return self._webhook("gitea", "Gitea", self._gitea, event["data"], event["headers"]) + @utils.hook("api.post.gitlab") + def _api_gitlab_webhook(self, event): + return self._webhook("gitlab", "GitLab", self._gitlab, + event["data"], event["headers"]) + def _webhook(self, webhook_type, webhook_name, handler, payload_str, headers): payload = payload_str.decode("utf8") diff --git a/modules/git_webhooks/gitlab.py b/modules/git_webhooks/gitlab.py new file mode 100644 index 00000000..50e819dd --- /dev/null +++ b/modules/git_webhooks/gitlab.py @@ -0,0 +1,169 @@ +from src import ModuleManager, utils +from . import colors + +EVENT_CATEGORIES = { + "ping": [ + "ping" # new webhook received + ], + "code": ["push"], + "pr-minimal": [ + "pull_request/opened", "pull_request/closed", "pull_request/reopened" + ], + "pr": [ + "pull_request/opened", "pull_request/closed", "pull_request/reopened", + "pull_request/edited", "pull_request/assigned", + "pull_request/unassigned" + ], + "pr-all": ["pull_request"], + "issue-minimal": [ + "issues/opened", "issues/closed", "issues/reopened", "issues/deleted" + ], + "issue": [ + "issues/opened", "issues/closed", "issues/reopened", "issues/deleted", + "issues/edited", "issues/assigned", "issues/unassigned", "issue_comment" + ], + "issue-all": [ + "issues", "issue_comment" + ], + "issue-comment-minimal": [ + "issue_comment/created", "issue_comment/deleted" + ], + "repo": [ + "create", # a repository, branch or tag has been created + "delete", # same as above but deleted + "release", + "fork", + "repository" + ] +} + +COMMENT_ACTIONS = { + "created": "commented", + "edited": "edited a comment", + "deleted": "deleted a comment" +} + +class GitLab(object): + def is_private(self, data, headers): + if "project" in data: + return not data["project"]["visibility_level"] == 20 + return False + + def names(self, data, headers): + full_name = data["project"]["path_with_namespace"] + repo_username = data["project"]["namespace"] + repo_name = data["project"]["name"] + + return full_name, repo_username, repo_name, None + + def branch(self, data, headers): + if "ref" in data: + return data["ref"].rpartition("/")[2] + return None + + def event(self, data, headers): + event = headers["X-GitLab-Event"].rsplit(" ", 1)[0].lower() + event = event.replace(" ", "_") + event_action = None + if ("object_attributes" in data and + "action" in data["object_attributes"]): + event_action = "%s/%s" % ( + event, data["object_attributes"]["action"]) + return event, event_action + + def event_categories(self, event): + return EVENT_CATEGORIES.get(event, [event]) + + def webhook(self, full_name, event, data, headers): + if event == "push": + return self.push(full_name, data) + elif event == "pull_request": + return self.pull_request(full_name, data) + elif event == "issues": + return self.issues(full_name, data) + elif event == "note": + return self.note(full_name, data) + + def _short_hash(self, hash): + return hash[:8] + + def push(self, full_name, data): + outputs = [] + branch = data["ref"].rpartition("/")[2] + branch = utils.irc.color(branch, colors.COLOR_BRANCH) + author = utils.irc.bold(data["user_username"]) + + if len(data["commits"]) <= 3: + for commit in data["commits"]: + hash = commit["id"] + hash_colored = utils.irc.color(self._short_hash(hash), + colors.COLOR_ID) + message = commit["message"].split("\n")[0].strip() + url = commit["url"] + + outputs.append( + "%s pushed %s to %s: %s - %s" + % (author, hash_colored, branch, message, url)) + else: + first_id = data["before"] + last_id = data["after"] + url = data["compare_url"] + + outputs.append("%s pushed %d commits to %s" + % (author, len(data["commits"]), branch)) + + return outputs + + def merge_request(self, full_name, data): + number = utils.irc.color("#%s" % data["object_attributes"]["iid"], + colors.COLOR_ID) + action = data["object_attributes"]["action"] + action_desc = "%s %s" % (action, number) + branch = data["object_attributes"]["target_branch"] + colored_branch = utils.irc.color(branch, colors.COLOR_BRANCH) + + if action == "open": + action_desc = "requested %s merge into %s" % (number, + colored_branch) + elif action == "close": + if data["pull_request"]["merged"]: + action_desc = "%s %s into %s" % ( + utils.irc.color("merged", colors.COLOR_POSITIVE), number, + colored_branch) + else: + action_desc = "%s %s" % ( + utils.irc.color("closed", colors.COLOR_NEGATIVE), number) + + pr_title = data["object_attributes"]["title"] + author = utils.irc.bold(data["user"]["username"]) + url = data["object_attributes"]["url"] + return ["[PR] %s %s: %s - %s" % ( + author, action_desc, pr_title, url)] + + def issues(self, full_name, data): + number = utils.irc.color("#%s" % data["object_attributes"]["iid"], + colors.COLOR_ID) + action = data["object_attributes"]["action"] + issue_title = data["object_attributes"]["description"] + author = utils.irc.bold(data["user"]["username"]) + url = data["object_attributes"]["url"] + + return ["[issue] %s %s %s: %s - %s" % + (author, action, number, issue_title, url)] + + def note(self, full_name, data): + type = data["object_attributes"]["noteable_type"] + if type in ["Issue", "MergeRequest"]: + return self.issue_comment(full_name, data) + + def issue_note(self, full_name, data): + number = utils.irc.color("#%s" % data["object_attributes"]["iid"], + colors.COLOR_ID) + type = data["object_attributes"]["noteable_type"] + type == "issue" if type == "Issue" else "PR" + + issue_title = data["issue"]["title"] + commenter = utils.irc.bold(data["user"]["username"]) + url = data["object_attributes"]["url"] + return ["[%s] %s commented on %s: %s - %s" % + (type, commenter, number, issue_title, url)] |
