2020-12-23 14:39:46 +01:00
|
|
|
'use strict';
|
2020-12-23 19:28:23 +01:00
|
|
|
|
2021-01-18 09:53:15 +01:00
|
|
|
// As of v14, Node.js does not exit when there is an unhandled Promise rejection. Convert an
|
|
|
|
// unhandled rejection into an uncaught exception, which does cause Node.js to exit.
|
|
|
|
process.on('unhandledRejection', (err) => { throw err; });
|
|
|
|
|
2024-03-13 20:31:29 +01:00
|
|
|
import fs from 'fs';
|
|
|
|
import childProcess from 'child_process';
|
|
|
|
import log4js from 'log4js';
|
|
|
|
import path from 'path';
|
|
|
|
import semver from 'semver';
|
|
|
|
import {exec} from 'child_process';
|
2020-12-23 14:39:46 +01:00
|
|
|
|
2023-10-28 16:30:41 +02:00
|
|
|
log4js.configure({appenders: {console: {type: 'console'}},
|
|
|
|
categories: {
|
|
|
|
default: {appenders: ['console'], level: 'info'},
|
|
|
|
}});
|
2021-02-15 23:28:15 +01:00
|
|
|
|
2020-12-23 14:39:46 +01:00
|
|
|
/*
|
|
|
|
|
|
|
|
Usage
|
|
|
|
|
2024-02-21 21:50:11 +01:00
|
|
|
node bin/release.js patch
|
2020-12-23 14:39:46 +01:00
|
|
|
|
|
|
|
*/
|
2021-02-05 00:43:27 +01:00
|
|
|
const usage =
|
2024-02-21 21:50:11 +01:00
|
|
|
'node bin/release.js [patch/minor/major] -- example: "node bin/release.js patch"';
|
2020-12-23 14:39:46 +01:00
|
|
|
|
|
|
|
const release = process.argv[2];
|
|
|
|
|
2021-01-18 09:53:15 +01:00
|
|
|
if (!release) {
|
2020-12-23 14:39:46 +01:00
|
|
|
console.log(usage);
|
|
|
|
throw new Error('No release type included');
|
|
|
|
}
|
|
|
|
|
2024-03-13 20:31:29 +01:00
|
|
|
if (release !== 'patch' && release !== 'minor' && release !== 'major') {
|
|
|
|
console.log(usage);
|
|
|
|
throw new Error('Invalid release type');
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-02-21 21:50:11 +01:00
|
|
|
const cwd = path.join(fs.realpathSync(__dirname), '../');
|
2021-02-15 21:47:47 +01:00
|
|
|
process.chdir(cwd);
|
|
|
|
|
2021-02-15 21:54:21 +01:00
|
|
|
// Run command capturing stdout. Trailing newlines are stripped (like the shell does).
|
|
|
|
const runc =
|
2024-03-13 20:31:29 +01:00
|
|
|
(cmd:string, opts = {}) => childProcess.execSync(cmd, {encoding: 'utf8', ...opts}).replace(/\n+$/, '');
|
2021-02-15 22:07:37 +01:00
|
|
|
// Run command without capturing stdout.
|
2024-03-13 20:31:29 +01:00
|
|
|
const run = (cmd: string, opts = {}) => childProcess.execSync(cmd, {stdio: 'inherit', ...opts});
|
2021-02-15 21:51:07 +01:00
|
|
|
|
2024-03-13 20:31:29 +01:00
|
|
|
const readJson = (filename: string) => JSON.parse(fs.readFileSync(filename, {encoding: 'utf8', flag: 'r'}));
|
|
|
|
const writeJson = (filename: string, obj:object) => {
|
2021-02-15 20:04:47 +01:00
|
|
|
let json = JSON.stringify(obj, null, 2);
|
|
|
|
if (json !== '' && !json.endsWith('\n')) json += '\n';
|
|
|
|
fs.writeFileSync(filename, json);
|
|
|
|
};
|
2021-02-15 19:56:00 +01:00
|
|
|
|
2024-03-13 20:31:29 +01:00
|
|
|
const assertWorkDirClean = (opts:{
|
|
|
|
cwd?: string;
|
|
|
|
} = {}) => {
|
2021-02-15 21:54:21 +01:00
|
|
|
opts.cwd = runc('git rev-parse --show-cdup', opts) || cwd;
|
|
|
|
const m = runc('git diff-files --name-status', opts);
|
|
|
|
if (m !== '') throw new Error(`modifications in working directory ${opts.cwd}:\n${m}`);
|
|
|
|
const u = runc('git ls-files -o --exclude-standard', opts);
|
|
|
|
if (u !== '') throw new Error(`untracked files in working directory ${opts.cwd}:\n${u}`);
|
|
|
|
const s = runc('git diff-index --cached --name-status HEAD', opts);
|
|
|
|
if (s !== '') throw new Error(`uncommitted changes in working directory ${opts.cwd}:\n${s}`);
|
|
|
|
};
|
|
|
|
|
2024-03-13 20:31:29 +01:00
|
|
|
const assertBranchCheckedOut = (branch: string, opts:{
|
|
|
|
cwd?: string;
|
|
|
|
} = {}) => {
|
2021-02-15 21:54:21 +01:00
|
|
|
const b = runc('git symbolic-ref HEAD', opts);
|
|
|
|
if (b !== `refs/heads/${branch}`) {
|
|
|
|
const d = opts.cwd ? path.resolve(cwd, opts.cwd) : cwd;
|
|
|
|
throw new Error(`${branch} must be checked out (cwd: ${d})`);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-03-13 20:31:29 +01:00
|
|
|
const assertUpstreamOk = (branch: string, opts:{
|
|
|
|
cwd?: string;
|
|
|
|
} = {}) => {
|
2021-02-15 21:54:21 +01:00
|
|
|
const upstream = runc(`git rev-parse --symbolic-full-name ${branch}@{u}`, opts);
|
|
|
|
if (!(new RegExp(`^refs/remotes/[^/]+/${branch}`)).test(upstream)) {
|
|
|
|
throw new Error(`${branch} should track origin/${branch}; see git branch --set-upstream-to`);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
run(`git merge-base --is-ancestor ${branch} ${branch}@{u}`);
|
2024-03-13 20:31:29 +01:00
|
|
|
} catch (err:any) {
|
2021-02-15 21:54:21 +01:00
|
|
|
if (err.status !== 1) throw err;
|
|
|
|
throw new Error(`${branch} is ahead of origin/${branch}; do you need to push?`);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-06-21 14:13:31 +02:00
|
|
|
// Check if asciidoctor is installed
|
2024-03-13 20:31:29 +01:00
|
|
|
exec('asciidoctor -v', (err) => {
|
2023-10-28 16:30:41 +02:00
|
|
|
if (err) {
|
|
|
|
console.log('Please install asciidoctor');
|
|
|
|
console.log('https://asciidoctor.org/docs/install-toolchain/');
|
|
|
|
process.exit(1);
|
2023-06-21 14:13:31 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2024-03-13 20:31:29 +01:00
|
|
|
const dirExists = (dir: string) => {
|
2021-02-15 21:54:21 +01:00
|
|
|
try {
|
|
|
|
return fs.statSync(dir).isDirectory();
|
2024-03-13 20:31:29 +01:00
|
|
|
} catch (err:any) {
|
2021-02-15 21:54:21 +01:00
|
|
|
if (err.code !== 'ENOENT') throw err;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Sanity checks for Etherpad repo.
|
|
|
|
assertWorkDirClean();
|
|
|
|
assertBranchCheckedOut('develop');
|
|
|
|
assertUpstreamOk('develop');
|
|
|
|
assertUpstreamOk('master');
|
|
|
|
|
|
|
|
// Sanity checks for documentation repo.
|
|
|
|
if (!dirExists('../ether.github.com')) {
|
|
|
|
throw new Error('please clone documentation repo: ' +
|
|
|
|
'(cd .. && git clone git@github.com:ether/ether.github.com.git)');
|
|
|
|
}
|
|
|
|
assertWorkDirClean({cwd: '../ether.github.com/'});
|
|
|
|
assertBranchCheckedOut('master', {cwd: '../ether.github.com/'});
|
|
|
|
assertUpstreamOk('master', {cwd: '../ether.github.com/'});
|
|
|
|
|
2020-12-23 14:39:46 +01:00
|
|
|
const changelog = fs.readFileSync('CHANGELOG.md', {encoding: 'utf8', flag: 'r'});
|
2021-02-15 19:56:00 +01:00
|
|
|
const pkg = readJson('./src/package.json');
|
|
|
|
const currentVersion = pkg.version;
|
2020-12-23 14:39:46 +01:00
|
|
|
|
|
|
|
const newVersion = semver.inc(currentVersion, release);
|
2021-01-18 09:53:15 +01:00
|
|
|
if (!newVersion) {
|
2020-12-23 14:39:46 +01:00
|
|
|
console.log(usage);
|
|
|
|
throw new Error('Unable to generate new version from input');
|
|
|
|
}
|
|
|
|
|
2021-02-15 22:48:28 +01:00
|
|
|
if (!changelog.startsWith(`# ${newVersion}\n`)) {
|
2021-02-07 20:27:10 +01:00
|
|
|
throw new Error(`No changelog record for ${newVersion}, please create changelog record`);
|
2020-12-23 14:39:46 +01:00
|
|
|
}
|
|
|
|
|
2021-02-15 21:55:37 +01:00
|
|
|
// ////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Done with sanity checks, now it's time to make changes.
|
2020-12-23 14:39:46 +01:00
|
|
|
|
2021-02-15 23:19:17 +01:00
|
|
|
try {
|
|
|
|
console.log('Updating develop branch...');
|
|
|
|
run('git pull --ff-only');
|
2021-02-15 21:55:37 +01:00
|
|
|
|
2021-02-15 23:19:17 +01:00
|
|
|
console.log(`Bumping ${release} version (to ${newVersion})...`);
|
|
|
|
pkg.version = newVersion;
|
2020-12-23 14:39:46 +01:00
|
|
|
|
2021-02-15 23:19:17 +01:00
|
|
|
writeJson('./src/package.json', pkg);
|
2020-12-23 14:39:46 +01:00
|
|
|
|
2021-02-15 23:19:17 +01:00
|
|
|
// run npm version `release` where release is patch, minor or major
|
|
|
|
run('npm install --package-lock-only', {cwd: 'src/'});
|
|
|
|
// run npm install --package-lock-only <-- required???
|
2020-12-23 14:39:46 +01:00
|
|
|
|
2021-02-15 23:19:17 +01:00
|
|
|
// Many users will be using the latest LTS version of npm, and the latest LTS version of npm uses
|
|
|
|
// lockfileVersion 1. Enforce v1 so that users don't see a (benign) compatibility warning.
|
2023-06-20 16:44:40 +02:00
|
|
|
const pkglock = readJson('./src/package-lock.json');
|
|
|
|
pkglock.lockfileVersion = 1;
|
|
|
|
writeJson('./src/package-lock.json', pkglock);
|
2021-02-15 23:19:17 +01:00
|
|
|
|
|
|
|
run('git add src/package.json');
|
|
|
|
run('git add src/package-lock.json');
|
|
|
|
run('git commit -m "bump version"');
|
|
|
|
console.log('Switching to master...');
|
|
|
|
run('git checkout master');
|
|
|
|
console.log('Updating master branch...');
|
|
|
|
run('git pull --ff-only');
|
|
|
|
console.log('Merging develop into master...');
|
|
|
|
run('git merge --no-ff --no-edit develop');
|
|
|
|
console.log(`Creating ${newVersion} tag...`);
|
|
|
|
run(`git tag -s '${newVersion}' -m '${newVersion}'`);
|
2023-12-11 20:09:45 +01:00
|
|
|
run(`git tag -s 'v${newVersion}' -m 'v${newVersion}'`);
|
2021-02-15 23:19:17 +01:00
|
|
|
console.log('Switching back to develop...');
|
|
|
|
run('git checkout develop');
|
|
|
|
console.log('Merging master into develop...');
|
|
|
|
run('git merge --no-ff --no-edit master');
|
2024-03-13 20:31:29 +01:00
|
|
|
} catch (err:any) {
|
2021-02-15 23:19:17 +01:00
|
|
|
console.error(err.toString());
|
|
|
|
console.warn('Resetting repository...');
|
|
|
|
console.warn('Resetting master...');
|
|
|
|
run('git checkout -f master');
|
|
|
|
run('git reset --hard @{u}');
|
|
|
|
console.warn('Resetting develop...');
|
|
|
|
run('git checkout -f develop');
|
|
|
|
run('git reset --hard @{u}');
|
|
|
|
console.warn(`Deleting ${newVersion} tag...`);
|
|
|
|
run(`git rev-parse -q --verify refs/tags/'${newVersion}' >/dev/null || exit 0; ` +
|
|
|
|
`git tag -d '${newVersion}'`);
|
2023-12-11 20:16:43 +01:00
|
|
|
run(`git rev-parse -q --verify refs/tags/'v${newVersion}' >/dev/null || exit 0; ` +
|
|
|
|
`git tag -d 'v${newVersion}'`);
|
2021-02-15 23:19:17 +01:00
|
|
|
throw err;
|
2021-02-15 19:56:41 +01:00
|
|
|
}
|
|
|
|
|
2021-02-15 23:19:17 +01:00
|
|
|
try {
|
|
|
|
console.log('Building documentation...');
|
2023-06-21 14:13:31 +02:00
|
|
|
run('node ./make_docs.js');
|
2021-02-15 23:19:17 +01:00
|
|
|
console.log('Updating ether.github.com master branch...');
|
|
|
|
run('git pull --ff-only', {cwd: '../ether.github.com/'});
|
|
|
|
console.log('Committing documentation...');
|
2023-09-04 21:58:08 +02:00
|
|
|
run(`cp -R out/doc/ ../ether.github.com/public/doc/v'${newVersion}'`);
|
2023-10-28 16:30:41 +02:00
|
|
|
run(`npm version ${newVersion}`, {cwd: '../ether.github.com'});
|
2021-02-15 23:19:17 +01:00
|
|
|
run('git add .', {cwd: '../ether.github.com/'});
|
|
|
|
run(`git commit -m '${newVersion} docs'`, {cwd: '../ether.github.com/'});
|
2024-03-13 20:31:29 +01:00
|
|
|
} catch (err:any) {
|
2021-02-15 23:19:17 +01:00
|
|
|
console.error(err.toString());
|
|
|
|
console.warn('Resetting repository...');
|
|
|
|
console.warn('Resetting master...');
|
|
|
|
run('git checkout -f master', {cwd: '../ether.github.com/'});
|
|
|
|
run('git reset --hard @{u}', {cwd: '../ether.github.com/'});
|
|
|
|
throw err;
|
|
|
|
}
|
2021-02-15 21:55:37 +01:00
|
|
|
|
|
|
|
console.log('Done.');
|
|
|
|
console.log('Review the new commits and the new tag:');
|
|
|
|
console.log(' git log --graph --date-order --boundary --oneline --decorate develop@{u}..develop');
|
|
|
|
console.log(` git show '${newVersion}'`);
|
|
|
|
console.log(' (cd ../ether.github.com && git show)');
|
|
|
|
console.log('If everything looks good then push:');
|
2023-12-24 20:36:56 +01:00
|
|
|
console.log('Run ./bin/push-after-release.sh');
|
2023-09-04 21:58:08 +02:00
|
|
|
console.log('Creating a Windows build is not necessary anymore and will be created by GitHub action');
|
2023-12-24 20:36:56 +01:00
|
|
|
console.log('After the windows binary is created a new release with the set version is created automatically.' +
|
|
|
|
' Just paste the release notes in there');
|
2023-09-04 21:58:08 +02:00
|
|
|
console.log('The docs are updated automatically with the new version. While the windows build' +
|
|
|
|
' is generated people can still download the older versions.');
|
2020-12-23 14:39:46 +01:00
|
|
|
console.log('Finally go public with an announcement via our comms channels :)');
|