- Run with one command and do not rely on error exit codes to stop the process when a release is not necessary - Break `index.js` in smaller modules in order to improve testability and simplify the code - Add several missing unit and integration tests to reach 100% coverage - Integration tests now test end to end, including publishing to Github (with http://www.mock-server.com on Docker) - Use `tj/commander.js` to print an help message, verify and parse CLI arguments - Semantic-release can now be called via Javascript API: `require('semantic-release')(options)` - Remove npmlog dependency and add more log messages - Logger is now passed to plugins - Add debug logs with `visionmedia/debug`. `debug` is enabled for both semantic-release and plugins with `--debug` - Use `kevva/npm-conf` in place of the deprecated `npm/npmconf` - Pass lastRelease, nextRelease and commits to generate-notes plugin - In dry-run mode, print the release note instead of publishing it to Github as draft, and skip the CI verifications - The dry-run mode does not require npm and Github TOKEN to be set anymore and can be run locally BREAKING CHANGE: Semantic-Release must now be executed with `semantic-release` instead of `semantic-release pre && npm publish && semantic-release post`. BREAKING CHANGE: The `semantic-release` command now returns with exit code 0 on expected exception (no release has to be done, running on a PR, gitHead not found, other CI job failed etc...). It only returns with 1 when there is an unexpected error (code error in a plugin, plugin not found, git command cannot be run etc..). BREAKING CHANGE: Calling the `semantic-release` command with unexpected argument(s) now exit with 1 and print an help message. BREAKING CHANGE: Semantic-Release does not rely on `npmlog` anymore and the log level cannot be configured. Debug logs can be activated with CLI option `--debug` or with environment variable `DEBUG=semantic-release:*` BREAKING CHANGE: The CLI options `--debug` doesn't enable the dry-run mode anymore but activate the debugs. The dry run mode is now set with the CLI command `--dry-run` or `-d`.
124 lines
5.4 KiB
JavaScript
124 lines
5.4 KiB
JavaScript
import {promisify} from 'util';
|
|
import test from 'ava';
|
|
import proxyquire from 'proxyquire';
|
|
import {stub, match} from 'sinon';
|
|
|
|
test.beforeEach(t => {
|
|
// Stub the logger functions
|
|
t.context.log = stub();
|
|
t.context.plugins = proxyquire('../src/lib/plugins', {'./logger': {log: t.context.log}});
|
|
});
|
|
|
|
test('Export plugins', t => {
|
|
// Call the plugin module
|
|
const defaultPlugins = t.context.plugins({});
|
|
|
|
// Verify the module returns a function for each plugin
|
|
t.is(typeof defaultPlugins.analyzeCommits, 'function');
|
|
t.is(typeof defaultPlugins.generateNotes, 'function');
|
|
t.is(typeof defaultPlugins.verifyConditions, 'function');
|
|
t.is(typeof defaultPlugins.verifyRelease, 'function');
|
|
t.is(typeof defaultPlugins.getLastRelease, 'function');
|
|
});
|
|
|
|
test('Pipeline - Get all results', async t => {
|
|
// Call the plugin module with a verifyRelease plugin pipeline
|
|
const pipelinePlugins = t.context.plugins({
|
|
verifyRelease: ['./src/lib/plugin-noop', './test/fixtures/plugin-result-a', './test/fixtures/plugin-result-b'],
|
|
});
|
|
|
|
// Call the verifyRelease pipeline
|
|
const results = await pipelinePlugins.verifyRelease({});
|
|
|
|
// Verify the pipeline return the expected result for each plugin, in order
|
|
t.deepEqual(results, [undefined, 'a', 'b']);
|
|
// Verify the logger has been called with the plugins path
|
|
t.true(t.context.log.calledWith(match.string, './src/lib/plugin-noop'));
|
|
t.true(t.context.log.calledWith(match.string, './test/fixtures/plugin-result-a'));
|
|
t.true(t.context.log.calledWith(match.string, './test/fixtures/plugin-result-b'));
|
|
});
|
|
|
|
test('Pipeline - Pass pluginConfig and options to each plugins', async t => {
|
|
// Plugin configuration with options (plugin-result-config is a mock plugin returning its pluginConfig and options parameters)
|
|
const pluginConfig = {path: './test/fixtures/plugin-result-config', pluginParam: 'param1'};
|
|
// Semantic-release global options
|
|
const options = {semanticReleaseParam: 'param2'};
|
|
// Call the plugin module with a verifyRelease plugin pipeline
|
|
const pipelinePlugins = t.context.plugins({
|
|
verifyRelease: [pluginConfig, './test/fixtures/plugin-result-config'],
|
|
});
|
|
|
|
// Call the verifyRelease pipeline
|
|
const results = await pipelinePlugins.verifyRelease(options);
|
|
|
|
// Verify the pipeline first result is the pluginConfig and options parameters (to verify the plugin was called with the defined pluginConfig and options parameters)
|
|
t.deepEqual(results, [{pluginConfig, options}, {pluginConfig: {}, options}]);
|
|
// Verify the logger has been called with the plugins path
|
|
t.true(t.context.log.calledWith(match.string, './test/fixtures/plugin-result-config'));
|
|
});
|
|
|
|
test('Pipeline - Get first error', async t => {
|
|
// Call the plugin module with a verifyRelease plugin pipeline
|
|
const pipelinePlugins = t.context.plugins({
|
|
verifyRelease: ['./src/lib/plugin-noop', './test/fixtures/plugin-error-a', './test/fixtures/plugin-error-b'],
|
|
});
|
|
|
|
// Call the verifyRelease pipeline and verify it returns the error thrown by './test/fixtures/plugin-error-a'
|
|
await t.throws(pipelinePlugins.verifyRelease({}), 'a');
|
|
// Verify the logger has been called with the plugins path
|
|
t.true(t.context.log.calledWith(match.string, './src/lib/plugin-noop'));
|
|
t.true(t.context.log.calledWith(match.string, './test/fixtures/plugin-error-a'));
|
|
});
|
|
|
|
test('Normalize and load plugin from string', t => {
|
|
// Call the normalize function with a path
|
|
const plugin = t.context.plugins.normalize('./src/lib/plugin-noop');
|
|
|
|
// Verify the plugin is loaded
|
|
t.is(typeof plugin, 'function');
|
|
// Verify the logger has been called with the plugins path
|
|
t.true(t.context.log.calledWith(match.string, './src/lib/plugin-noop'));
|
|
});
|
|
|
|
test('Normalize and load plugin from object', t => {
|
|
// Call the normalize function with an object (with path property)
|
|
const plugin = t.context.plugins.normalize({path: './src/lib/plugin-noop'});
|
|
|
|
// Verify the plugin is loaded
|
|
t.is(typeof plugin, 'function');
|
|
// Verify the logger has been called with the plugins path
|
|
t.true(t.context.log.calledWith(match.string, './src/lib/plugin-noop'));
|
|
});
|
|
|
|
test('Load from fallback', t => {
|
|
// Call the normalize function with a fallback
|
|
const plugin = t.context.plugins.normalize(null, '../lib/plugin-noop');
|
|
|
|
// Verify the fallback plugin is loaded
|
|
t.is(typeof plugin, 'function');
|
|
});
|
|
|
|
test('Always pass a defined "pluginConfig" for plugin defined with string', async t => {
|
|
// Call the normalize function with the path of a plugin that returns its config
|
|
const plugin = t.context.plugins.normalize('./test/fixtures/plugin-result-config');
|
|
const pluginResult = await promisify(plugin)({});
|
|
|
|
t.deepEqual(pluginResult.pluginConfig, {});
|
|
});
|
|
|
|
test('Always pass a defined "pluginConfig" for plugin defined with path', async t => {
|
|
// Call the normalize function with the path of a plugin that returns its config
|
|
const plugin = t.context.plugins.normalize({path: './test/fixtures/plugin-result-config'});
|
|
const pluginResult = await promisify(plugin)({});
|
|
|
|
t.deepEqual(pluginResult.pluginConfig, {path: './test/fixtures/plugin-result-config'});
|
|
});
|
|
|
|
test('Always pass a defined "pluginConfig" for fallback plugin', async t => {
|
|
// Call the normalize function with the path of a plugin that returns its config
|
|
const plugin = t.context.plugins.normalize(null, '../../test/fixtures/plugin-result-config');
|
|
const pluginResult = await promisify(plugin)({});
|
|
|
|
t.deepEqual(pluginResult.pluginConfig, {});
|
|
});
|