monkeytype/bin/buildChangelog.mjs
2024-01-24 13:44:30 +01:00

316 lines
9.5 KiB
JavaScript

import conventionalChangelog from "conventional-changelog";
import { exec } from "child_process";
const stream = conventionalChangelog(
{
preset: {
name: "conventionalcommits",
types: [
{ type: "feat", section: "Features" },
{ type: "impr", section: "Improvements" },
{ type: "fix", section: "Fixes" },
],
},
},
undefined,
undefined,
undefined,
{
headerPartial: "",
}
);
const header =
"Thank you to all the contributors who made this release possible!";
let log = "";
for await (const chunk of stream) {
log += chunk;
}
log = log.replace(/^\*/gm, "-");
// console.log(log);
// console.log(header + log + footer);
//i might come back to the approach below at some point
async function getLog() {
return new Promise((resolve, reject) => {
exec(
`git log --oneline $(git describe --tags --abbrev=0 @^)..@ --pretty="format:%H %h %s"`,
(err, stdout, stderr) => {
if (err) {
reject(err);
}
resolve(stdout);
}
);
});
}
function itemIsAddingQuotes(item) {
const scopeIsQuote =
item.scope?.includes("quote") ||
item.scope?.includes("quotes") ||
item.message?.includes("quote");
const messageAdds =
item.message.includes("add") ||
item.message.includes("added") ||
item.message.includes("adding") ||
item.message.includes("adds");
return scopeIsQuote && messageAdds;
}
function itemIsAddressingQuoteReports(item) {
const scopeIsQuote =
item.scope?.includes("quote") || item.scope?.includes("quotes");
const messageReport =
item.message.includes("report") || item.message.includes("reports");
return scopeIsQuote && messageReport;
}
const titles = {
feat: "Features",
impr: "Improvements",
fix: "Fixes",
};
function getPrLink(pr) {
const prNum = pr.replace("#", "");
return `[#${prNum}](https://github.com/monkeytypegame/monkeytype/issues/${prNum})`;
}
function getCommitLink(hash, longHash) {
return `[${hash}](https://github.com/monkeytypegame/monkeytype/commit/${longHash})`;
}
function buildSection(type, allItems) {
let ret = `### ${titles[type]}\n\n`;
const items = allItems.filter((item) => item.type === type);
if (items.length === 0) {
return "";
}
for (let item of items) {
const scope = item.scope ? `**${item.scope}:** ` : "";
const usernames =
item.usernames.length > 0 ? ` (${item.usernames.join(", ")})` : "";
const pr =
item.prs.length > 0
? ` (${item.prs.map((p) => getPrLink(p)).join(", ")})`
: "";
const hash = ` (${item.hashes
.map((h) => getCommitLink(h.short, h.full))
.join(", ")})`;
ret += `- ${scope}${item.message}${usernames}${pr}${hash}\n`;
}
return ret;
}
function buildFooter(logs) {
const styleLogs = logs.filter((item) => item.type === "style");
const docLogs = logs.filter((item) => item.type === "docs");
const refactorLogs = logs.filter((item) => item.type === "refactor");
const perfLogs = logs.filter((item) => item.type === "perf");
const ciLogs = logs.filter((item) => item.type === "ci");
const testLogs = logs.filter((item) => item.type === "test");
const buildLogs = logs.filter((item) => item.type === "build");
const otherStrings = [];
if (styleLogs.length > 0) {
otherStrings.push("style");
}
if (docLogs.length > 0) {
otherStrings.push("documentation");
}
if (refactorLogs.length > 0) {
otherStrings.push("refactoring");
}
if (perfLogs.length > 0) {
otherStrings.push("performance");
}
if (ciLogs.length > 0) {
otherStrings.push("CI");
}
if (testLogs.length > 0) {
otherStrings.push("testing");
}
if (buildLogs.length > 0) {
otherStrings.push("build");
}
if (otherStrings.length === 0) {
return "";
}
//build a string where otherStrings are joined by commas and the last one is joined by "and"
const finalString =
otherStrings.length > 1
? otherStrings.slice(0, -1).join(", ") + " and " + otherStrings.slice(-1)
: otherStrings[0];
return `\n### Other\n\n- Various ${finalString} changes`;
}
function convertStringToLog(logString) {
let log = [];
for (let line of logString) {
//split line based on the format: d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 style: adjusted testConfig and modesNotice.
//use regex to split
const [_, hash, shortHash, fullMessage] = line.split(
/(\w{40}) (\w{9,10}) (.*)/
);
//split message using regex based on fix(language): spelling mistakes in Nepali wordlist and quotes (sapradhan) (#4528)
//scope is optional, username is optional, pr number is optional
const [__, type, scope, message, message2, message3] = fullMessage.split(
/^(\w+)(?:\(([^)]+)\))?:\s+(.+?)\s*(?:\(([^)]+)\))?(?:\s+\(([^)]+)\))?(?:\s+\(([^)]+)\))?$/
);
const usernames = message2 && message3 ? message2.split(", ") : [];
let pr;
if (message2 && message3) {
pr = message3;
} else if (message2 && !message3) {
pr = message2;
}
const prs = pr ? pr.split(", ") : [];
if (type && message) {
log.push({
hashes: [
{
short: shortHash,
full: hash,
},
],
type,
scope,
message,
usernames,
prs,
});
} else {
console.warn("skipping line due to invalid format: " + line);
}
}
return log;
}
async function main() {
let logString = await getLog();
logString = logString.split("\n");
//test commits
// const logString = [
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 build: add new feature (miodec, someone) (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 build(scope): add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 chore: add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 chore(scope): add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 ci: add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 ci(scope): add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 docs: add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 docs(scope): add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 feat: add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 feat(scope): add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 impr: add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 impr(scope): add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 fix: add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 fix(score): add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 perf: add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 perf(scope): add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 refactor: add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 refactor(scope): add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 revert: add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 revert(scope): add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 style: add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 style(scope): add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 test: add new feature (#1234)",
// "d2739e4f193137db4d86450f0d50b3489d75c106 d2739e4f1 test(scope): add new feature (#1234)",
// ];
let log = convertStringToLog(logString);
const contributorCount = log
.map((l) => {
const filtered = l.usernames.filter((u) => {
const lowerCased = u.toLowerCase();
return (
lowerCased !== "monkeytype-bot" &&
lowerCased !== "dependabot" &&
lowerCased !== "miodec"
);
});
return filtered;
})
.flat().length;
let quoteAddCommits = log.filter((item) => itemIsAddingQuotes(item));
log = log.filter((item) => !itemIsAddingQuotes(item));
let quoteReportCommits = log.filter((item) =>
itemIsAddressingQuoteReports(item)
);
log = log.filter((item) => !itemIsAddressingQuoteReports(item));
if (quoteAddCommits.length > 0) {
log.push({
hashes: quoteAddCommits.map((item) => item.hashes).flat(),
type: "impr",
scope: "quote",
message: "add quotes in various languages",
usernames: quoteAddCommits.map((item) => item.usernames).flat(),
prs: quoteAddCommits.map((item) => item.prs).flat(),
});
}
if (quoteReportCommits.length > 0) {
log.push({
hashes: quoteReportCommits.map((item) => item.hashes).flat(),
type: "fix",
scope: "quote",
message: "update or remove quotes reported by users",
usernames: quoteReportCommits.map((item) => item.usernames).flat(),
prs: quoteReportCommits.map((item) => item.prs).flat(),
});
}
let final = "";
if (contributorCount > 0) {
final += header + "\n\n\n";
}
const sections = [];
for (const type of Object.keys(titles)) {
const section = buildSection(type, log);
if (section) {
sections.push(section);
}
}
final += sections.join("\n\n");
const footer = buildFooter(log);
if (footer) {
final += "\n" + footer;
}
console.log(final);
}
main();