gh-action-bump-version/index.js

169 lines
6.7 KiB
JavaScript

const { Toolkit } = require('actions-toolkit');
const { execSync } = require('child_process');
// Change working directory if user defined PACKAGEJSON_DIR
if (process.env.PACKAGEJSON_DIR) {
process.env.GITHUB_WORKSPACE = `${process.env.GITHUB_WORKSPACE}/${process.env.PACKAGEJSON_DIR}`;
process.chdir(process.env.GITHUB_WORKSPACE);
}
// Run your GitHub Action!
Toolkit.run(async (tools) => {
const pkg = tools.getPackageJSON();
const event = tools.context.payload;
if (!event.commits) {
console.log("Couldn't find any commits in this event, incrementing patch version...");
}
const messages = event.commits ? event.commits.map((commit) => commit.message + '\n' + commit.body) : [];
const commitMessage = process.env['INPUT_COMMIT-MESSAGE'] || 'ci: version bump to {{version}}';
console.log('messages:', messages);
const commitMessageRegex = new RegExp(commitMessage.replace(/{{version}}/g, 'v\\d+\\.\\d+\\.\\d+'), 'ig');
const isVersionBump = messages.find((message) => commitMessageRegex.test(message)) !== undefined;
if (isVersionBump) {
tools.exit.success('No action necessary!');
return;
}
const majorWords = process.env['INPUT_MAJOR-WORDING'].split(',');
const minorWords = process.env['INPUT_MINOR-WORDING'].split(',');
const preReleaseWords = process.env['INPUT_RC-WORDING'].split(',');
const patchWords = process.env['INPUT_PATCH-WORDING'].split(',');
let version = process.env.INPUT_DEFAULT;
let foundWord = null;
let preid = process.env.INPUT_PREID;
tools.log('Version is set to Begining: ' + version);
if (
messages.some(
(message) => /^([a-zA-Z]+)(\(.+\))?(\!)\:/.test(message) || majorWords.some((word) => message.includes(word)),
)
) {
version = 'major';
tools.log('Version is set to major');
} else if (messages.some((message) => minorWords.some((word) => message.includes(word)))) {
version = 'minor';
tools.log('Version is set to minor');
} else if (
messages.some((message) =>
preReleaseWords.some((word) => {
if (message.includes(word)) {
foundWord = word;
return true;
} else {
return false;
}
}),
)
) {
preid = foundWord.split('-')[1];
version = 'prerelease';
tools.log('Version is set to prerelease2');
} else if (Array.isArray(patchWords) && patchWords.length) {
tools.log('PatchWords is set : ' + patchWords);
if (!messages.some((message) => patchWords.some((word) => message.includes(word)))) {
tools.log('Messages found on the patchwords : ' + messages);
version = null;
tools.log('Version is set to null 2');
}
} else if (process.env.INPUT_DEFAULT === 'prerelease'){
tools.log('Version is set to prerelease');
version = 'prerelease';
}
// case: if default=prerelease, but rc-wording is also set
// then unset it and do not run, when no rc words found in message
tools.log('Before Login');
tools.log(version); // null
tools.log('PreID: ' + preid); //prc
tools.log(process.env.INPUT_DEFAULT);
if (version === 'prerelease' && !messages.some((message) => preReleaseWords.some((word) => message.includes(word))) && process.env.INPUT_DEFAULT != 'prerelease') {
tools.log('Version is set to null');
version = null;
}
if (version === 'prerelease' || process.env.INPUT_DEFAULT === 'prerelease' && preid) {
tools.log('Version is prerelease and it contains a preid');
version = 'prerelease';
version = `${version} --preid=${preid}`;
}
if (version === null) {
tools.exit.success('No version keywords found, skipping bump.');
return;
}
try {
const current = pkg.version.toString();
// set git user
await tools.runInWorkspace('git', [
'config',
'user.name',
`"${process.env.GITHUB_USER || 'Automated Version Bump'}"`,
]);
await tools.runInWorkspace('git', [
'config',
'user.email',
`"${process.env.GITHUB_EMAIL || 'gh-action-bump-version@users.noreply.github.com'}"`,
]);
let currentBranch = /refs\/[a-zA-Z]+\/(.*)/.exec(process.env.GITHUB_REF)[1];
let isPullRequest = false;
if (process.env.GITHUB_HEAD_REF) {
// Comes from a pull request
currentBranch = process.env.GITHUB_HEAD_REF;
isPullRequest = true;
}
if (process.env['INPUT_TARGET-BRANCH']) {
// We want to override the branch that we are pulling / pushing to
currentBranch = process.env['INPUT_TARGET-BRANCH'];
}
console.log('currentBranch:', currentBranch);
// do it in the current checked out github branch (DETACHED HEAD)
// important for further usage of the package.json version
await tools.runInWorkspace('npm', ['version', '--allow-same-version=true', '--git-tag-version=false', current]);
console.log('current:', current, '/', 'version:', version);
let newVersion = execSync(`npm version --git-tag-version=false ${version}`).toString().trim().replace(/^v/, '');
newVersion = `${process.env['INPUT_TAG-PREFIX']}${newVersion}`;
await tools.runInWorkspace('git', ['commit', '-a', '-m', commitMessage.replace(/{{version}}/g, newVersion)]);
// now go to the actual branch to perform the same versioning
if (isPullRequest) {
// First fetch to get updated local version of branch
await tools.runInWorkspace('git', ['fetch']);
}
await tools.runInWorkspace('git', ['checkout', currentBranch]);
await tools.runInWorkspace('npm', ['version', '--allow-same-version=true', '--git-tag-version=false', current]);
console.log('current:', current, '/', 'version:', version);
newVersion = execSync(`npm version --git-tag-version=false ${version}`).toString().trim().replace(/^v/, '');
newVersion = `${process.env['INPUT_TAG-PREFIX']}${newVersion}`;
console.log(`::set-output name=newTag::${newVersion}`);
try {
// to support "actions/checkout@v1"
await tools.runInWorkspace('git', ['commit', '-a', '-m', commitMessage.replace(/{{version}}/g, newVersion)]);
} catch (e) {
console.warn(
'git commit failed because you are using "actions/checkout@v2"; ' +
'but that doesnt matter because you dont need that git commit, thats only for "actions/checkout@v1"',
);
}
const remoteRepo = `https://${process.env.GITHUB_ACTOR}:${process.env.GITHUB_TOKEN}@github.com/${process.env.GITHUB_REPOSITORY}.git`;
if (process.env['INPUT_SKIP-TAG'] !== 'true') {
await tools.runInWorkspace('git', ['tag', newVersion]);
await tools.runInWorkspace('git', ['push', remoteRepo, '--follow-tags']);
await tools.runInWorkspace('git', ['push', remoteRepo, '--tags']);
} else {
await tools.runInWorkspace('git', ['push', remoteRepo]);
}
} catch (e) {
tools.log.fatal(e);
tools.exit.failure('Failed to bump version');
}
tools.exit.success('Version bumped!');
});