github-webhook-filter/lib/filter.ts

84 lines
3.1 KiB
TypeScript
Raw Normal View History

2022-09-03 15:10:58 +00:00
import { log, TTL } from "../deps.ts";
import { UrlConfig } from "./types.d.ts";
2022-09-03 15:10:58 +00:00
const reviewComments = new TTL<number>(2 * 1000);
export default function filter(headers: Headers, json: any, config: UrlConfig): string | null {
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)
) {
2022-09-03 14:09:11 +00:00
return "no-op PR 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)
) {
2022-09-03 14:09:11 +00:00
return "no-op issue 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
2022-09-03 14:09:11 +00:00
if (json.action !== "submitted") return "no-op PR review action";
2022-09-03 14:08:20 +00:00
// if comment (not approval or changes requested), ignore empty review body
else 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";
// 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}`;
log.debug(`filter: checking cache key ${cacheKey}`);
log.debug(`filter: full comment cache ${JSON.stringify(Array.from(reviewComments))}`);
const curr = reviewComments.get(cacheKey);
if (curr && curr >= config.commentBurstLimit) {
return `exceeded comment burst limit ${config.commentBurstLimit} for review ${reviewId}`;
}
reviewComments.set(cacheKey, (curr ?? 0) + 1);
}
}
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";
}
}
return null;
}