ci: guard changelog bot attributions

This commit is contained in:
Peter Steinberger
2026-04-27 14:28:05 +01:00
parent 6e8aaef1cc
commit bbbc80ddcc
7 changed files with 92 additions and 8 deletions

View File

@@ -54,6 +54,7 @@ export function createChangedCheckPlan(result, options = {}) {
const addLint = (name, args) => add(name, args, baseEnv);
add("conflict markers", ["check:no-conflict-markers"]);
add("changelog attributions", ["check:changelog-attributions"]);
if (result.docsOnly) {
return {

View File

@@ -0,0 +1,50 @@
#!/usr/bin/env node
import fs from "node:fs";
import path from "node:path";
import { fileURLToPath } from "node:url";
export const FORBIDDEN_CHANGELOG_THANKS_HANDLES = ["codex", "openclaw", "steipete"];
const HANDLE_PATTERN = FORBIDDEN_CHANGELOG_THANKS_HANDLES.join("|");
const FORBIDDEN_THANKS_PATTERN = new RegExp(
`\\bThanks\\b[^\\n]*@(${HANDLE_PATTERN})(?=\\b|[^A-Za-z0-9-])`,
"iu",
);
export function findForbiddenChangelogThanks(content) {
return content
.split(/\r?\n/u)
.map((text, index) => {
const match = text.match(FORBIDDEN_THANKS_PATTERN);
return match ? { line: index + 1, handle: match[1].toLowerCase(), text } : null;
})
.filter(Boolean);
}
export async function main(argv = process.argv.slice(2)) {
const changelogPath = argv[0] ?? "CHANGELOG.md";
const absolutePath = path.resolve(process.cwd(), changelogPath);
const content = fs.readFileSync(absolutePath, "utf8");
const violations = findForbiddenChangelogThanks(content);
if (violations.length === 0) {
return;
}
console.error("Forbidden changelog thanks attribution:");
for (const violation of violations) {
const relativePath = path.relative(process.cwd(), absolutePath) || changelogPath;
console.error(`- ${relativePath}:${violation.line} uses Thanks @${violation.handle}`);
}
console.error(
`Use a credited external GitHub username instead of ${FORBIDDEN_CHANGELOG_THANKS_HANDLES.map((handle) => `@${handle}`).join(", ")}.`,
);
process.exitCode = 1;
}
if (process.argv[1] && path.resolve(process.argv[1]) === fileURLToPath(import.meta.url)) {
main().catch((error) => {
console.error(error);
process.exit(1);
});
}

View File

@@ -30,6 +30,7 @@ export async function main(argv = process.argv.slice(2)) {
parallel: true,
commands: [
{ name: "conflict markers", args: ["check:no-conflict-markers"] },
{ name: "changelog attributions", args: ["check:changelog-attributions"] },
{ name: "tool display", args: ["tool-display:check"] },
{ name: "host env policy", args: ["check:host-env-policy:swift"] },
],