Skip to content

Commit ea2b204

Browse files
committedAug 21, 2019
feat: allow to disable default release rules with release: false
1 parent 1414f6c commit ea2b204

7 files changed

+89
-12
lines changed
 

‎README.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -83,7 +83,8 @@ This is an `Array` of rule objects. A rule object has a `release` property and 1
8383
"releaseRules": [
8484
{"type": "docs", "scope": "README", "release": "patch"},
8585
{"type": "refactor", "scope": "core-*", "release": "minor"},
86-
{"type": "refactor", "release": "patch"}
86+
{"type": "refactor", "release": "patch"},
87+
{"scope": "no-release", "release": false}
8788
]
8889
}],
8990
"@semantic-release/release-notes-generator"
@@ -101,6 +102,7 @@ With the previous example:
101102
- Commits with `type` 'docs' and `scope` 'README' will be associated with a `patch` release.
102103
- Commits with `type` 'refactor' and `scope` starting with 'core-' (i.e. 'core-ui', 'core-rules', ...) will be associated with a `minor` release.
103104
- Other commits with `type` 'refactor' (without `scope` or with a `scope` not matching the glob `core-*`) will be associated with a `patch` release.
105+
- Commits with scope `no-release` will not be associated with a release type.
104106

105107
##### Default rules matching
106108

@@ -111,6 +113,7 @@ With the previous example:
111113
- Commits with `type` 'feat' will be associated with a `minor` release.
112114
- Commits with `type` 'fix' will be associated with a `patch` release.
113115
- Commits with `type` 'perf' will be associated with a `patch` release.
116+
- Commits with scope `no-release` will not be associated with a release type even if they have a breaking change or the `type` 'feat', 'fix' or 'perf'.
114117

115118
##### No rules matching
116119

‎index.js

