-
Notifications
You must be signed in to change notification settings - Fork 583
/
dangerfile.ts
173 lines (149 loc) · 6.14 KB
/
dangerfile.ts
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
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
import * as fs from "fs"
import { danger, fail, markdown, warn } from "danger"
import { isArray, pickBy } from "lodash"
import { changelogTemplateSections } from "./scripts/changelog/changelogTemplateSections"
import { ParseResult, parsePRDescription } from "./scripts/changelog/parsePRDescription"
// TypeScript thinks we're in React Native,
// so the node API gives us errors:
/**
* Helpers
*/
const typescriptOnly = (file: string) => file.includes(".ts")
const filesOnly = (file: string) => fs.existsSync(file) && fs.lstatSync(file).isFile()
// Modified or Created can be treated the same a lot of the time
const getCreatedFileNames = (createdFiles: string[]) => createdFiles.filter(filesOnly)
const testOnlyFilter = (filename: string) => filename.includes(".tests") && typescriptOnly(filename)
/**
* Danger Rules
*/
// We are trying to migrate away from moment towards luxon
const preventUsingMoment = () => {
const newMomentImports = getCreatedFileNames(danger.git.created_files).filter((filename) => {
const content = fs.readFileSync(filename).toString()
return content.includes('from "moment"') || content.includes('from "moment-timezone"')
})
if (newMomentImports.length > 0) {
warn(`We are trying to migrate away from moment towards \`luxon\`, but found moment imports in the following new files:
${newMomentImports.map((filename) => `- \`${filename}\``).join("\n")}
See [docs](https://moment.github.io/luxon/api-docs/index.html).
`)
}
}
// We are trying to migrate away from test-renderer towards @testing-library/react-native
const preventUsingTestRenderer = () => {
const newTRImports = getCreatedFileNames(danger.git.created_files)
.filter(testOnlyFilter)
.filter((filename) => {
const content = fs.readFileSync(filename).toString()
return (
content.includes('from "app/utils/tests/renderWithWrappers"') &&
(content.includes("renderWithWrappersLEGACY ") ||
content.includes("renderWithWrappersLEGACY,"))
)
})
if (newTRImports.length > 0) {
warn(`We are trying to migrate away from \`react-test-renderer\` towards \`@testing-library/react-native\`, but found Test-Renderer imports in the following new unit test files:
${newTRImports.map((filename) => `- \`${filename}\``).join("\n")}
See [\`Pill.tests.tsx\`](https://github.com/artsy/eigen/blob/2f32d462bb3b4ce358c8a14e3ed09b42523de8bd/src/palette/elements/Pill/__tests__/Pill-tests.tsx) as an example, or [the docs](https://callstack.github.io/react-native-testing-library/docs/api-queries).
`)
}
}
// Validates that we've not accidentally let in a testing
// shortcut to simplify dev work
const verifyRemainingDevWork = () => {
const modified = danger.git.modified_files
const editedFiles = modified.concat(danger.git.created_files)
const testFiles = editedFiles.filter((f) => f?.includes("Tests") && f.match(/\.(swift|m)$/))
const testingShortcuts = ["fdescribe(", "describe.only(", "fit(", "fit(", "it.only("]
for (const file of testFiles) {
const content = fs.readFileSync(file).toString()
for (const shortcut of testingShortcuts) {
if (content.includes(shortcut)) {
fail(`Found a testing shortcut in ${file}`)
}
}
}
// A shortcut to say "I know what I'm doing thanks"
const knownDevTools = danger.github.pr.body?.includes("#known") ?? false
// These files are ones we really don't want changes to, except in really occasional
// cases, so offer a way out.
const devOnlyFiles = ["Artsy.xcodeproj/xcshareddata/xcschemes/Artsy.xcscheme"]
for (const file of devOnlyFiles) {
if (modified.includes(file) && !knownDevTools) {
fail(
"Developer Specific file shouldn't be changed, you can skip by adding #known to the PR body and re-running CI"
)
}
}
}
// Force the usage of WebPs
const IMAGE_EXTENSIONS_TO_AVOID = ["png", "jpg", "jpeg"]
const IMAGE_EXTENSIONS = [...IMAGE_EXTENSIONS_TO_AVOID, "webp"]
export const useWebPs = (fileNames: string[]) => {
const hasNonWebImages = Boolean(
fileNames
.map((fileName) => fileName.split(".").pop() || "")
.filter((fileExtension) => {
return IMAGE_EXTENSIONS.includes(fileExtension)
})
.find((imageFileExtension) => IMAGE_EXTENSIONS_TO_AVOID.includes(imageFileExtension))
)
if (hasNonWebImages) {
warn(
"❌ **It seems like you added some non WebP images to Eigen, please convert them to WebPs using `source images/script.sh` script **"
)
}
}
// Require changelog on Eigen PRs to be valid
// See Eigen RFC: https://github.com/artsy/eigen/issues/4499
export const validatePRChangelog = () => {
const pr = danger.github.pr
const isDraft = danger.github.pr.draft
if (isDraft) {
console.log("Skipping this check because the PR is a draft")
return
}
const isOpen = pr.state === "open"
if (!isOpen) {
console.log("Skipping this check because the PR is not open")
return
}
const content = pr.body
const res = parsePRDescription(content) as ParseResult
if (res.type === "error") {
console.log("Something went wrong while parsing the PR description")
warn(
"❌ **An error occurred while validating your changelog, please make sure you provided a valid changelog.**"
)
return
}
if (res.type === "no_changes") {
console.log("PR has no changes")
warn("✅ **No changelog changes**")
return
}
// At this point, the PR description changelog changes are valid
// and res contains a list of the changes
console.log("PR Changelog is valid")
const { ...changedSections } = res
const message =
"### This PR contains the following changes:\n" +
Object.entries(
pickBy(changedSections, (changesArray) => isArray(changesArray) && changesArray.length > 0)
)
.map(([section, sectionValue]) => {
return `\n- ${
changelogTemplateSections[section as keyof typeof changedSections]
} (${sectionValue} - ${pr.user.login})`
})
.join("")
return markdown(message)
}
;(async function () {
const newCreatedFileNames = getCreatedFileNames(danger.git.created_files)
preventUsingMoment()
preventUsingTestRenderer()
verifyRemainingDevWork()
useWebPs(newCreatedFileNames)
validatePRChangelog()
})()