github-webhook-filter/lib/filter.ts

91 lines
3.1 KiB
TypeScript
Raw Normal View History

import { commentManager } from "./manager.ts";
import { UrlConfig } from "./types.d.ts";
import { requestLog } from "./util.ts";
export default async function filter(
headers: Headers,
json: any,
config: UrlConfig,
): Promise<string | null> {
const reqLog = requestLog(headers);
2022-09-03 00:39:36 +00:00
const event = headers.get("x-github-event") || "unknown";
2022-09-03 15:10:58 +00:00
const login: string | undefined = json.sender?.login?.toLowerCase();
2022-09-03 14:48:28 +00:00
// ignore events that Discord won't render anyway
2022-09-03 00:39:36 +00:00
if (["status", "pull_request_review_thread"].includes(event)) {
return event;
}
2022-09-03 14:48:28 +00:00
// ignore all PR actions except "opened", "closed", "reopened"
2022-09-03 00:39:36 +00:00
if (
event === "pull_request" && json.action &&
!["opened", "closed", "reopened"].includes(json.action)
) {
return `no-op PR action '${json.action}'`;
2022-09-03 00:39:36 +00:00
}
2022-09-03 14:48:28 +00:00
// ignore all issue actions except "opened", "deleted", "closed", "reopened", "transferred"
2022-09-03 00:39:36 +00:00
if (
event === "issues" && json.action &&
!["opened", "deleted", "closed", "reopened", "transferred"].includes(json.action)
) {
return `no-op issue action '${json.action}'`;
2022-09-03 00:39:36 +00:00
}
2022-09-03 14:48:28 +00:00
// ignore some PR review actions
2022-09-03 01:31:48 +00:00
if (event === "pull_request_review") {
2022-09-03 14:08:20 +00:00
// ignore edit/dismiss actions
if (json.action !== "submitted") return `no-op PR review action '${json.action}'`;
2022-09-03 14:08:20 +00:00
// if comment (not approval or changes requested), ignore empty review body
if (json.review?.state === "commented" && !json.review?.body) return "empty PR review";
2022-09-03 01:31:48 +00:00
}
2022-09-03 15:10:58 +00:00
// ignore some PR comment events
if (event === "pull_request_review_comment") {
// ignore edit/delete actions
if (json.action !== "created") return `no-op PR comment action '${json.action}'`;
2022-09-03 15:10:58 +00:00
// check if more than x comments on a PR review in a short timespan
const reviewId: number = json.comment?.pull_request_review_id;
if (config.commentBurstLimit && reviewId) {
const cacheKey = `${reviewId}-${login}`;
reqLog.debug(`filter: checking cache key ${cacheKey}`);
const curr = await commentManager.getAndIncrement(cacheKey);
reqLog.debug(`filter: current value: ${curr}`);
2022-09-03 15:10:58 +00:00
if (curr && curr >= config.commentBurstLimit) {
2022-09-04 13:09:33 +00:00
return `exceeded comment burst limit (${config.commentBurstLimit}) for review ${reviewId}`;
2022-09-03 15:10:58 +00:00
}
}
}
2022-09-03 14:48:28 +00:00
// ignore bots
if (
login &&
["coveralls[bot]", "netlify[bot]", "pre-commit-ci[bot]"].some((n) => login.includes(n))
) {
return "bot";
}
2022-09-03 14:48:28 +00:00
// ignore branch/tag if configured
const refMatch = /^refs\/([^\/]+)\/(.+)$/.exec(json.ref);
if (event === "push" && refMatch) {
// check if branch is allowed
if (
refMatch[1] == "heads" && config.allowBranches !== undefined &&
!config.allowBranches.includes(refMatch[2])
) {
return `branch '${refMatch[2]}' not in ${JSON.stringify(config.allowBranches)}`;
}
// check if it's a tag
if (refMatch[1] == "tags" && config.hideTags === true) {
return `tag '${refMatch[2]}'`;
}
}
return null;
}