| 
									
										
										
										
											2022-03-15 01:42:38 +01:00
										 |  |  | import { exec } from "child_process" | 
					
						
							| 
									
										
										
										
											2023-02-03 04:48:32 +01:00
										 |  |  | import { describe, it } from "vitest" | 
					
						
							| 
									
										
										
										
											2022-03-15 01:42:38 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param forbidden: a GREP-regex. This means that '.' is a wildcard and should be escaped to match a literal dot | 
					
						
							|  |  |  |  * @param reason | 
					
						
							|  |  |  |  * @private | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2023-05-16 01:34:57 +02:00
										 |  |  | function detectInCode(forbidden: string, reason: string): Promise<void> { | 
					
						
							|  |  |  |     return new Promise<void>((done) => { | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |         const excludedDirs = [ | 
					
						
							|  |  |  |             ".git", | 
					
						
							|  |  |  |             "node_modules", | 
					
						
							|  |  |  |             "dist", | 
					
						
							|  |  |  |             ".cache", | 
					
						
							|  |  |  |             ".parcel-cache", | 
					
						
							|  |  |  |             "assets", | 
					
						
							|  |  |  |             "vendor", | 
					
						
							|  |  |  |             ".idea/", | 
					
						
							|  |  |  |         ] | 
					
						
							| 
									
										
										
										
											2022-03-15 01:42:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |         exec( | 
					
						
							|  |  |  |             'grep -n "' + | 
					
						
							|  |  |  |                 forbidden + | 
					
						
							|  |  |  |                 '" -r . ' + | 
					
						
							|  |  |  |                 excludedDirs.map((d) => "--exclude-dir=" + d).join(" "), | 
					
						
							|  |  |  |             (error, stdout, stderr) => { | 
					
						
							|  |  |  |                 if (error?.message?.startsWith("Command failed: grep")) { | 
					
						
							|  |  |  |                     console.warn("Command failed!", error) | 
					
						
							|  |  |  |                     return | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (error !== null) { | 
					
						
							|  |  |  |                     throw error | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (stderr !== "") { | 
					
						
							|  |  |  |                     throw stderr | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2022-03-15 01:42:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |                 const found = stdout | 
					
						
							|  |  |  |                     .split("\n") | 
					
						
							|  |  |  |                     .filter((s) => s !== "") | 
					
						
							|  |  |  |                     .filter((s) => !s.startsWith("./test/")) | 
					
						
							|  |  |  |                 if (found.length > 0) { | 
					
						
							|  |  |  |                     const msg = `Found a '${forbidden}' at \n    ${found.join( | 
					
						
							|  |  |  |                         "\n     " | 
					
						
							|  |  |  |                     )}.\n ${reason}`
 | 
					
						
							|  |  |  |                     console.error(msg) | 
					
						
							|  |  |  |                     console.error(found.length, "issues found") | 
					
						
							|  |  |  |                     throw msg | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2023-05-16 01:34:57 +02:00
										 |  |  |     }) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-01 14:32:45 +02:00
										 |  |  | function wrap(promise: Promise<void>): (done: () => void) => void { | 
					
						
							|  |  |  |     return (done) => { | 
					
						
							| 
									
										
										
										
											2023-05-16 01:34:57 +02:00
										 |  |  |         promise.then(done) | 
					
						
							| 
									
										
										
										
											2023-06-01 14:32:45 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-16 01:34:57 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-01 14:32:45 +02:00
										 |  |  | function itAsync(name: string, promise: Promise<void>) { | 
					
						
							| 
									
										
										
										
											2023-05-16 01:34:57 +02:00
										 |  |  |     it(name, wrap(promise)) | 
					
						
							| 
									
										
										
										
											2022-03-15 01:42:38 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | describe("Code quality", () => { | 
					
						
							| 
									
										
										
										
											2023-05-16 01:34:57 +02:00
										 |  |  |     itAsync( | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |         "should not contain reverse", | 
					
						
							| 
									
										
										
										
											2022-03-15 01:42:38 +01:00
										 |  |  |         detectInCode( | 
					
						
							|  |  |  |             "reverse()", | 
					
						
							|  |  |  |             "Reverse is stateful and changes the source list. This often causes subtle bugs" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |     ) | 
					
						
							| 
									
										
										
										
											2022-03-15 01:42:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-16 01:34:57 +02:00
										 |  |  |     itAsync( | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |         "should not contain 'constructor.name'", | 
					
						
							| 
									
										
										
										
											2022-03-15 01:42:38 +01:00
										 |  |  |         detectInCode("constructor\\.name", "This is not allowed, as minification does erase names.") | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |     ) | 
					
						
							| 
									
										
										
										
											2022-06-28 03:21:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-16 01:34:57 +02:00
										 |  |  |     itAsync( | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |         "should not contain 'innerText'", | 
					
						
							| 
									
										
										
										
											2022-06-28 03:21:18 +02:00
										 |  |  |         detectInCode( | 
					
						
							|  |  |  |             "innerText", | 
					
						
							|  |  |  |             "innerText is not allowed as it is not testable with fakeDom. Use 'textContent' instead." | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-16 01:34:57 +02:00
										 |  |  |     itAsync( | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |         "should not contain 'import * as name from \"xyz.json\"'", | 
					
						
							|  |  |  |         detectInCode( | 
					
						
							|  |  |  |             'import \\* as [a-zA-Z0-9_]\\+ from \\"[.-_/a-zA-Z0-9]\\+\\.json\\"', | 
					
						
							|  |  |  |             "With vite, json files have a default export. Use import name from file.json instead" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-16 01:34:57 +02:00
										 |  |  |     itAsync( | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  |         "should not contain '[\"default\"]'", | 
					
						
							|  |  |  |         detectInCode('\\[\\"default\\"\\]', "Possible leftover of faulty default import") | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2022-03-15 01:42:38 +01:00
										 |  |  | }) |