/[osn-commons]/trunk/git/genchangelog
ViewVC logotype

Diff of /trunk/git/genchangelog

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 54 by rakinar2, Thu Aug 29 06:17:33 2024 UTC revision 56 by rakinar2, Fri Aug 30 11:55:09 2024 UTC
# Line 22  Line 22 
22  const path = require("path");  const path = require("path");
23  const { existsSync } = require("fs");  const { existsSync } = require("fs");
24  const { exit } = require("process");  const { exit } = require("process");
25  const { execSync } = require("child_process");  const { execSync, exec, spawn } = require("child_process");
26  const { writeFile } = require("fs/promises");  const { writeFile } = require("fs/promises");
27  const { parseArgs } = require("util");  const { parseArgs } = require("util");
28    const crypto = require("crypto");
29    
30  const ME = path.basename(process.argv[1]);  const ME = path.basename(process.argv[1]);
31    const GIT_COMMIT_BOUNDARY = crypto.randomBytes(64).toString("hex");
32    const GIT_SPACE_BOUNDARY = crypto.randomBytes(64).toString("hex");
33    
34  function print(...args) {  function print(...args) {
35      console.log(`${ME}: ${args.join(" ")}`);      console.log(`${ME}: ${args.join(" ")}`);
# Line 38  function perror(...args) { Line 41  function perror(...args) {
41    
42  function findInPath(executable) {  function findInPath(executable) {
43      for (const segment of process.env.PATH?.split(      for (const segment of process.env.PATH?.split(
44          process.platform === "win32" ? ";" : ":"          process.platform === "win32" ? ";" : ":",
45      ) ?? []) {      ) ?? []) {
46          const executablePath = path.join(          const executablePath = path.join(
47              segment,              segment,
48              executable + (process.platform === "win32" ? ".exe" : "")              executable + (process.platform === "win32" ? ".exe" : ""),
49          );          );
50    
51          if (existsSync(executablePath)) {          if (existsSync(executablePath)) {
# Line 65  function checkForGit() { Line 68  function checkForGit() {
68      return gitPath;      return gitPath;
69  }  }
70    
71  function getGitLog(gitPath) {  async function getGitLog(gitPath) {
72      try {      try {
73          return execSync(gitPath + " --no-pager log", { encoding: "utf8" });          let output = "";
74      } catch {  
75          perror("command `git --no-pager log' failed");          const child = spawn(
76                gitPath,
77                [
78                    "--no-pager",
79                    "log",
80                    `--pretty=format:%h %H %an${GIT_SPACE_BOUNDARY} %ae %ad${GIT_SPACE_BOUNDARY} %B${GIT_COMMIT_BOUNDARY}`,
81                ],
82                { encoding: "utf8", stdio: "pipe" },
83            );
84    
85            child.stdout.on("data", (data) => {
86                output += data;
87            });
88    
89            child.stderr.on("data", (data) => {
90                console.error(data);
91            });
92    
93            await new Promise((resolve) => {
94                child.on("close", (code) => {
95                    if (code !== 0) {
96                        perror("command `git log' failed with exit code " + code);
97                        exit(1);
98                    } else {
99                        resolve();
100                    }
101                });
102            });
103    
104            return output;
105        } catch (error) {
106            console.error(error);
107            perror("command `git log' failed");
108          exit(1);          exit(1);
109      }      }
110  }  }
111    
112    function strUntil(str, boundary) {
113        let output = "";
114        let i = 0;
115    
116        for (i = 0; i < str.length; i++) {
117            if (str[i] === boundary[0]) {
118                let boundaryIndex = 0;
119                let previousI = i;
120    
121                while (
122                    boundaryIndex < boundary.length &&
123                    i < str.length &&
124                    str[i] === boundary[boundaryIndex]
125                ) {
126                    i++;
127                    boundaryIndex++;
128                }
129    
130                if (boundaryIndex === boundary.length) {
131                    return { output, size: i };
132                }
133    
134                i = previousI;
135            }
136    
137            output += str[i];
138        }
139    
140        return { output, size: i };
141    }
142    
143  function parseGitLog(gitLog) {  function parseGitLog(gitLog) {
     const lines = gitLog.split("\n");  
144      const commits = [];      const commits = [];
145    
146      for (let i = 0; i < lines.length; ) {      let i = 0;
         if (!lines[i].startsWith("commit")) {  
             i++;  
             continue;  
         }  
147    
148          const [, hash] = lines[i++].split(" ");      while (i < gitLog.length) {
149          const headerProps = {};          const { output, size } = strUntil(gitLog.slice(i), GIT_COMMIT_BOUNDARY);
150            i += size;
151    
152          while (          let outputIndex = 0;
             i < lines.length &&  
             lines[i].trim() !== "" &&  
             !/^\s/.test(lines[i])  
         ) {  
             const colonIndex = lines[i].indexOf(":");  
             const name = lines[i].slice(0, colonIndex).toLowerCase();  
             const value = lines[i].slice(colonIndex + 1).trim();  
             headerProps[name] = value;  
             i++;  
         }  
153    
154          const messageLines = [];          const shortIdSpaceIndex = output.indexOf(" ");
155            const idSpaceIndex = output.indexOf(" ", shortIdSpaceIndex + 1);
156            const shortId = output.slice(outputIndex, shortIdSpaceIndex);
157            const id = output.slice(shortIdSpaceIndex + 1, idSpaceIndex);
158    
159          while (i < lines.length && !lines[i].startsWith("commit")) {          outputIndex += shortId.length + id.length + 2;
             const lineToPush = lines[i++].replace(/^    /, "");  
160    
161              if (!lineToPush) {          const { output: authorName, size: authorNameSize } = strUntil(
162                  continue;              output.slice(outputIndex),
163              }              GIT_SPACE_BOUNDARY,
164            );
165    
166              messageLines.push(lineToPush);          outputIndex += authorNameSize + 1;
167          }  
168            const authorEmailSpaceIndex = output.indexOf(" ", outputIndex + 1);
169            const authorEmail = output.slice(outputIndex, authorEmailSpaceIndex);
170    
171            outputIndex += authorEmail.length + 1;
172    
173            const { output: date, size: dateSize } = strUntil(
174                output.slice(outputIndex),
175                GIT_SPACE_BOUNDARY,
176            );
177    
178            outputIndex += dateSize + 1;
179    
180            const message = output.slice(outputIndex);
181            const newlineIndex = message.indexOf("\n");
182            const subject = message.slice(
183                0,
184                newlineIndex === -1 ? undefined : newlineIndex,
185            );
186            const body = newlineIndex === -1 ? "" : message.slice(newlineIndex + 1);
187    
         let mindex = messageLines.length - 1;  
         const footerProps = {};  
188          const validFooterProps = [          const validFooterProps = [
189              "signed-off-by",              "signed-off-by",
190              "co-authored-by",              "co-authored-by",
191              "on-behalf-of",              "on-behalf-of",
192          ];          ];
193            const footerProps = {};
194            const coAuthors = [];
195    
196          while (          for (const line of body.split("\n")) {
197              mindex >= 1 &&              const colonIndex = line.indexOf(":");
198              /^[A-Za-z0-9-]+: /.test(messageLines.at(mindex))              const key = line.slice(0, colonIndex).trim().toLowerCase();
199          ) {              const value = line.slice(colonIndex + 1).trim();
             const messageLine = messageLines[mindex--];  
             const colonIndex = messageLine.indexOf(":");  
             const name = messageLine.slice(0, colonIndex).toLowerCase();  
200    
201              if (!validFooterProps.includes(name)) {              if (validFooterProps.includes(key)) {
202                  continue;                  footerProps[key] = value;
203              }              }
204    
205              const value = messageLine.slice(colonIndex + 1).trim();              if (key === "co-authored-by") {
206                    const name = value.slice(0, value.lastIndexOf(" ")).trim();
207    
208              if (name in footerProps && !Array.isArray(footerProps[name])) {                  coAuthors.push({
209                  footerProps[name] = [footerProps[name]];                      name,
210                        email: value
211                            .slice(name.length)
212                            .trim()
213                            .replace(/^<|>$/g, ""),
214                    });
215              }              }
   
             if (Array.isArray(footerProps[name])) {  
                 footerProps[name].push(value);  
             } else {  
                 footerProps[name] = value;  
             }  
   
             messageLines.splice(mindex - 1, 1);  
216          }          }
217    
         const message = messageLines.join("\n");  
   
218          commits.push({          commits.push({
219              hash,              shortId,
220              message,              id,
221              headerProps,              author: {
222                    name: authorName,
223                    email: authorEmail,
224                },
225                date: new Date(date),
226                subject,
227                body,
228              footerProps,              footerProps,
229              signedOffBy: footerProps["signed-off-by"],              message,
230              onBehalfOf: footerProps["on-behalf-of"],              coAuthors,
231              author: headerProps["author"],              authors: [
232              createdAt: new Date(headerProps["date"]),                  {
233                        name: authorName,
234                        email: authorEmail,
235                    },
236                    ...coAuthors,
237                ],
238          });          });
239      }      }
240    
241      return commits;      return commits;
242  }  }
243    
244    function escapeMarkdown(str) {
245        return str.replace(/([_*~`])/g, "\\$1");
246    }
247    
248  function generateMarkdownChangelog(commits) {  function generateMarkdownChangelog(commits) {
249      let output = "# Changelog\n\n";      let output = "# Changelog\n\n";
250    
251      const grouppedCommitsByDate = {};      const grouppedCommitsByDate = {};
252    
253      for (const commit of commits) {      for (const commit of commits) {
254          const key = `${commit.createdAt          const key = `${commit.date.getUTCDate().toString().padStart(2, 0)}-${(
255              .getUTCDate()              commit.date.getUTCMonth() + 1
256              .toString()          )
             .padStart(2, 0)}-${(commit.createdAt.getUTCMonth() + 1)  
257              .toString()              .toString()
258              .padStart(2, "0")}-${commit.createdAt.getUTCFullYear()}::${              .padStart(2, "0")}-${commit.date.getUTCFullYear()}::${commit.authors
259              Array.isArray(commit.author)              .map((a) => `[${escapeMarkdown(a.name)}](mailto:${a.email})`)
260                  ? commit.author.join(":")              .join(":")}`;
                 : commit.author  
         }`;  
261          grouppedCommitsByDate[key] ??= [];          grouppedCommitsByDate[key] ??= [];
262          grouppedCommitsByDate[key].push(commit);          grouppedCommitsByDate[key].push(commit);
263      }      }
264    
265      for (const key in grouppedCommitsByDate) {      for (const key in grouppedCommitsByDate) {
266          const [date, author] = key.split("::");          const [date, author] = key.split("::");
267          output += `### ${date} [${author}]\n\n`;          output += `### ${date} - [${author}]\n\n`;
268    
269          for (const commit of grouppedCommitsByDate[key]) {          for (const commit of grouppedCommitsByDate[key]) {
270              const newLineIndex = commit.message.indexOf("\n");              const conventionalCommitType = commit.subject.match(
271              output += `* ${commit.message.slice(0, newLineIndex === -1 ? undefined : newLineIndex)}\n`;                  /^(build|chore|ci|docs|feat|fix|perf|refactor|revert|style|test|release|deps)(\(.+\))?:/,
272                );
273                let needsBreakingChangeWarning = conventionalCommitType?.[0]
274                    ?.trim()
275                    ?.endsWith("!:");
276    
277                if (conventionalCommitType) {
278                    output += `  * **${conventionalCommitType[0]}**${commit.subject.slice(conventionalCommitType[0].length)}`;
279                } else {
280                    output += `  * ${commit.subject}`;
281                }
282    
283                if (commit.body) {
284                    output += "    \n";
285    
286                    for (const line of commit.body.split("\n")) {
287                        const colonIndex = line.indexOf(":");
288    
289                        if (colonIndex !== -1) {
290                            const name = line.slice(0, colonIndex);
291                            const value = line.slice(colonIndex + 1);
292                            output += `    **${name}:** ${value}  \n`;
293    
294                            if (name === "BREAKING CHANGE") {
295                                needsBreakingChangeWarning = false;
296                            }
297                        } else {
298                            output += `    ${line}  \n`;
299                        }
300                    }
301                }
302    
303                if (needsBreakingChangeWarning) {
304                    output += "    **This is a breaking change.**  \n";
305                    output += "    \n";
306                }
307    
308                if (!commit.body && !needsBreakingChangeWarning) {
309                    output += "\n";
310                }
311          }          }
312    
313          output += "\n";          output += "\n";
314      }      }
315    
316      return output;      return output.trimEnd();
317  }  }
318    
319  function generateChangelog(commits) {  function generateChangelog(commits) {
# Line 204  function generateChangelog(commits) { Line 322  function generateChangelog(commits) {
322      const grouppedCommitsByDate = {};      const grouppedCommitsByDate = {};
323    
324      for (const commit of commits) {      for (const commit of commits) {
325          const key = `${commit.createdAt          const key = `${commit.date.getUTCDate().toString().padStart(2, 0)}-${(
326              .getUTCDate()              commit.date.getUTCMonth() + 1
327              .toString()          )
             .padStart(2, 0)}-${(commit.createdAt.getUTCMonth() + 1)  
328              .toString()              .toString()
329              .padStart(2, "0")}-${commit.createdAt.getUTCFullYear()}::${              .padStart(
330              Array.isArray(commit.author)                  2,
331                  ? commit.author.join(":")                  "0",
332                  : commit.author              )}-${commit.date.getUTCFullYear()}::${commit.authors.map((a) => `${a.name} <${a.email}>`).join(":")}`;
         }`;  
333          grouppedCommitsByDate[key] ??= [];          grouppedCommitsByDate[key] ??= [];
334          grouppedCommitsByDate[key].push(commit);          grouppedCommitsByDate[key].push(commit);
335      }      }
# Line 223  function generateChangelog(commits) { Line 339  function generateChangelog(commits) {
339          const date = key.slice(0, separatorPosition);          const date = key.slice(0, separatorPosition);
340          const commits = grouppedCommitsByDate[key];          const commits = grouppedCommitsByDate[key];
341    
342          output += `${date}  ${          output += `${date}  ${commits[0].authors
343              Array.isArray(commits[0].author)              .map((a) => `${a.name} <${a.email}>`)
344                  ? commits[0].author.join(", ")              .join(", ")}\n\n`;
                 : commits[0].author  
         }\n\n`;  
345    
346          for (const commit of commits) {          for (const commit of commits) {
347              output += `        ${commit.message.replaceAll(              output += `     [*] ${commit.subject}\n${commit.body
348                  "\n",                  .split("\n")
349                  "\n        "                  .map((part) => `         ${part}`)
350              )}\n\n`;                  .join("\n")}\n\n`;
351          }          }
352      }      }
353    
# Line 248  function printHelp() { Line 362  function printHelp() {
362      console.log("Options:");      console.log("Options:");
363      console.log("  -h, --help           Show this help and exit.");      console.log("  -h, --help           Show this help and exit.");
364      console.log("  -v, --version        Show this script's version.");      console.log("  -v, --version        Show this script's version.");
365      console.log("  -f, --format=        Set the changelog format.");      console.log("  -f, --format         Set the changelog format.");
366      console.log("                       Supported formats are: plain,");      console.log("                       Supported formats are: plain,");
367      console.log("                       markdown.");      console.log("                       markdown.");
368      console.log("  -o, --output=<FILE>  Write the generated changelog to");      console.log("  -o, --output=<FILE>  Write the generated changelog to");
# Line 263  function printHelp() { Line 377  function printHelp() {
377  function printVersion() {  function printVersion() {
378      console.log("Copyright (C) 2024 OSN, Inc.");      console.log("Copyright (C) 2024 OSN, Inc.");
379      console.log(      console.log(
380          "License GPLv3+: GNU GPL version 3 or later <https://gnu.org/licenses/gpl.html>."          "License GPLv3+: GNU GPL version 3 or later <https://gnu.org/licenses/gpl.html>.",
381      );      );
382      console.log(      console.log(
383          "This is free software: you are free to change and redistribute it."          "This is free software: you are free to change and redistribute it.",
384      );      );
385      console.log("There is NO WARRANTY, to the extent permitted by law.");      console.log("There is NO WARRANTY, to the extent permitted by law.");
386      console.log();      console.log();
# Line 299  async function main() { Line 413  async function main() {
413                  },                  },
414                  format: {                  format: {
415                      type: "string",                      type: "string",
416                      short: "f"                      short: "f",
417                  }                  },
418              },              },
419          }).values;          }).values;
420      } catch (error) {      } catch (error) {
# Line 319  async function main() { Line 433  async function main() {
433      }      }
434    
435      if (options.format && !["markdown", "plain"].includes(options.format)) {      if (options.format && !["markdown", "plain"].includes(options.format)) {
436          perror("option `--format` or `-f` only accepts one of the following: markdown, plain");          perror(
437                "option `--format` or `-f` only accepts one of the following: markdown, plain",
438            );
439          exit(1);          exit(1);
440      }      }
441    
442      if (!options.output && options["no-overwrite"]) {      if (!options.output && options["no-overwrite"]) {
443          perror(          perror(
444              "option `--no-overwrite' without `--output` does not make sense"              "option `--no-overwrite' without `--output` does not make sense",
445          );          );
446          exit(1);          exit(1);
447      }      }
# Line 340  async function main() { Line 456  async function main() {
456      }      }
457    
458      const gitPath = checkForGit();      const gitPath = checkForGit();
459      const gitLog = getGitLog(gitPath);      const gitLog = await getGitLog(gitPath);
460      const commits = parseGitLog(gitLog);      const commits = parseGitLog(gitLog);
461      const filteredCommits = commits.filter(      const filteredCommits = commits.filter(
462          (commit) =>          (commit) =>
463              !/Merge pull request #\d+ from|Merge branch '\S+' of/.test(              !/Merge pull request #\d+ from|Merge branch '\S+' of/.test(
464                  commit.message                  commit.message,
465              )              ),
466      );      );
467      const changelog = options.format === "markdown" ? generateMarkdownChangelog(filteredCommits) : generateChangelog(filteredCommits);      const changelog =
468            options.format === "markdown"
469                ? generateMarkdownChangelog(filteredCommits)
470                : generateChangelog(filteredCommits);
471    
472      if (options.output) {      if (options.output) {
473          try {          try {
# Line 357  async function main() { Line 476  async function main() {
476              perror(              perror(
477                  `${options.output}: failed to write changelog: ${                  `${options.output}: failed to write changelog: ${
478                      error?.message ?? error                      error?.message ?? error
479                  }`                  }`,
480              );              );
481              exit(1);              exit(1);
482          }          }

Legend:
Removed from v.54  
changed lines
  Added in v.56

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26