- Add a new plugin type: `publish` - Add support for multi-plugin. A plugin module can now return an object with a property for each plugin type - Uses by default [npm](https://github.com/semantic-release/npm) and [github](https://github.com/semantic-release/github) in addition of Travis for the verify condition plugin - Uses by default [npm](https://github.com/semantic-release/npm) and [github](https://github.com/semantic-release/github) for the publish plugin - `gitTag` if one can be found is passed to `generateNotes` for both `lastRelease` and `nextRelease` - `semantic-release` now verifies the plugin configuration (in the `release` property of `package.json`) and throws an error if it's invalid - `semantic-release` now verifies each plugin output and will throw an error if a plugin returns an unexpected value. BREAKING CHANGE: `githubToken`, `githubUrl` and `githubApiPathPrefix` have to be set at the [github](https://github.com/semantic-release/github) plugin level. They can be set via `GH_TOKEN`, `GH_URL` and `GH_PREFIX` environment variables. BREAKING CHANGE: the `npm` parameter is not passed to any plugin anymore. Each plugin have to read `.npmrc` if they needs to (with https://github.com/kevva/npm-conf for example).
108 lines
4.7 KiB
JavaScript
108 lines
4.7 KiB
JavaScript
const gitLogParser = require('git-log-parser');
|
|
const getStream = require('get-stream');
|
|
const debug = require('debug')('semantic-release:get-commits');
|
|
const {unshallow} = require('./git');
|
|
const getVersionHead = require('./get-version-head');
|
|
|
|
/**
|
|
* Commit message.
|
|
*
|
|
* @typedef {Object} Commit
|
|
* @property {string} hash The commit hash.
|
|
* @property {string} message The commit message.
|
|
*/
|
|
|
|
/**
|
|
* Last release.
|
|
*
|
|
* @typedef {Object} LastRelease
|
|
* @property {string} version The version number of the last release.
|
|
* @property {string} [gitHead] The commit sha used to make the last release.
|
|
*/
|
|
|
|
/**
|
|
* Result object.
|
|
*
|
|
* @typedef {Object} Result
|
|
* @property {Array<Commit>} commits The list of commits since the last release.
|
|
* @property {LastRelease} lastRelease The updated lastRelease.
|
|
*/
|
|
|
|
/**
|
|
* Retrieve the list of commits on the current branch since the last released version, or all the commits of the current branch if there is no last released version.
|
|
*
|
|
* The commit correspoding to the last released version is determined as follow:
|
|
* - Use `lastRelease.gitHead` if defined and present in `branch` history.
|
|
* - If `lastRelease.gitHead` is not in the `branch` history, unshallow the repository and try again.
|
|
* - If `lastRelease.gitHead` is still not in the `branch` history, search for a tag named `v<version>` or `<version>` and verify if it's associated commit sha is present in `branch` history.
|
|
*
|
|
* @param {LastRelease} lastRelease The lastRelease object obtained from the getLastRelease plugin.
|
|
* @param {string} branch The branch to release from.
|
|
* @param {Object} logger Global logger.
|
|
*
|
|
* @return {Promise<Result>} The list of commits on the branch `branch` since the last release and the updated lastRelease with the gitHead used to retrieve the commits.
|
|
*
|
|
* @throws {SemanticReleaseError} with code `ENOTINHISTORY` if `lastRelease.gitHead` or the commit sha derived from `config.lastRelease.version` is not in the direct history of `branch`.
|
|
* @throws {SemanticReleaseError} with code `ENOGITHEAD` if `lastRelease.gitHead` is undefined and no commit sha can be found for the `config.lastRelease.version`.
|
|
*/
|
|
module.exports = async ({version, gitHead}, branch, logger) => {
|
|
let gitTag;
|
|
if (gitHead || version) {
|
|
try {
|
|
({gitHead, gitTag} = await getVersionHead(gitHead, version, branch));
|
|
} catch (err) {
|
|
if (err.code === 'ENOTINHISTORY') {
|
|
logger.error(notInHistoryMessage(err.gitHead, branch, version));
|
|
} else {
|
|
logger.error(noGitHeadMessage(branch, version));
|
|
}
|
|
throw err;
|
|
}
|
|
logger.log('Retrieving commits since %s, corresponding to version %s', gitHead, version);
|
|
} else {
|
|
logger.log('No previous release found, retrieving all commits');
|
|
// If there is no gitHead nor a version, there is no previous release. Unshallow the repo in order to retrieve all commits
|
|
await unshallow();
|
|
}
|
|
|
|
Object.assign(gitLogParser.fields, {hash: 'H', message: 'B', gitTags: 'd', committerDate: {key: 'ci', type: Date}});
|
|
const commits = (await getStream.array(gitLogParser.parse({_: `${gitHead ? gitHead + '..' : ''}HEAD`}))).map(
|
|
commit => {
|
|
commit.message = commit.message.trim();
|
|
commit.gitTags = commit.gitTags.trim();
|
|
return commit;
|
|
}
|
|
);
|
|
logger.log('Found %s commits since last release', commits.length);
|
|
debug('Parsed commits: %o', commits);
|
|
return {commits, lastRelease: {version, gitHead, gitTag}};
|
|
};
|
|
|
|
function noGitHeadMessage(branch, version) {
|
|
return `The commit the last release of this package was derived from cannot be determined from the release metadata nor from the repository tags.
|
|
This means semantic-release can not extract the commits between now and then.
|
|
This is usually caused by releasing from outside the repository directory or with innaccessible git metadata.
|
|
|
|
You can recover from this error by creating a tag for the version "${
|
|
version
|
|
}" on the commit corresponding to this release:
|
|
$ git tag -f v${version} <commit sha1 corresponding to last release>
|
|
$ git push -f --tags origin ${branch}
|
|
`;
|
|
}
|
|
|
|
function notInHistoryMessage(gitHead, branch, version) {
|
|
return `The commit the last release of this package was derived from is not in the direct history of the "${
|
|
branch
|
|
}" branch.
|
|
This means semantic-release can not extract the commits between now and then.
|
|
This is usually caused by force pushing, releasing from an unrelated branch, or using an already existing package name.
|
|
|
|
You can recover from this error by restoring the commit "${gitHead}" or by creating a tag for the version "${
|
|
version
|
|
}" on the commit corresponding to this release:
|
|
$ git tag -f v${version || '<version>'} <commit sha1 corresponding to last release>
|
|
$ git push -f --tags origin ${branch}
|
|
`;
|
|
}
|