#!/usr/bin/env node 'use strict'; const childProcess = require('child_process'); const fs = require('fs'); const os = require('os'); const path = require('path'); const { GitProcess } = require('dugite'); const octokit = require('@octokit/rest')({ auth: process.env.ELECTRON_GITHUB_TOKEN }); const semver = require('semver'); const { ELECTRON_VERSION, SRC_DIR } = require('../../lib/utils'); const MAX_FAIL_COUNT = 3; const CHECK_INTERVAL = 5000; const NO_NOTES = 'No notes'; const FOLLOW_REPOS = ['electron/electron', 'electron/node']; const docTypes = new Set(['doc', 'docs']); const featTypes = new Set(['feat', 'feature']); const fixTypes = new Set(['fix']); const otherTypes = new Set(['spec', 'build', 'test', 'chore', 'deps', 'refactor', 'tools', 'vendor', 'perf', 'style', 'ci']); const knownTypes = new Set([...docTypes.keys(), ...featTypes.keys(), ...fixTypes.keys(), ...otherTypes.keys()]); const getCacheDir = () => process.env.NOTES_CACHE_PATH || path.resolve(__dirname, '.cache'); /** *** **/ // link to a GitHub item, e.g. an issue or pull request class GHKey { constructor (owner, repo, number) { this.owner = owner; this.repo = repo; this.number = number; } static NewFromPull (pull) { const owner = pull.base.repo.owner.login; const repo = pull.base.repo.name; const number = pull.number; return new GHKey(owner, repo, number); } } class Commit { constructor (hash, owner, repo) { this.hash = hash; // string this.owner = owner; // string this.repo = repo; // string this.isBreakingChange = false; this.note = null; // string this.prKeys = new Set(); // GHKey this.revertHash = null; // string this.semanticType = null; // string this.subject = null; // string } } class Pool { constructor () { this.commits = []; // Array this.processedHashes = new Set(); this.pulls = {}; // GHKey.number => octokit pull object } } /** *** **/ const runGit = async (dir, args) => { const response = await GitProcess.exec(args, dir); if (response.exitCode !== 0) { throw new Error(response.stderr.trim()); } return response.stdout.trim(); }; const getCommonAncestor = async (dir, point1, point2) => { return runGit(dir, ['merge-base', point1, point2]); }; const getNoteFromClerk = async (ghKey) => { const comments = await getComments(ghKey); if (!comments || !comments.data) return; const CLERK_LOGIN = 'release-clerk[bot]'; const CLERK_NO_NOTES = '**No Release Notes**'; const PERSIST_LEAD = '**Release Notes Persisted**\n\n'; const QUOTE_LEAD = '> '; for (const comment of comments.data.reverse()) { if (comment.user.login !== CLERK_LOGIN) { continue; } if (comment.body === CLERK_NO_NOTES) { return NO_NOTES; } if (comment.body.startsWith(PERSIST_LEAD)) { return comment.body .slice(PERSIST_LEAD.length).trim() // remove PERSIST_LEAD .split('\r?\n') // break into lines .map(line => line.trim()) .filter(line => line.startsWith(QUOTE_LEAD)) // notes are quoted .map(line => line.slice(QUOTE_LEAD.length)) // unquote the lines .join(' ') // join the note lines .trim(); } } }; // copied from https://github.com/electron/clerk/blob/master/src/index.ts#L4-L13 const OMIT_FROM_RELEASE_NOTES_KEYS = [ 'no-notes', 'no notes', 'no_notes', 'none', 'no', 'nothing', 'empty', 'blank' ]; /** * Looks for our project's conventions in the commit message: * * 'semantic: some description' -- sets semanticType, subject * 'some description (#99999)' -- sets subject, pr * 'Merge pull request #99999 from ${branchname}' -- sets pr * 'This reverts commit ${sha}' -- sets revertHash * line starting with 'BREAKING CHANGE' in body -- sets isBreakingChange * 'Backport of #99999' -- sets pr */ const parseCommitMessage = (commitMessage, commit) => { const { owner, repo } = commit; // split commitMessage into subject & body let subject = commitMessage; let body = ''; const pos = subject.indexOf('\n'); if (pos !== -1) { body = subject.slice(pos).trim(); subject = subject.slice(0, pos).trim(); } // if the subject ends in ' (#dddd)', treat it as a pull request id let match; if ((match = subject.match(/^(.*)\s\(#(\d+)\)$/))) { commit.prKeys.add(new GHKey(owner, repo, parseInt(match[2]))); subject = match[1]; } // if the subject begins with 'word:', treat it as a semantic commit if ((match = subject.match(/^(\w+):\s(.*)$/))) { const semanticType = match[1].toLocaleLowerCase(); if (knownTypes.has(semanticType)) { commit.semanticType = semanticType; subject = match[2]; } } // Check for GitHub commit message that indicates a PR if ((match = subject.match(/^Merge pull request #(\d+) from (.*)$/))) { commit.prKeys.add(new GHKey(owner, repo, parseInt(match[1]))); } // Check for a comment that indicates a PR const backportPattern = /(?:^|\n)(?:manual |manually )?backport.*(?:#(\d+)|\/pull\/(\d+))/im; if ((match = commitMessage.match(backportPattern))) { // This might be the first or second capture group depending on if it's a link or not. const backportNumber = match[1] ? parseInt(match[1], 10) : parseInt(match[2], 10); commit.prKeys.add(new GHKey(owner, repo, backportNumber)); } // https://help.github.com/articles/closing-issues-using-keywords/ if ((match = body.match(/\b(?:close|closes|closed|fix|fixes|fixed|resolve|resolves|resolved|for)\s#(\d+)\b/i))) { commit.semanticType = commit.semanticType || 'fix'; } // https://www.conventionalcommits.org/en if (commitMessage .split(/\r?\n/) // split into lines .map(line => line.trim()) .some(line => line.startsWith('BREAKING CHANGE'))) { commit.isBreakingChange = true; } // Check for a reversion commit if ((match = body.match(/This reverts commit ([a-f0-9]{40})\./))) { commit.revertHash = match[1]; } commit.subject = subject.trim(); return commit; }; const parsePullText = (pull, commit) => parseCommitMessage(`${pull.data.title}\n\n${pull.data.body}`, commit); const getLocalCommitHashes = async (dir, ref) => { const args = ['log', '--format=%H', ref]; return (await runGit(dir, args)).split(/[\r\n]+/).map(hash => hash.trim()); }; // return an array of Commits const getLocalCommits = async (module, point1, point2) => { const { owner, repo, dir } = module; const fieldSep = ','; const format = ['%H', '%s'].join(fieldSep); const args = ['log', '--cherry-pick', '--right-only', '--first-parent', `--format=${format}`, `${point1}..${point2}`]; const logs = (await runGit(dir, args)).split(/[\r\n]+/).map(field => field.trim()); const commits = []; for (const log of logs) { if (!log) { continue; } const [hash, subject] = log.split(fieldSep, 2).map(field => field.trim()); commits.push(parseCommitMessage(subject, new Commit(hash, owner, repo))); } return commits; }; const checkCache = async (name, operation) => { const filename = path.resolve(getCacheDir(), name); if (fs.existsSync(filename)) { return JSON.parse(fs.readFileSync(filename, 'utf8')); } process.stdout.write('.'); const response = await operation(); if (response) { fs.writeFileSync(filename, JSON.stringify(response)); } return response; }; // helper function to add some resiliency to volatile GH api endpoints async function runRetryable (fn, maxRetries) { let lastError; for (let i = 0; i < maxRetries; i++) { try { return await fn(); } catch (error) { await new Promise((resolve, reject) => setTimeout(resolve, CHECK_INTERVAL)); lastError = error; } } // Silently eat 404s. // Silently eat 422s, which come from "No commit found for SHA" if (lastError.status !== 404 && lastError.status !== 422) throw lastError; } const getPullCacheFilename = ghKey => `${ghKey.owner}-${ghKey.repo}-pull-${ghKey.number}`; const getCommitPulls = async (owner, repo, hash) => { const name = `${owner}-${repo}-commit-${hash}`; const retryableFunc = () => octokit.repos.listPullRequestsAssociatedWithCommit({ owner, repo, commit_sha: hash }); let ret = await checkCache(name, () => runRetryable(retryableFunc, MAX_FAIL_COUNT)); // only merged pulls belong in release notes if (ret && ret.data) { ret.data = ret.data.filter(pull => pull.merged_at); } // cache the pulls if (ret && ret.data) { for (const pull of ret.data) { const cachefile = getPullCacheFilename(GHKey.NewFromPull(pull)); const payload = { ...ret, data: pull }; await checkCache(cachefile, () => payload); } } // ensure the return value has the expected structure, even on failure if (!ret || !ret.data) { ret = { data: [] }; } return ret; }; const getPullRequest = async (ghKey) => { const { number, owner, repo } = ghKey; const name = getPullCacheFilename(ghKey); const retryableFunc = () => octokit.pulls.get({ pull_number: number, owner, repo }); return checkCache(name, () => runRetryable(retryableFunc, MAX_FAIL_COUNT)); }; const getComments = async (ghKey) => { const { number, owner, repo } = ghKey; const name = `${owner}-${repo}-issue-${number}-comments`; const retryableFunc = () => octokit.issues.listComments({ issue_number: number, owner, repo, per_page: 100 }); return checkCache(name, () => runRetryable(retryableFunc, MAX_FAIL_COUNT)); }; const addRepoToPool = async (pool, repo, from, to) => { const commonAncestor = await getCommonAncestor(repo.dir, from, to); // mark the old branch's commits as old news for (const oldHash of await getLocalCommitHashes(repo.dir, from)) { pool.processedHashes.add(oldHash); } // get the new branch's commits and the pulls associated with them const commits = await getLocalCommits(repo, commonAncestor, to); for (const commit of commits) { const { owner, repo, hash } = commit; for (const pull of (await getCommitPulls(owner, repo, hash)).data) { commit.prKeys.add(GHKey.NewFromPull(pull)); } } pool.commits.push(...commits); // add the pulls for (const commit of commits) { let prKey; for (prKey of commit.prKeys.values()) { const pull = await getPullRequest(prKey); if (!pull || !pull.data) continue; // couldn't get it pool.pulls[prKey.number] = pull; parsePullText(pull, commit); } } }; /*** **** Other Repos ***/ // other repos - gn const getDepsVariable = async (ref, key) => { // get a copy of that reference point's DEPS file const deps = await runGit(ELECTRON_VERSION, ['show', `${ref}:DEPS`]); const filename = path.resolve(os.tmpdir(), 'DEPS'); fs.writeFileSync(filename, deps); // query the DEPS file const response = childProcess.spawnSync( 'gclient', ['getdep', '--deps-file', filename, '--var', key], { encoding: 'utf8' } ); // cleanup fs.unlinkSync(filename); return response.stdout.trim(); }; const getDependencyCommitsGN = async (pool, fromRef, toRef) => { const repos = [{ // just node owner: 'electron', repo: 'node', dir: path.resolve(SRC_DIR, 'third_party', 'electron_node'), deps_variable_name: 'node_version' }]; for (const repo of repos) { // the 'DEPS' file holds the dependency reference point const key = repo.deps_variable_name; const from = await getDepsVariable(fromRef, key); const to = await getDepsVariable(toRef, key); await addRepoToPool(pool, repo, from, to); } }; // Changes are interesting if they make a change relative to a previous // release in the same series. For example if you fix a Y.0.0 bug, that // should be included in the Y.0.1 notes even if it's also tropped back // to X.0.1. // // The phrase 'previous release' is important: if this is the first // prerelease or first stable release in a series, we omit previous // branches' changes. Otherwise we will have an overwhelmingly long // list of mostly-irrelevant changes. const shouldIncludeMultibranchChanges = (version) => { let show = true; if (semver.valid(version)) { const prerelease = semver.prerelease(version); show = prerelease ? parseInt(prerelease.pop()) > 1 : semver.patch(version) > 0; } return show; }; function getOldestMajorBranchOfPull (pull) { return pull.data.labels .map(label => label.name.match(/merged\/(\d+)-(\d+)-x/) || label.name.match(/merged\/(\d+)-x-y/)) .filter(label => !!label) .map(label => parseInt(label[1])) .filter(major => !!major) .sort() .shift(); } function getOldestMajorBranchOfCommit (commit, pool) { return [ ...commit.prKeys.values() ] .map(prKey => pool.pulls[prKey.number]) .filter(pull => !!pull) .map(pull => getOldestMajorBranchOfPull(pull)) .filter(major => !!major) .sort() .shift(); } function commitExistsBeforeMajor (commit, pool, major) { const firstAppearance = getOldestMajorBranchOfCommit(commit, pool); return firstAppearance && (firstAppearance < major); } /*** **** Main ***/ const getNotes = async (fromRef, toRef, newVersion) => { const cacheDir = getCacheDir(); if (!fs.existsSync(cacheDir)) { fs.mkdirSync(cacheDir); } const pool = new Pool(); // get the electron/electron commits const electron = { owner: 'electron', repo: 'electron', dir: path.resolve(SRC_DIR, 'electron') }; await addRepoToPool(pool, electron, fromRef, toRef); // Don't include submodules if comparing across major versions; // there's just too much churn otherwise. const includeDeps = semver.valid(fromRef) && semver.valid(toRef) && semver.major(fromRef) === semver.major(toRef); if (includeDeps) { await getDependencyCommitsGN(pool, fromRef, toRef); } // remove any old commits pool.commits = pool.commits.filter(commit => !pool.processedHashes.has(commit.hash)); // if a commmit _and_ revert occurred in the unprocessed set, skip them both for (const commit of pool.commits) { const revertHash = commit.revertHash; if (!revertHash) { continue; } const revert = pool.commits.find(commit => commit.hash === revertHash); if (!revert) { continue; } commit.note = NO_NOTES; revert.note = NO_NOTES; pool.processedHashes.add(commit.hash); pool.processedHashes.add(revertHash); } // ensure the commit has a note for (const commit of pool.commits) { for (const prKey of commit.prKeys.values()) { if (commit.note) { break; } commit.note = await getNoteFromClerk(prKey); } } // remove non-user-facing commits pool.commits = pool.commits .filter(commit => commit.note && (commit.note !== NO_NOTES)) .filter(commit => !((commit.note || commit.subject).match(/^[Bb]ump v\d+\.\d+\.\d+/))); if (!shouldIncludeMultibranchChanges(newVersion)) { const { major } = semver.parse(newVersion); pool.commits = pool.commits.filter(commit => !commitExistsBeforeMajor(commit, pool, major)); } pool.commits = removeSupercededStackUpdates(pool.commits); const notes = { breaking: [], docs: [], feat: [], fix: [], other: [], unknown: [], name: newVersion }; pool.commits.forEach(commit => { const str = commit.semanticType; if (commit.isBreakingChange) { notes.breaking.push(commit); } else if (!str) { notes.unknown.push(commit); } else if (docTypes.has(str)) { notes.docs.push(commit); } else if (featTypes.has(str)) { notes.feat.push(commit); } else if (fixTypes.has(str)) { notes.fix.push(commit); } else if (otherTypes.has(str)) { notes.other.push(commit); } else { notes.unknown.push(commit); } }); return notes; }; const removeSupercededStackUpdates = (commits) => { const updateRegex = /^Updated ([a-zA-Z.]+) to v?([\d.]+)/; const notupdates = []; const newest = {}; for (const commit of commits) { const match = (commit.note || commit.subject).match(updateRegex); if (!match) { notupdates.push(commit); continue; } const [ , dep, version ] = match; if (!newest[dep] || newest[dep].version < version) { newest[dep] = { commit, version }; } } return [ ...notupdates, ...Object.values(newest).map(o => o.commit) ]; }; /*** **** Render ***/ const renderLink = (commit, explicitLinks) => { let link; const { owner, repo } = commit; const keyIt = commit.prKeys.values().next(); if (keyIt.done) /* no PRs */ { const { hash } = commit; const url = `https://github.com/${owner}/${repo}/commit/${hash}`; const text = owner === 'electron' && repo === 'electron' ? `${hash.slice(0, 8)}` : `${owner}/${repo}@${hash.slice(0, 8)}`; link = explicitLinks ? `[${text}](${url})` : text; } else { const { number } = keyIt.value; const url = `https://github.com/${owner}/${repo}/pull/${number}`; const text = owner === 'electron' && repo === 'electron' ? `#${number}` : `${owner}/${repo}#${number}`; link = explicitLinks ? `[${text}](${url})` : text; } return link; }; const renderCommit = (commit, explicitLinks) => { // clean up the note let note = commit.note || commit.subject; note = note.trim(); if (note.length !== 0) { note = note[0].toUpperCase() + note.substr(1); if (!note.endsWith('.')) { note = note + '.'; } const commonVerbs = { Added: ['Add'], Backported: ['Backport'], Cleaned: ['Clean'], Disabled: ['Disable'], Ensured: ['Ensure'], Exported: ['Export'], Fixed: ['Fix', 'Fixes'], Handled: ['Handle'], Improved: ['Improve'], Made: ['Make'], Removed: ['Remove'], Repaired: ['Repair'], Reverted: ['Revert'], Stopped: ['Stop'], Updated: ['Update'], Upgraded: ['Upgrade'] }; for (const [key, values] of Object.entries(commonVerbs)) { for (const value of values) { const start = `${value} `; if (note.startsWith(start)) { note = `${key} ${note.slice(start.length)}`; } } } } const link = renderLink(commit, explicitLinks); return { note, link }; }; const renderNotes = (notes, explicitLinks) => { const rendered = [`# Release Notes for ${notes.name}\n\n`]; const renderSection = (title, commits) => { if (commits.length === 0) { return; } const notes = new Map(); for (const note of commits.map(commit => renderCommit(commit, explicitLinks))) { if (!notes.has(note.note)) { notes.set(note.note, [note.link]); } else { notes.get(note.note).push(note.link); } } rendered.push(`## ${title}\n\n`); const lines = []; notes.forEach((links, key) => lines.push(` * ${key} ${links.map(link => link.toString()).sort().join(', ')}\n`)); rendered.push(...lines.sort(), '\n'); }; renderSection('Breaking Changes', notes.breaking); renderSection('Features', notes.feat); renderSection('Fixes', notes.fix); renderSection('Other Changes', notes.other); if (notes.docs.length) { const docs = notes.docs.map(commit => renderLink(commit, explicitLinks)).sort(); rendered.push('## Documentation\n\n', ` * Documentation changes: ${docs.join(', ')}\n`, '\n'); } renderSection('Unknown', notes.unknown); return rendered.join(''); }; /*** **** Module ***/ module.exports = { get: getNotes, render: renderNotes };