+8-9
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
const {isUndefined} = require('lodash');
12
const parser = require('conventional-commits-parser').sync;
23
const filter = require('conventional-commits-filter');
34
const debug = require('debug')('semantic-release:commit-analyzer');
@@ -38,20 +39,18 @@ async function analyzeCommits(pluginConfig, context) {
3839
if (releaseRules) {
3940
debug('Analyzing with custom rules');
4041
commitReleaseType = analyzeCommit(releaseRules, commit);
41-
if (commitReleaseType) {
42-
logger.log('The release type for the commit is %s', commitReleaseType);
43-
}
4442
}
4543

4644
// If no custom releaseRules or none matched the commit, try with default releaseRules
47-
if (!commitReleaseType) {
45+
if (isUndefined(commitReleaseType)) {
4846
debug('Analyzing with default rules');
4947
commitReleaseType = analyzeCommit(DEFAULT_RELEASE_RULES, commit);
50-
if (commitReleaseType) {
51-
logger.log('The release type for the commit is %s', commitReleaseType);
52-
} else {
53-
logger.log('The commit should not trigger a release');
54-
}
48+
}
49+
50+
if (commitReleaseType) {
51+
logger.log('The release type for the commit is %s', commitReleaseType);
52+
} else {
53+
logger.log('The commit should not trigger a release');
5554
}
5655

5756
// Set releaseType if commit's release type is higher

‎lib/load-release-rules.js

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
const {isUndefined} = require('lodash');
12
const importFrom = require('import-from');
23
const RELEASE_TYPES = require('./default-release-types');
34

@@ -28,9 +29,9 @@ module.exports = ({releaseRules}, {cwd}) => {
2829
}
2930

3031
loadedReleaseRules.forEach(rule => {
31-
if (!rule || !rule.release) {
32+
if (!rule || isUndefined(rule.release)) {
3233
throw new Error('Error in commit-analyzer configuration: rules must be an object with a "release" property');
33-
} else if (RELEASE_TYPES.indexOf(rule.release) === -1) {
34+
} else if (RELEASE_TYPES.indexOf(rule.release) === -1 && rule.release !== null && rule.release !== false) {
3435
throw new Error(
3536
`Error in commit-analyzer configuration: "${
3637
rule.release

‎test/analyze-commit.test.js

+12
Original file line numberDiff line numberDiff line change
@@ -97,3 +97,15 @@ test('Return highest release type if multiple rules match', t => {
9797
'major'
9898
);
9999
});
100+
101+
test('Return "false" for release type if the matching rule has "release" set to "false"', t => {
102+
const commit = {type: 'fix'};
103+
104+
t.is(analyzeCommit([{type: 'fix', release: false}], commit), false);
105+
});
106+
107+
test('Return "null" for release type if the matching rule has "release" set to "null"', t => {
108+
const commit = {type: 'fix'};
109+
110+
t.is(analyzeCommit([{type: 'fix', release: null}], commit), null);
111+
});

‎test/compare-release-types.test.js

+8
Original file line numberDiff line numberDiff line change
@@ -10,4 +10,12 @@ test('Compares release types', t => {
1010
t.false(compareReleaseTypes('major', 'minor'));
1111
t.false(compareReleaseTypes('major', 'patch'));
1212
t.false(compareReleaseTypes('minor', 'patch'));
13+
14+
t.true(compareReleaseTypes('major', false));
15+
t.true(compareReleaseTypes('minor', false));
16+
t.true(compareReleaseTypes('patch', false));
17+
18+
t.true(compareReleaseTypes('major', null));
19+
t.true(compareReleaseTypes('minor', null));
20+
t.true(compareReleaseTypes('patch', null));
1321
});

‎test/integration.test.js

+45
Original file line numberDiff line numberDiff line change
@@ -201,6 +201,51 @@ test('Process rules in order and apply highest match from config even if default
201201
t.true(t.context.log.calledWith('Analysis of %s commits complete: %s release', 2, 'minor'));
202202
});
203203

204+
test('Allow to overwrite default "releaseRules" with "false"', async t => {
205+
const commits = [{message: 'chore: First chore'}, {message: 'feat: new feature'}];
206+
const releaseType = await analyzeCommits(
207+
{preset: 'angular', releaseRules: [{type: 'feat', release: false}]},
208+
{cwd, commits, logger: t.context.logger}
209+
);
210+
211+
t.is(releaseType, null);
212+
t.true(t.context.log.calledWith('Analyzing commit: %s', commits[0].message));
213+
t.true(t.context.log.calledWith('The commit should not trigger a release'));
214+
t.true(t.context.log.calledWith('Analyzing commit: %s', commits[1].message));
215+
t.true(t.context.log.calledWith('The commit should not trigger a release'));
216+
t.true(t.context.log.calledWith('Analysis of %s commits complete: %s release', 2, 'no'));
217+
});
218+
219+
test('Commits with an associated custom release type have higher priority than commits with release "false"', async t => {
220+
const commits = [{message: 'feat: Feature to skip'}, {message: 'docs: update README'}];
221+
const releaseType = await analyzeCommits(
222+
{preset: 'angular', releaseRules: [{type: 'feat', release: false}, {type: 'docs', release: 'patch'}]},
223+
{cwd, commits, logger: t.context.logger}
224+
);
225+
226+
t.is(releaseType, 'patch');
227+
t.true(t.context.log.calledWith('Analyzing commit: %s', commits[0].message));
228+
t.true(t.context.log.calledWith('The commit should not trigger a release'));
229+
t.true(t.context.log.calledWith('Analyzing commit: %s', commits[1].message));
230+
t.true(t.context.log.calledWith('The release type for the commit is %s', 'patch'));
231+
t.true(t.context.log.calledWith('Analysis of %s commits complete: %s release', 2, 'patch'));
232+
});
233+
234+
test('Commits with an associated default release type have higher priority than commits with release "false"', async t => {
235+
const commits = [{message: 'feat: new feature'}, {message: 'fix: new Fix'}];
236+
const releaseType = await analyzeCommits(
237+
{preset: 'angular', releaseRules: [{type: 'feat', release: false}]},
238+
{cwd, commits, logger: t.context.logger}
239+
);
240+
241+
t.is(releaseType, 'patch');
242+
t.true(t.context.log.calledWith('Analyzing commit: %s', commits[0].message));
243+
t.true(t.context.log.calledWith('The commit should not trigger a release'));
244+
t.true(t.context.log.calledWith('Analyzing commit: %s', commits[1].message));
245+
t.true(t.context.log.calledWith('The release type for the commit is %s', 'patch'));
246+
t.true(t.context.log.calledWith('Analysis of %s commits complete: %s release', 2, 'patch'));
247+
});
248+
204249
test('Use default "releaseRules" if none of provided match', async t => {
205250
const commits = [{message: 'Chore: First chore'}, {message: 'Update: new feature'}];
206251
const releaseType = await analyzeCommits(

‎test/load-release-rules.test.js

+9
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,15 @@ test('Return undefined if "releaseRules" not set', t => {
2222
t.is(releaseRules, undefined);
2323
});
2424

25+
test('Preserve release rules set to "false" or "null"', t => {
26+
const releaseRules = loadReleaseRules(
27+
{releaseRules: [{type: 'feat', release: false}, {type: 'fix', release: null}]},
28+
{cwd}
29+
);
30+
31+
t.deepEqual(releaseRules, [{type: 'feat', release: false}, {type: 'fix', release: null}]);
32+
});
33+
2534
test('Throw error if "releaseRules" reference invalid commit type', t => {
2635
t.throws(
2736
() => loadReleaseRules({releaseRules: [{tag: 'Update', release: 'invalid'}]}, {cwd}),

0 commit comments

Comments
 (0)
Please sign in to comment.