ci: skip previously cleaned PRs (#27670)

This commit is contained in:
Aiden Cline
2026-05-15 00:23:09 -05:00
committed by GitHub
parent d59d99665b
commit ca8f578f2f

View File

@@ -8,6 +8,7 @@ const defaultThreshold = 2
const defaultSleepMs = 20_000
const defaultPrintLimit = 50
const positiveReactions = new Set(["THUMBS_UP", "HEART", "HOORAY", "ROCKET"])
const cleanupLabel = "automated-pr-cleanup"
const { values } = parseArgs({
args: Bun.argv.slice(2),
@@ -87,6 +88,11 @@ type PullRequest = {
totalCount: number
}
}>
labels: {
nodes: Array<{
name: string
}>
}
}
type GraphqlResponse = {
@@ -140,16 +146,18 @@ async function main() {
const prs = await fetchOpenPullRequests()
const recentCount = prs.filter((pr) => new Date(pr.createdAt) >= cutoff).length
const candidates = prs
const matching = prs
.map((pr) => ({ ...pr, positiveReactions: positiveReactionCount(pr) }))
.filter((pr) => new Date(pr.createdAt) < cutoff && pr.positiveReactions < threshold)
const candidates = matching.filter((pr) => !hasPriorCleanup(pr))
const selected = maxClose === undefined ? candidates : candidates.slice(0, maxClose)
console.log(`Fetched ${prs.length} open PRs`)
console.log(`Matching cleanup criteria: ${candidates.length}`)
console.log(`Matching cleanup criteria: ${matching.length}`)
console.log(`Skipped previously cleaned PRs: ${matching.length - candidates.length}`)
console.log(`Recent PRs untouched: ${recentCount}`)
console.log(
`Older PRs with at least ${threshold} positive reactions untouched: ${prs.length - candidates.length - recentCount}`,
`Older PRs with at least ${threshold} positive reactions untouched: ${prs.length - matching.length - recentCount}`,
)
if (selected.length === 0) return
@@ -164,6 +172,8 @@ async function main() {
return
}
await ensureCleanupLabel()
console.log(`\nCommenting and closing ${selected.length} PRs...`)
for (const pr of selected) {
await closePullRequest(pr)
@@ -201,6 +211,11 @@ async function fetchOpenPullRequests() {
totalCount
}
}
labels(first: 100) {
nodes {
name
}
}
}
}
}
@@ -249,9 +264,34 @@ async function closePullRequest(pr: CleanupCandidate) {
method: "PATCH",
body: JSON.stringify({ state: "closed" }),
})
await githubRequest(`/repos/${repo.owner}/${repo.name}/issues/${pr.number}/labels`, {
method: "POST",
body: JSON.stringify({ labels: [cleanupLabel] }),
})
console.log(`Closed #${pr.number} positive=${pr.positiveReactions} ${pr.url}`)
}
async function ensureCleanupLabel() {
const response = await fetch(
`https://api.github.com/repos/${repo.owner}/${repo.name}/labels/${encodeURIComponent(cleanupLabel)}`,
{
headers,
},
)
if (response.ok) return
if (response.status !== 404)
throw new Error(`Failed to check cleanup label: ${response.status} ${response.statusText}`)
await githubRequest(`/repos/${repo.owner}/${repo.name}/labels`, {
method: "POST",
body: JSON.stringify({
name: cleanupLabel,
color: "ededed",
description: "PR was closed by automated cleanup",
}),
})
}
async function githubRequest(path: string, init: RequestInit, attempt = 0): Promise<Response> {
const response = await fetch(path.startsWith("https://") ? path : `https://api.github.com${path}`, {
...init,
@@ -272,10 +312,12 @@ async function githubRequest(path: string, init: RequestInit, attempt = 0): Prom
? Math.max(0, Number(reset) * 1000 - Date.now()) + 1_000
: body.toLowerCase().includes("secondary rate limit")
? 300_000
: 0
: response.status >= 500
? Math.min(300_000, 10_000 * 2 ** attempt)
: 0
if ((response.status === 403 || response.status === 429) && retryMs > 0 && attempt < 10) {
console.warn(`GitHub rate limit hit; sleeping ${Math.ceil(retryMs / 1000)}s before retry ${attempt + 1}`)
if ((response.status === 403 || response.status === 429 || response.status >= 500) && retryMs > 0 && attempt < 10) {
console.warn(`GitHub request failed; sleeping ${Math.ceil(retryMs / 1000)}s before retry ${attempt + 1}`)
await sleep(retryMs)
return githubRequest(path, init, attempt + 1)
}
@@ -289,6 +331,10 @@ function positiveReactionCount(pr: PullRequest) {
.reduce((total, group) => total + group.users.totalCount, 0)
}
function hasPriorCleanup(pr: PullRequest) {
return pr.labels.nodes.some((label) => label.name === cleanupLabel)
}
function requireRepo(value: string | undefined) {
if (!value) throw new Error("repo is required")
const [owner, name] = value.split("/")