-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdangerfile.js
109 lines (94 loc) · 4.67 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
const { danger, warn, fail, message } = require('danger');
const fs = require('fs');
const MAX_COMMIT_MESSAGE_LENGTH = 72;
const SMOKE_TEST_BRANCH = 'smoke/';
const SMOKE_TEST_WORKFLOW_FILE_PATH = '.github/workflows/smoke-tests.yml';
if (danger.github && danger.github.pr) {
const commitizenRegex = /^(feat|fix|chore|test|docs|perf|refactor|revert)(\(.*\))?:(.+)$/;
const ghCommits = danger.github.commits;
let willTriggerRelease = false;
for (const { commit } of ghCommits) {
const { message, url } = commit;
const firstLine = message.split('\n')[0];
if (message.startsWith('feat') || message.startsWith('fix')) {
willTriggerRelease = true;
}
const regexMatch = commitizenRegex.exec(firstLine);
if (!regexMatch) {
fail(
`Commit ["${firstLine}"](${url}) is not a valid commitizen message. See [Contributing page](https://github.com/snyk/snyk/blob/master/.github/CONTRIBUTING.md#commit-types) with required commit message format.`,
);
}
if (firstLine.length >= MAX_COMMIT_MESSAGE_LENGTH) {
warn(
`Your commit message ["${firstLine}"](${url}) is too long. Keep first line of your commit under ${MAX_COMMIT_MESSAGE_LENGTH} characters.`,
);
}
}
if (!willTriggerRelease) {
message(
"This PR will not trigger a new version. It doesn't include any commit message with `feat` or `fix`.",
);
}
// Forgotten tests check
const modifiedTest =
danger.git.modified_files.some((f) => f.startsWith('test/')) ||
danger.git.created_files.some((f) => f.startsWith('test/'));
const modifiedSrc =
danger.git.modified_files.some((f) => f.startsWith('src/')) ||
danger.git.created_files.some((f) => f.startsWith('src/'));
if (modifiedSrc && !modifiedTest) {
// TODO: let's be careful about wording here. Maybe including Contributing guidelines and project goals document here
warn(
"You've modified files in src/ directory, but haven't updated anything in test folder. Is there something that could be tested?",
);
}
// `.spec.ts` is always used for Jest tests
// `.test.ts` is normally used for Tap tests and but there are also `.spec.ts` files which are used be Tap tests in test/acceptance.
// either way, we should warn about new `.test.ts` or `.spec.ts` files being created outside the `/test/jest` folder
const newTestFiles = danger.git.created_files.filter((f) => {
const inTestFolder = f.startsWith('test/');
const isATestFile = f.includes('.test.ts') || f.includes('.spec.ts');
const inJestFolder = f.startsWith('test/jest/');
const inFixturesFolder = f.startsWith('test/fixtures/');
return inTestFolder && isATestFile && !inJestFolder && !inFixturesFolder;
});
if (newTestFiles.length) {
const joinedFileList = newTestFiles.map((f) => '- `' + f + '`').join('\n');
const msg = `Looks like you added a new Tap test. Consider making it a Jest test instead. See files in \`test/jest/(unit|system|acceptance)\` for examples. Files found:\n${joinedFileList}`;
warn(msg);
}
// Smoke test modification check
const modifiedSmokeTest =
danger.git.modified_files.some((f) => f.startsWith('test/smoke/')) ||
danger.git.created_files.some((f) => f.startsWith('test/smoke/')) ||
danger.git.modified_files.includes(SMOKE_TEST_WORKFLOW_FILE_PATH);
const isOnSmokeTestBranch = danger.github.pr.head.ref.startsWith(
SMOKE_TEST_BRANCH,
);
if (modifiedSmokeTest && !isOnSmokeTestBranch) {
message(
`You are modifying something in test/smoke directory, yet you are not on the branch starting with ${SMOKE_TEST_BRANCH}. You can prefix your branch with ${SMOKE_TEST_BRANCH} and Smoke tests will trigger for this PR.`,
);
}
// Enforce usage of ES6 modules
const filesUsingNodeJSImportExport = danger.git.modified_files
.filter((filePath) => {
if (filePath.endsWith('.js')) {
return false;
}
const fileContent = fs.readFileSync(filePath, 'utf8');
return (
fileContent.includes('module.exports') ||
fileContent.includes('= require(')
);
})
.map((filePath) => `- \`${filePath}\``)
.join('\n');
if (filesUsingNodeJSImportExport) {
const message =
"Since the CLI is unifying on a standard and improved tooling, we're starting to migrate old-style `import`s and `export`s to ES6 ones.\nA file you've modified is using either `module.exports` or `require()`. If you can, please update them to ES6 [import syntax](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/import) and [export syntax](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/export).\n Files found:\n" +
filesUsingNodeJSImportExport;
warn(message);
}
}