-
Notifications
You must be signed in to change notification settings - Fork 38
/
Copy pathdangerfile.js
138 lines (120 loc) · 6.51 KB
/
dangerfile.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
/* global danger, fail, message */
const { pr } = danger.github;
const bodyAndTitle = (pr.body + pr.title).toLowerCase();
const isTrivial = bodyAndTitle.includes('#trivial'); // turns off all danger checks
const isGlobalConfigUpdate = bodyAndTitle.includes('#globalconfig'); // turns off danger checks for packages outside the root
if (!isTrivial) {
const failedChangelogs = [];
const failedVersionBumps = [];
const packageDirectories = [
'packages/components/atoms/',
'packages/components/molecules/',
'packages/components/organisms/',
'packages/components/templates/',
'packages/components/pages/',
'packages/services/',
'packages/tools/',
'packages/storybook'
];
const modifiedFiles = danger.git.modified_files;
// Get an array of package paths that have changed
const modifiedPackagePaths = modifiedFiles.filter(filepath => filepath.startsWith('packages/'));
const modifiedPackages = {};
// loop through package paths and return an object with the package name and path to that package
modifiedPackagePaths.forEach(filepath => {
const directoryMatch = packageDirectories.filter(subDirectory => filepath.includes(subDirectory));
let packageName;
let path;
if (directoryMatch[0] === 'packages/storybook') {
packageName = 'storybook';
path = 'packages/';
} else {
// eslint-disable-next-line prefer-destructuring
packageName = (filepath.split(directoryMatch)[1]) // splits the string based on the matching directory
.split('/')[0]; // and then get the first part of the string that's left (the package name)
[path] = directoryMatch;
}
modifiedPackages[packageName] = {
path
};
});
// Get an Array of just the package names, and then make that Array unique
const modifiedPackageNames = Object.keys(modifiedPackages);
const uniqueModifiedPackages = new Set(modifiedPackageNames);
// Fail if the title of the PR isn't in the format of a version i.e. {package-name}@vX.X.X (such as f-header@v1.4.0)
const versionRegex = /^([a-z\-]+@v?[0-9]+\.[0-9]+\.[0-9]+)/;
const isPRTitleVersioned = (pr.title).match(versionRegex);
if (!isPRTitleVersioned) {
fail(':exclamation: PR title should start with the package version in the format {package-name}@v(x.x.x) (such as f-header@v1.4.0)');
}
// Get an array of files in the root monorepo that have changed
const modifiedRootFiles = modifiedFiles.filter(filepath => !filepath.startsWith('packages/') && filepath !== 'yarn.lock');
const modifiedRootPackage = modifiedRootFiles.length > 0; // check for changes in the root package
if (modifiedRootPackage && !modifiedRootFiles.includes('CHANGELOG.md')) {
const changelogLink = 'https://github.com/justeat/fozzie-components/blob/master/CHANGELOG.md';
fail(`:memo: Please include a <code>CHANGELOG</code> entry for the root mono-repo. You can find the current version at <a href="${changelogLink}">CHANGELOG.md</a>`);
}
// Check for version update at root level
const hasRootPackageJsonChanged = modifiedRootFiles.includes('package.json');
const rootPackageDiff = danger.git.JSONDiffForFile('package.json');
rootPackageDiff.then(result => {
if (modifiedRootPackage && (!hasRootPackageJsonChanged || (hasRootPackageJsonChanged && !result.version))) {
const semverLink = 'https://docs.npmjs.com/getting-started/semantic-versioning';
fail(`:arrow_up: This PR should include a <a href="${semverLink}"><code>SEMVER</code></a> version bump at the root of the mono-repo, as you have changed root level config.`);
}
}, err => {
/* eslint-disable no-console */
console.log(err);
/* eslint-enable no-console */
});
// If this PR is just an internal config update, then don't run
// the danger checks for every sub-package (as we shouldn't mandate CHANGELOG and version bumps for this at package level)
if (!isGlobalConfigUpdate) {
/** Specific Package Checks **/
/**
* checkPackageDiff
* Checks if there is a change in the package.json for a specific package
* Returns with a Promise resolve() when completed
*
* @param {*} pkg
* @param {*} resolve
*/
const checkPackageDiff = (pkg, resolve) => {
// Check for version update
const hasPackageJsonChanged = danger.git.modified_files.includes(`${modifiedPackages[pkg].path}${pkg}/package.json`);
const packageDiff = danger.git.JSONDiffForFile(`${modifiedPackages[pkg].path}${pkg}/package.json`);
packageDiff.then(result => {
if (!hasPackageJsonChanged || (hasPackageJsonChanged && !result.version)) {
failedVersionBumps.push(pkg);
}
resolve();
}, err => {
/* eslint-disable no-console */
console.log(err);
/* eslint-enable no-console */
});
};
// Loops through each package that includes files that been modified and runs DangerJS checks for each
const requests = Array.from(uniqueModifiedPackages).map(pkg => {
// Add package to `failedChangelogs` if there isn’t a CHANGELOG entry – should update for every PR
if (!danger.git.modified_files.includes(`${modifiedPackages[pkg].path}${pkg}/CHANGELOG.md`)) {
failedChangelogs.push(pkg);
}
// Add package to `failedVersionBumps` if there isn’t a package.json bump
return new Promise(resolve => {
checkPackageDiff(pkg, resolve);
});
});
// Once all packages have been checked, if `failedVersionBumps` isn't empty, return DangerJS fail
Promise.all(requests).then(() => {
if (failedVersionBumps.length) {
const semverLink = 'https://docs.npmjs.com/getting-started/semantic-versioning';
fail(`:arrow_up: This PR should include a <a href="${semverLink}"><code>SEMVER</code></a> version bump for <strong>${failedVersionBumps.join(', ')}</strong>, so that these packages can be published once merged.`);
}
});
// if `failedChangelogs` isn't empty, return DangerJS fail
if (failedChangelogs.length) {
fail(`:memo: Please include a <code>CHANGELOG</code> entry for <strong>${failedChangelogs.join(', ')}</strong>`);
}
}
}