electron/script/release/prepare-release.ts

265 lines
8.8 KiB
TypeScript
Raw Normal View History

#!/usr/bin/env node
import { Octokit } from '@octokit/rest';
import { GitProcess } from 'dugite';
import { execSync } from 'node:child_process';
import { join } from 'node:path';
import { createInterface } from 'node:readline';
import { parseArgs } from 'node:util';
import ciReleaseBuild from './ci-release-build';
import releaseNotesGenerator from './notes';
import { getCurrentBranch, ELECTRON_DIR } from '../lib/utils.js';
import { createGitHubTokenStrategy } from './github-token';
import { ELECTRON_REPO, ElectronReleaseRepo, NIGHTLY_REPO } from './types';
const { values: { notesOnly, dryRun: dryRunArg, stable: isStableArg, branch: branchArg, automaticRelease }, positionals } = parseArgs({
options: {
notesOnly: {
type: 'boolean'
},
dryRun: {
type: 'boolean'
},
stable: {
type: 'boolean'
},
branch: {
type: 'string'
},
automaticRelease: {
type: 'boolean'
}
},
allowPositionals: true
2020-03-20 20:28:31 +00:00
});
const bumpType = positionals[0];
const targetRepo = getRepo();
function getRepo (): ElectronReleaseRepo {
return bumpType === 'nightly' ? NIGHTLY_REPO : ELECTRON_REPO;
}
const octokit = new Octokit({
authStrategy: createGitHubTokenStrategy(getRepo())
});
2020-03-20 20:28:31 +00:00
require('colors');
const pass = '✓'.green;
const fail = '✗'.red;
if (!bumpType && !notesOnly) {
console.log('Usage: prepare-release [stable | minor | beta | alpha | nightly]' +
2020-03-20 20:28:31 +00:00
' (--stable) (--notesOnly) (--automaticRelease) (--branch)');
process.exit(1);
}
enum DryRunMode {
DRY_RUN,
REAL_RUN,
}
async function getNewVersion (dryRunMode: DryRunMode) {
if (dryRunMode === DryRunMode.REAL_RUN) {
2020-03-20 20:28:31 +00:00
console.log(`Bumping for new "${bumpType}" version.`);
}
const bumpScript = join(__dirname, 'version-bumper.ts');
const scriptArgs = [
'node',
'node_modules/.bin/ts-node',
bumpScript,
`--bump=${bumpType}`
];
if (dryRunMode === DryRunMode.DRY_RUN) scriptArgs.push('--dryRun');
try {
let bumpVersion = execSync(scriptArgs.join(' '), { encoding: 'utf-8' });
2020-03-20 20:28:31 +00:00
bumpVersion = bumpVersion.substr(bumpVersion.indexOf(':') + 1).trim();
const newVersion = `v${bumpVersion}`;
if (dryRunMode === DryRunMode.REAL_RUN) {
2020-03-20 20:28:31 +00:00
console.log(`${pass} Successfully bumped version to ${newVersion}`);
}
2020-03-20 20:28:31 +00:00
return newVersion;
} catch (err) {
2020-03-20 20:28:31 +00:00
console.log(`${fail} Could not bump version, error was:`, err);
throw err;
}
}
async function getReleaseNotes (currentBranch: string, newVersion: string) {
if (bumpType === 'nightly') {
2020-03-20 20:28:31 +00:00
return { text: 'Nightlies do not get release notes, please compare tags for info.' };
}
2020-03-20 20:28:31 +00:00
console.log(`Generating release notes for ${currentBranch}.`);
const releaseNotes = await releaseNotesGenerator(currentBranch, newVersion);
better release notes (#15169) * fix: use PR 'Notes' comment in release notes * fix: follow links in roller-bot PRs * refactor: better reference point version selection * if we're a stable release, use the current brnach's previous stable * if we're a beta release, use the current branch's previous beta * if no match found, use the newest stable that precedes this branch * refactor: dedup the caching functions' code * refactor: partially rewrite release note generator * parse release notes comments from PRs * do not display no-notes PRs * handle roller-bot commits by following cross-repo commits/PRs * minor tweaks to note rendering, e.g. capitalization * fix: fix lint:js script typo * fix: copy originalPr value to rollerbot PR chains * fix: handle more cases in release notes generator * handle force-pushes where no PR * better type guessing on pre-semantic commits * fix: handle more edge cases in the note generator * better removal of commits that landed before the reference point * ensure '<!-- One-line Change Summary Here-->' is removed from notes * handle more legacy commit body notes e.g. "Chore(docs)" * check for fix markdown in PR body e.g. a link to the issue page * chore: tweak code comments * refactor: easier note generator command-line args * refactor: group related notes together * feat: query commits locally for gyp and gn deps * chore: slightly better filtering of old commits * feat: omit submodule commits for .0.0 releases More specifically, only include them if generating release notes relative to another release on the same branch. Before that first release, there's just too much churn. * refactor: make release-notes usable as a module Calling it from the command line and from require()() now do pretty much the same thing. * refactor: passing command-line args means use HEAD * chore: plug in the release note generator * feat: support multiline 'Notes:' messages. xref: https://github.com/electron/trop/pull/56 xref: https://github.com/electron/clerk/pull/16 * remove accidental change in package.json * simplify an overcomplicated require() call * Don't use PascalCase on releaseNotesGenerator() * Remove code duplication in release notes warnings * remove commented-out code. * don't use single-character variable names. For example, use 'tag' instead of 't'. The latter was being used for map/filter arrow function args. * Look for 'backport' rather than 'ackport'. * Wrap all block statements in curly braces. * fix tyop * fix oops * Check semver validity before calling semver.sort()
2018-11-06 20:06:11 +00:00
if (releaseNotes.warning) {
2020-03-20 20:28:31 +00:00
console.warn(releaseNotes.warning);
}
2020-03-20 20:28:31 +00:00
return releaseNotes;
}
async function createRelease (branchToTarget: string, isPreRelease: boolean) {
const newVersion = await getNewVersion(DryRunMode.REAL_RUN);
2020-03-20 20:28:31 +00:00
const releaseNotes = await getReleaseNotes(branchToTarget, newVersion);
await tagRelease(newVersion);
2020-03-20 20:28:31 +00:00
console.log('Checking for existing draft release.');
const releases = await octokit.repos.listReleases({
owner: 'electron',
repo: targetRepo
}).catch(err => {
2020-03-20 20:28:31 +00:00
console.log(`${fail} Could not get releases. Error was: `, err);
throw err;
2020-03-20 20:28:31 +00:00
});
const drafts = releases.data.filter(release => release.draft &&
2020-03-20 20:28:31 +00:00
release.tag_name === newVersion);
if (drafts.length > 0) {
console.log(`${fail} Aborting because draft release for
2020-03-20 20:28:31 +00:00
${drafts[0].tag_name} already exists.`);
process.exit(1);
}
2020-03-20 20:28:31 +00:00
console.log(`${pass} A draft release does not exist; creating one.`);
2020-03-20 20:28:31 +00:00
let releaseBody;
let releaseIsPrelease = false;
if (isPreRelease) {
if (newVersion.indexOf('nightly') > 0) {
releaseBody = 'Note: This is a nightly release. Please file new issues ' +
'for any bugs you find in it.\n \n This release is published to npm ' +
'under the electron-nightly package and can be installed via `npm install electron-nightly`, ' +
`or \`npm install electron-nightly@${newVersion.substr(1)}\`.\n \n ${releaseNotes.text}`;
} else if (newVersion.indexOf('alpha') > 0) {
releaseBody = 'Note: This is an alpha release. Please file new issues ' +
'for any bugs you find in it.\n \n This release is published to npm ' +
'under the alpha tag and can be installed via `npm install electron@alpha`, ' +
`or \`npm install electron@${newVersion.substr(1)}\`.\n \n ${releaseNotes.text}`;
} else {
releaseBody = 'Note: This is a beta release. Please file new issues ' +
'for any bugs you find in it.\n \n This release is published to npm ' +
'under the beta tag and can be installed via `npm install electron@beta`, ' +
`or \`npm install electron@${newVersion.substr(1)}\`.\n \n ${releaseNotes.text}`;
}
2020-03-20 20:28:31 +00:00
releaseIsPrelease = true;
} else {
2020-03-20 20:28:31 +00:00
releaseBody = releaseNotes.text;
}
const release = await octokit.repos.createRelease({
owner: 'electron',
repo: targetRepo,
tag_name: newVersion,
draft: true,
name: `electron ${newVersion}`,
body: releaseBody,
prerelease: releaseIsPrelease,
target_commitish: newVersion.includes('nightly') ? 'main' : branchToTarget
}).catch(err => {
2020-03-20 20:28:31 +00:00
console.log(`${fail} Error creating new release: `, err);
process.exit(1);
});
2020-03-20 20:28:31 +00:00
console.log(`Release has been created with id: ${release.data.id}.`);
console.log(`${pass} Draft release for ${newVersion} successful.`);
}
async function pushRelease (branch: string) {
2020-03-20 20:28:31 +00:00
const pushDetails = await GitProcess.exec(['push', 'origin', `HEAD:${branch}`, '--follow-tags'], ELECTRON_DIR);
if (pushDetails.exitCode === 0) {
console.log(`${pass} Successfully pushed the release. Wait for ` +
2020-03-20 20:28:31 +00:00
'release builds to finish before running "npm run release".');
} else {
2020-03-20 20:28:31 +00:00
console.log(`${fail} Error pushing the release: ${pushDetails.stderr}`);
process.exit(1);
}
}
async function runReleaseBuilds (branch: string, newVersion: string) {
await ciReleaseBuild(branch, {
ci: undefined,
ghRelease: true,
newVersion
2020-03-20 20:28:31 +00:00
});
}
async function tagRelease (version: string) {
2020-03-20 20:28:31 +00:00
console.log(`Tagging release ${version}.`);
const checkoutDetails = await GitProcess.exec(['tag', '-a', '-m', version, version], ELECTRON_DIR);
if (checkoutDetails.exitCode === 0) {
2020-03-20 20:28:31 +00:00
console.log(`${pass} Successfully tagged ${version}.`);
} else {
console.log(`${fail} Error tagging ${version}: ` +
2020-03-20 20:28:31 +00:00
`${checkoutDetails.stderr}`);
process.exit(1);
}
}
async function verifyNewVersion () {
const newVersion = await getNewVersion(DryRunMode.DRY_RUN);
2020-03-20 20:28:31 +00:00
let response;
if (automaticRelease) {
2020-03-20 20:28:31 +00:00
response = 'y';
} else {
2020-03-20 20:28:31 +00:00
response = await promptForVersion(newVersion);
}
if (response.match(/^y/i)) {
2020-03-20 20:28:31 +00:00
console.log(`${pass} Starting release of ${newVersion}`);
} else {
2020-03-20 20:28:31 +00:00
console.log(`${fail} Aborting release of ${newVersion}`);
process.exit();
}
return newVersion;
}
async function promptForVersion (version: string) {
return new Promise<string>(resolve => {
const rl = createInterface({
input: process.stdin,
output: process.stdout
2020-03-20 20:28:31 +00:00
});
rl.question(`Do you want to create the release ${version.green} (y/N)? `, (answer) => {
2020-03-20 20:28:31 +00:00
rl.close();
resolve(answer);
});
});
}
// function to determine if there have been commits to main since the last release
async function changesToRelease () {
const lastCommitWasRelease = /^Bump v[0-9]+.[0-9]+.[0-9]+(-beta.[0-9]+)?(-alpha.[0-9]+)?(-nightly.[0-9]+)?$/g;
2020-03-20 20:28:31 +00:00
const lastCommit = await GitProcess.exec(['log', '-n', '1', '--pretty=format:\'%s\''], ELECTRON_DIR);
return !lastCommitWasRelease.test(lastCommit.stdout);
}
async function prepareRelease (isPreRelease: boolean, dryRunMode: DryRunMode) {
if (dryRunMode === DryRunMode.DRY_RUN) {
const newVersion = await getNewVersion(DryRunMode.DRY_RUN);
2020-03-20 20:28:31 +00:00
console.log(newVersion);
} else {
const currentBranch = branchArg || await getCurrentBranch(ELECTRON_DIR);
if (notesOnly) {
const newVersion = await getNewVersion(DryRunMode.DRY_RUN);
2020-03-20 20:28:31 +00:00
const releaseNotes = await getReleaseNotes(currentBranch, newVersion);
console.log(`Draft release notes are: \n${releaseNotes.text}`);
} else {
const changes = await changesToRelease();
if (changes) {
const newVersion = await verifyNewVersion();
await createRelease(currentBranch, isPreRelease);
2020-03-20 20:28:31 +00:00
await pushRelease(currentBranch);
await runReleaseBuilds(currentBranch, newVersion);
} else {
2020-03-20 20:28:31 +00:00
console.log('There are no new changes to this branch since the last release, aborting release.');
process.exit(1);
}
}
}
}
prepareRelease(!isStableArg, dryRunArg ? DryRunMode.DRY_RUN : DryRunMode.REAL_RUN)
.catch((err) => {
console.error(err);
process.exit(1);
});