react-doctor 0.0.7 → 0.0.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/cli.js CHANGED
@@ -73,6 +73,7 @@ const SCORE_GOOD_THRESHOLD = 75;
73
73
  const SCORE_OK_THRESHOLD = 50;
74
74
  const SCORE_BAR_WIDTH_CHARS = 50;
75
75
  const SCORE_API_URL = "https://www.react.doctor/api/score";
76
+ const SHARE_BASE_URL = "https://www.react.doctor/share";
76
77
  const OFFLINE_MESSAGE = "You are offline, could not calculate score. Reconnect to calculate.";
77
78
 
78
79
  //#endregion
@@ -974,6 +975,17 @@ const printBranding = (score) => {
974
975
  logger.log(` React Doctor ${highlighter.dim("(www.react.doctor)")}`);
975
976
  logger.break();
976
977
  };
978
+ const buildShareUrl = (diagnostics, scoreResult) => {
979
+ const errorCount = diagnostics.filter((diagnostic) => diagnostic.severity === "error").length;
980
+ const warningCount = diagnostics.filter((diagnostic) => diagnostic.severity === "warning").length;
981
+ const affectedFileCount = collectAffectedFiles(diagnostics).size;
982
+ const params = new URLSearchParams();
983
+ if (scoreResult) params.set("s", String(scoreResult.score));
984
+ if (errorCount > 0) params.set("e", String(errorCount));
985
+ if (warningCount > 0) params.set("w", String(warningCount));
986
+ if (affectedFileCount > 0) params.set("f", String(affectedFileCount));
987
+ return `${SHARE_BASE_URL}?${params.toString()}`;
988
+ };
977
989
  const printSummary = (diagnostics, elapsedMilliseconds, scoreResult) => {
978
990
  const errorCount = diagnostics.filter((diagnostic) => diagnostic.severity === "error").length;
979
991
  const warningCount = diagnostics.filter((diagnostic) => diagnostic.severity === "warning").length;
@@ -1000,6 +1012,9 @@ const printSummary = (diagnostics, elapsedMilliseconds, scoreResult) => {
1000
1012
  } catch {
1001
1013
  logger.break();
1002
1014
  }
1015
+ const shareUrl = buildShareUrl(diagnostics, scoreResult);
1016
+ logger.break();
1017
+ logger.dim(` Share your results: ${highlighter.info(shareUrl)}`);
1003
1018
  };
1004
1019
  const scan = async (directory, options) => {
1005
1020
  const startTime = performance.now();
@@ -1160,8 +1175,9 @@ const maybePromptSkillInstall = async (shouldSkipPrompts) => {
1160
1175
  if (config.skillPromptDismissed) return;
1161
1176
  if (shouldSkipPrompts) return;
1162
1177
  logger.break();
1163
- logger.log(`${highlighter.info("💡")} Install the ${highlighter.info("react-doctor")} skill for your coding agent?`);
1164
- logger.dim(" Adds React best practices to Cursor, Claude Code, Copilot, and more.");
1178
+ logger.log(`${highlighter.info("💡")} Have your coding agent fix these issues automatically?`);
1179
+ logger.dim(` Install the ${highlighter.info("react-doctor")} skill to teach Cursor, Claude Code, Copilot,`);
1180
+ logger.dim(" Ami, and other AI agents how to diagnose and fix these React issues.");
1165
1181
  logger.break();
1166
1182
  const { shouldInstall } = await prompts({
1167
1183
  type: "confirm",
@@ -1188,7 +1204,7 @@ const maybePromptSkillInstall = async (shouldSkipPrompts) => {
1188
1204
 
1189
1205
  //#endregion
1190
1206
  //#region src/cli.ts
1191
- const VERSION = "0.0.7";
1207
+ const VERSION = "0.0.8";
1192
1208
  process.on("SIGINT", () => process.exit(0));
1193
1209
  process.on("SIGTERM", () => process.exit(0));
1194
1210
  const program = new Command().name("react-doctor").description("Diagnose React codebase health").version(VERSION, "-v, --version", "display the version number").argument("[directory]", "project directory to scan", ".").option("--no-lint", "skip linting").option("--no-dead-code", "skip dead code detection").option("--verbose", "show file details per rule").option("--score", "output only the score").option("-y, --yes", "skip prompts, scan all workspace projects").option("--project <name>", "select workspace project (comma-separated for multiple)").option("--fix", "open Ami to auto-fix all issues").action(async (directory, flags) => {
@@ -1205,7 +1221,17 @@ const program = new Command().name("react-doctor").description("Diagnose React c
1205
1221
  verbose: Boolean(flags.verbose),
1206
1222
  scoreOnly: isScoreOnly
1207
1223
  };
1208
- const shouldSkipPrompts = flags.yes || Boolean(process.env.CI) || Boolean(process.env.CLAUDECODE) || Boolean(process.env.AMI) || !process.stdin.isTTY;
1224
+ const isAutomatedEnvironment = [
1225
+ process.env.CI,
1226
+ process.env.CLAUDECODE,
1227
+ process.env.CURSOR_TRACE_ID,
1228
+ process.env.CURSOR_AGENT,
1229
+ process.env.CODEX_CI,
1230
+ process.env.OPENCODE,
1231
+ process.env.AMP_HOME,
1232
+ process.env.AMI
1233
+ ].some(Boolean);
1234
+ const shouldSkipPrompts = flags.yes || isAutomatedEnvironment || !process.stdin.isTTY;
1209
1235
  const projectDirectories = await selectProjects(resolvedDirectory, flags.project, shouldSkipPrompts);
1210
1236
  for (const projectDirectory of projectDirectories) {
1211
1237
  if (!isScoreOnly) {
@@ -1216,7 +1242,10 @@ const program = new Command().name("react-doctor").description("Diagnose React c
1216
1242
  if (!isScoreOnly) logger.break();
1217
1243
  }
1218
1244
  if (flags.fix) openAmiToFix(resolvedDirectory);
1219
- if (!isScoreOnly) await maybePromptSkillInstall(shouldSkipPrompts);
1245
+ if (!isScoreOnly) {
1246
+ await maybePromptSkillInstall(shouldSkipPrompts);
1247
+ if (!shouldSkipPrompts && !flags.fix) await maybePromptAmiFix(resolvedDirectory);
1248
+ }
1220
1249
  } catch (error) {
1221
1250
  handleError(error);
1222
1251
  }
@@ -1226,7 +1255,6 @@ ${highlighter.dim("Learn more:")}
1226
1255
  `);
1227
1256
  const AMI_INSTALL_URL = "https://ami.dev/install.sh";
1228
1257
  const AMI_FIX_PROMPT = "Run npx -y react-doctor@latest . --verbose, read every diagnostic, then fix all issues one by one. After fixing, re-run react-doctor to verify the score improved.";
1229
- const OPEN_PROJECT_DELAY_S = 2;
1230
1258
  const isAmiInstalled = () => {
1231
1259
  try {
1232
1260
  execSync("ls /Applications/Ami.app", { stdio: "ignore" });
@@ -1250,20 +1278,30 @@ const openAmiToFix = (directory) => {
1250
1278
  const resolvedDirectory = path.resolve(directory);
1251
1279
  if (!isAmiInstalled()) installAmi();
1252
1280
  logger.log("Opening Ami to fix react-doctor issues...");
1253
- const encodedDirectory = encodeURIComponent(resolvedDirectory);
1254
- const encodedPrompt = encodeURIComponent(AMI_FIX_PROMPT);
1255
- const openProjectDeeplink = `ami://open-project?cwd=${encodedDirectory}`;
1256
- const newChatDeeplink = `ami://new-chat?prompt=${encodedPrompt}&mode=agent&send=true`;
1281
+ const deeplink = `ami://open-project?cwd=${encodeURIComponent(resolvedDirectory)}&prompt=${encodeURIComponent(AMI_FIX_PROMPT)}&mode=agent`;
1257
1282
  try {
1258
- execSync(`open "${openProjectDeeplink}" && sleep ${OPEN_PROJECT_DELAY_S} && open "${newChatDeeplink}"`, { stdio: "ignore" });
1283
+ execSync(`open "${deeplink}"`, { stdio: "ignore" });
1259
1284
  logger.success("Opened Ami with react-doctor fix prompt.");
1260
1285
  } catch {
1261
1286
  logger.break();
1262
- logger.dim("Could not open Ami automatically. Open these URLs manually:");
1263
- logger.info(openProjectDeeplink);
1264
- logger.info(newChatDeeplink);
1287
+ logger.dim("Could not open Ami automatically. Open this URL manually:");
1288
+ logger.info(deeplink);
1265
1289
  }
1266
1290
  };
1291
+ const maybePromptAmiFix = async (directory) => {
1292
+ logger.break();
1293
+ logger.log(`Fix these issues with ${highlighter.info("Ami")}?`);
1294
+ logger.dim(" Ami is a coding agent built to understand your codebase and fix issues");
1295
+ logger.dim(` automatically. Learn more at ${highlighter.info("https://ami.dev")}`);
1296
+ logger.break();
1297
+ const { shouldFix } = await prompts({
1298
+ type: "confirm",
1299
+ name: "shouldFix",
1300
+ message: "Open Ami to fix?",
1301
+ initial: true
1302
+ });
1303
+ if (shouldFix) openAmiToFix(directory);
1304
+ };
1267
1305
  const fixAction = (directory) => {
1268
1306
  try {
1269
1307
  openAmiToFix(directory);
@@ -1272,7 +1310,7 @@ const fixAction = (directory) => {
1272
1310
  }
1273
1311
  };
1274
1312
  const fixCommand = new Command("fix").description("Open Ami to auto-fix react-doctor issues").argument("[directory]", "project directory", ".").action(fixAction);
1275
- const installAmiCommand = new Command("install-ami").description("Open Ami to auto-fix react-doctor issues").argument("[directory]", "project directory", ".").action(fixAction);
1313
+ const installAmiCommand = new Command("install-ami").description("Install Ami and open it to auto-fix issues").argument("[directory]", "project directory", ".").action(fixAction);
1276
1314
  program.addCommand(fixCommand);
1277
1315
  program.addCommand(installAmiCommand);
1278
1316
  const main$1 = async () => {
package/dist/cli.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"cli.js","names":["esmRequire","main"],"sources":["../src/utils/highlighter.ts","../src/utils/logger.ts","../src/utils/handle-error.ts","../src/constants.ts","../src/utils/calculate-score.ts","../src/utils/read-package-json.ts","../src/utils/discover-project.ts","../src/utils/group-by.ts","../src/plugin/constants.ts","../src/utils/check-reduced-motion.ts","../src/utils/run-knip.ts","../src/oxlint-config.ts","../src/utils/run-oxlint.ts","../src/utils/spinner.ts","../src/scan.ts","../src/utils/prompts.ts","../src/utils/select-projects.ts","../src/utils/skill-prompt.ts","../src/cli.ts"],"sourcesContent":["import pc from \"picocolors\";\n\nexport const highlighter = {\n error: pc.red,\n warn: pc.yellow,\n info: pc.cyan,\n success: pc.green,\n dim: pc.dim,\n};\n","import { highlighter } from \"./highlighter.js\";\n\nexport const logger = {\n error(...args: unknown[]) {\n console.log(highlighter.error(args.join(\" \")));\n },\n warn(...args: unknown[]) {\n console.log(highlighter.warn(args.join(\" \")));\n },\n info(...args: unknown[]) {\n console.log(highlighter.info(args.join(\" \")));\n },\n success(...args: unknown[]) {\n console.log(highlighter.success(args.join(\" \")));\n },\n dim(...args: unknown[]) {\n console.log(highlighter.dim(args.join(\" \")));\n },\n log(...args: unknown[]) {\n console.log(args.join(\" \"));\n },\n break() {\n console.log(\"\");\n },\n};\n","import { logger } from \"./logger.js\";\n\nexport const handleError = (error: unknown): void => {\n logger.break();\n logger.error(\"Something went wrong. Please check the error below for more details.\");\n logger.error(\"If the problem persists, please open an issue on GitHub.\");\n logger.error(\"\");\n if (error instanceof Error) {\n logger.error(error.message);\n }\n logger.break();\n process.exit(1);\n};\n","export const SOURCE_FILE_PATTERN = /\\.(tsx?|jsx?)$/;\n\nexport const JSX_FILE_PATTERN = /\\.(tsx|jsx)$/;\n\nexport const MILLISECONDS_PER_SECOND = 1000;\n\nexport const SEPARATOR_LENGTH_CHARS = 62;\n\nexport const ERROR_PREVIEW_LENGTH_CHARS = 200;\n\nexport const PERFECT_SCORE = 100;\n\nexport const SCORE_GOOD_THRESHOLD = 75;\n\nexport const SCORE_OK_THRESHOLD = 50;\n\nexport const SCORE_BAR_WIDTH_CHARS = 50;\n\nexport const SCORE_API_URL = \"https://www.react.doctor/api/score\";\n\nexport const OFFLINE_MESSAGE =\n \"You are offline, could not calculate score. Reconnect to calculate.\";\n","import { SCORE_API_URL } from \"../constants.js\";\nimport type { Diagnostic, ScoreResult } from \"../types.js\";\n\nexport const calculateScore = async (diagnostics: Diagnostic[]): Promise<ScoreResult | null> => {\n const payload = diagnostics.map((diagnostic) => ({\n plugin: diagnostic.plugin,\n rule: diagnostic.rule,\n severity: diagnostic.severity,\n }));\n\n try {\n const response = await fetch(SCORE_API_URL, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ diagnostics: payload }),\n });\n\n if (!response.ok) return null;\n\n return (await response.json()) as ScoreResult;\n } catch {\n return null;\n }\n};\n","import fs from \"node:fs\";\nimport type { PackageJson } from \"../types.js\";\n\nexport const readPackageJson = (packageJsonPath: string): PackageJson =>\n JSON.parse(fs.readFileSync(packageJsonPath, \"utf-8\"));\n","import fs from \"node:fs\";\nimport path from \"node:path\";\nimport { spawnSync } from \"node:child_process\";\nimport { SOURCE_FILE_PATTERN } from \"../constants.js\";\nimport type {\n DependencyInfo,\n Framework,\n PackageJson,\n ProjectInfo,\n WorkspacePackage,\n} from \"../types.js\";\nimport { readPackageJson } from \"./read-package-json.js\";\n\nconst REACT_COMPILER_PACKAGES = new Set([\n \"babel-plugin-react-compiler\",\n \"react-compiler-runtime\",\n \"eslint-plugin-react-compiler\",\n]);\n\nconst NEXT_CONFIG_FILENAMES = [\n \"next.config.js\",\n \"next.config.mjs\",\n \"next.config.ts\",\n \"next.config.cjs\",\n];\n\nconst BABEL_CONFIG_FILENAMES = [\n \".babelrc\",\n \".babelrc.json\",\n \"babel.config.js\",\n \"babel.config.json\",\n \"babel.config.cjs\",\n \"babel.config.mjs\",\n];\n\nconst VITE_CONFIG_FILENAMES = [\n \"vite.config.js\",\n \"vite.config.ts\",\n \"vite.config.mjs\",\n \"vite.config.cjs\",\n];\n\nconst REACT_COMPILER_CONFIG_PATTERN = /react-compiler|reactCompiler/;\n\nconst FRAMEWORK_PACKAGES: Record<string, Framework> = {\n next: \"nextjs\",\n vite: \"vite\",\n \"react-scripts\": \"cra\",\n \"@remix-run/react\": \"remix\",\n gatsby: \"gatsby\",\n};\n\nconst FRAMEWORK_DISPLAY_NAMES: Record<Framework, string> = {\n nextjs: \"Next.js\",\n vite: \"Vite\",\n cra: \"Create React App\",\n remix: \"Remix\",\n gatsby: \"Gatsby\",\n unknown: \"React\",\n};\n\nexport const formatFrameworkName = (framework: Framework): string =>\n FRAMEWORK_DISPLAY_NAMES[framework];\n\nconst countSourceFiles = (rootDirectory: string): number => {\n const result = spawnSync(\"git\", [\"ls-files\", \"--cached\", \"--others\", \"--exclude-standard\"], {\n cwd: rootDirectory,\n encoding: \"utf-8\",\n });\n\n if (result.error || result.status !== 0) {\n return 0;\n }\n\n return result.stdout\n .split(\"\\n\")\n .filter((filePath) => filePath.length > 0 && SOURCE_FILE_PATTERN.test(filePath)).length;\n};\n\nconst collectAllDependencies = (packageJson: PackageJson): Record<string, string> => ({\n ...packageJson.dependencies,\n ...packageJson.devDependencies,\n});\n\nconst detectFramework = (dependencies: Record<string, string>): Framework => {\n for (const [packageName, frameworkName] of Object.entries(FRAMEWORK_PACKAGES)) {\n if (dependencies[packageName]) {\n return frameworkName;\n }\n }\n return \"unknown\";\n};\n\nconst extractDependencyInfo = (packageJson: PackageJson): DependencyInfo => {\n const allDependencies = collectAllDependencies(packageJson);\n return {\n reactVersion: allDependencies.react ?? null,\n framework: detectFramework(allDependencies),\n };\n};\n\nconst parsePnpmWorkspacePatterns = (rootDirectory: string): string[] => {\n const workspacePath = path.join(rootDirectory, \"pnpm-workspace.yaml\");\n if (!fs.existsSync(workspacePath)) return [];\n\n const content = fs.readFileSync(workspacePath, \"utf-8\");\n const patterns: string[] = [];\n let isInsidePackagesBlock = false;\n\n for (const line of content.split(\"\\n\")) {\n const trimmed = line.trim();\n if (trimmed === \"packages:\") {\n isInsidePackagesBlock = true;\n continue;\n }\n if (isInsidePackagesBlock && trimmed.startsWith(\"-\")) {\n patterns.push(trimmed.replace(/^-\\s*/, \"\").replace(/[\"']/g, \"\"));\n } else if (isInsidePackagesBlock && trimmed.length > 0 && !trimmed.startsWith(\"#\")) {\n isInsidePackagesBlock = false;\n }\n }\n\n return patterns;\n};\n\nconst getWorkspacePatterns = (rootDirectory: string, packageJson: PackageJson): string[] => {\n const pnpmPatterns = parsePnpmWorkspacePatterns(rootDirectory);\n if (pnpmPatterns.length > 0) return pnpmPatterns;\n\n if (Array.isArray(packageJson.workspaces)) {\n return packageJson.workspaces;\n }\n\n if (packageJson.workspaces?.packages) {\n return packageJson.workspaces.packages;\n }\n\n return [];\n};\n\nconst resolveWorkspaceDirectories = (rootDirectory: string, pattern: string): string[] => {\n const cleanPattern = pattern.replace(/[\"']/g, \"\").replace(/\\/\\*\\*$/, \"/*\");\n\n if (!cleanPattern.includes(\"*\")) {\n const directoryPath = path.join(rootDirectory, cleanPattern);\n if (fs.existsSync(directoryPath) && fs.existsSync(path.join(directoryPath, \"package.json\"))) {\n return [directoryPath];\n }\n return [];\n }\n\n const baseDirectory = path.join(rootDirectory, cleanPattern.slice(0, cleanPattern.indexOf(\"*\")));\n\n if (!fs.existsSync(baseDirectory) || !fs.statSync(baseDirectory).isDirectory()) {\n return [];\n }\n\n return fs\n .readdirSync(baseDirectory)\n .map((entry) => path.join(baseDirectory, entry))\n .filter(\n (entryPath) =>\n fs.statSync(entryPath).isDirectory() && fs.existsSync(path.join(entryPath, \"package.json\")),\n );\n};\n\nconst findDependencyInfoFromAncestors = (startDirectory: string): DependencyInfo => {\n let currentDirectory = path.dirname(startDirectory);\n const result: DependencyInfo = { reactVersion: null, framework: \"unknown\" };\n\n while (currentDirectory !== path.dirname(currentDirectory)) {\n const packageJsonPath = path.join(currentDirectory, \"package.json\");\n if (fs.existsSync(packageJsonPath)) {\n const packageJson = readPackageJson(packageJsonPath);\n const info = extractDependencyInfo(packageJson);\n\n if (!result.reactVersion && info.reactVersion) {\n result.reactVersion = info.reactVersion;\n }\n if (result.framework === \"unknown\" && info.framework !== \"unknown\") {\n result.framework = info.framework;\n }\n\n if (result.reactVersion && result.framework !== \"unknown\") {\n return result;\n }\n }\n\n currentDirectory = path.dirname(currentDirectory);\n }\n\n return result;\n};\n\nconst findReactInWorkspaces = (rootDirectory: string, packageJson: PackageJson): DependencyInfo => {\n const patterns = getWorkspacePatterns(rootDirectory, packageJson);\n const result: DependencyInfo = { reactVersion: null, framework: \"unknown\" };\n\n for (const pattern of patterns) {\n const directories = resolveWorkspaceDirectories(rootDirectory, pattern);\n\n for (const workspaceDirectory of directories) {\n const workspacePackageJson = readPackageJson(path.join(workspaceDirectory, \"package.json\"));\n const info = extractDependencyInfo(workspacePackageJson);\n\n if (info.reactVersion && !result.reactVersion) {\n result.reactVersion = info.reactVersion;\n }\n if (info.framework !== \"unknown\" && result.framework === \"unknown\") {\n result.framework = info.framework;\n }\n\n if (result.reactVersion && result.framework !== \"unknown\") {\n return result;\n }\n }\n }\n\n return result;\n};\n\nconst hasReactDependency = (packageJson: PackageJson): boolean => {\n const allDependencies = collectAllDependencies(packageJson);\n return Object.keys(allDependencies).some(\n (packageName) => packageName === \"next\" || packageName.includes(\"react\"),\n );\n};\n\nexport const discoverReactSubprojects = (rootDirectory: string): WorkspacePackage[] => {\n if (!fs.existsSync(rootDirectory) || !fs.statSync(rootDirectory).isDirectory()) return [];\n\n const entries = fs.readdirSync(rootDirectory, { withFileTypes: true });\n const packages: WorkspacePackage[] = [];\n\n for (const entry of entries) {\n if (!entry.isDirectory() || entry.name.startsWith(\".\") || entry.name === \"node_modules\") {\n continue;\n }\n\n const subdirectory = path.join(rootDirectory, entry.name);\n const packageJsonPath = path.join(subdirectory, \"package.json\");\n if (!fs.existsSync(packageJsonPath)) continue;\n\n const packageJson = readPackageJson(packageJsonPath);\n if (!hasReactDependency(packageJson)) continue;\n\n const name = packageJson.name ?? entry.name;\n packages.push({ name, directory: subdirectory });\n }\n\n return packages;\n};\n\nexport const listWorkspacePackages = (rootDirectory: string): WorkspacePackage[] => {\n const packageJsonPath = path.join(rootDirectory, \"package.json\");\n if (!fs.existsSync(packageJsonPath)) return [];\n\n const packageJson = readPackageJson(packageJsonPath);\n const patterns = getWorkspacePatterns(rootDirectory, packageJson);\n if (patterns.length === 0) return [];\n\n const packages: WorkspacePackage[] = [];\n\n for (const pattern of patterns) {\n const directories = resolveWorkspaceDirectories(rootDirectory, pattern);\n for (const workspaceDirectory of directories) {\n const workspacePackageJson = readPackageJson(path.join(workspaceDirectory, \"package.json\"));\n\n if (!hasReactDependency(workspacePackageJson)) continue;\n\n const name = workspacePackageJson.name ?? path.basename(workspaceDirectory);\n packages.push({ name, directory: workspaceDirectory });\n }\n }\n\n return packages;\n};\n\nconst hasCompilerPackage = (packageJson: PackageJson): boolean => {\n const allDependencies = collectAllDependencies(packageJson);\n return Object.keys(allDependencies).some((packageName) =>\n REACT_COMPILER_PACKAGES.has(packageName),\n );\n};\n\nconst fileContainsPattern = (filePath: string, pattern: RegExp): boolean => {\n if (!fs.existsSync(filePath)) return false;\n const content = fs.readFileSync(filePath, \"utf-8\");\n return pattern.test(content);\n};\n\nconst hasCompilerInConfigFiles = (directory: string, filenames: string[]): boolean =>\n filenames.some((filename) =>\n fileContainsPattern(path.join(directory, filename), REACT_COMPILER_CONFIG_PATTERN),\n );\n\nconst detectReactCompiler = (directory: string, packageJson: PackageJson): boolean => {\n if (hasCompilerPackage(packageJson)) return true;\n\n if (hasCompilerInConfigFiles(directory, NEXT_CONFIG_FILENAMES)) return true;\n if (hasCompilerInConfigFiles(directory, BABEL_CONFIG_FILENAMES)) return true;\n if (hasCompilerInConfigFiles(directory, VITE_CONFIG_FILENAMES)) return true;\n\n let ancestorDirectory = path.dirname(directory);\n while (ancestorDirectory !== path.dirname(ancestorDirectory)) {\n const ancestorPackagePath = path.join(ancestorDirectory, \"package.json\");\n if (fs.existsSync(ancestorPackagePath)) {\n const ancestorPackageJson = readPackageJson(ancestorPackagePath);\n if (hasCompilerPackage(ancestorPackageJson)) return true;\n }\n ancestorDirectory = path.dirname(ancestorDirectory);\n }\n\n return false;\n};\n\nexport const discoverProject = (directory: string): ProjectInfo => {\n const packageJsonPath = path.join(directory, \"package.json\");\n if (!fs.existsSync(packageJsonPath)) {\n throw new Error(`No package.json found in ${directory}`);\n }\n\n const packageJson = readPackageJson(packageJsonPath);\n let { reactVersion, framework } = extractDependencyInfo(packageJson);\n\n if (!reactVersion || framework === \"unknown\") {\n const workspaceInfo = findReactInWorkspaces(directory, packageJson);\n if (!reactVersion && workspaceInfo.reactVersion) {\n reactVersion = workspaceInfo.reactVersion;\n }\n if (framework === \"unknown\" && workspaceInfo.framework !== \"unknown\") {\n framework = workspaceInfo.framework;\n }\n }\n\n if (!reactVersion || framework === \"unknown\") {\n const ancestorInfo = findDependencyInfoFromAncestors(directory);\n if (!reactVersion) {\n reactVersion = ancestorInfo.reactVersion;\n }\n if (framework === \"unknown\") {\n framework = ancestorInfo.framework;\n }\n }\n\n const hasTypeScript = fs.existsSync(path.join(directory, \"tsconfig.json\"));\n const sourceFileCount = countSourceFiles(directory);\n\n const hasReactCompiler = detectReactCompiler(directory, packageJson);\n\n return {\n rootDirectory: directory,\n reactVersion,\n framework,\n hasTypeScript,\n hasReactCompiler,\n sourceFileCount,\n };\n};\n","export const groupBy = <T>(items: T[], keyFn: (item: T) => string): Map<string, T[]> => {\n const groups = new Map<string, T[]>();\n\n for (const item of items) {\n const key = keyFn(item);\n const existing = groups.get(key) ?? [];\n existing.push(item);\n groups.set(key, existing);\n }\n\n return groups;\n};\n","export const GIANT_COMPONENT_LINE_THRESHOLD = 200;\nexport const CASCADING_SET_STATE_THRESHOLD = 2;\nexport const RELATED_USE_STATE_THRESHOLD = 3;\nexport const DEEP_NESTING_THRESHOLD = 3;\nexport const DUPLICATE_STORAGE_READ_THRESHOLD = 2;\nexport const SEQUENTIAL_AWAIT_THRESHOLD = 2;\nexport const SECRET_MIN_LENGTH_CHARS = 8;\nexport const AUTH_CHECK_LOOKAHEAD_STATEMENTS = 3;\n\nexport const LAYOUT_PROPERTIES = new Set([\n \"width\",\n \"height\",\n \"top\",\n \"left\",\n \"right\",\n \"bottom\",\n \"padding\",\n \"paddingTop\",\n \"paddingRight\",\n \"paddingBottom\",\n \"paddingLeft\",\n \"margin\",\n \"marginTop\",\n \"marginRight\",\n \"marginBottom\",\n \"marginLeft\",\n \"borderWidth\",\n \"fontSize\",\n \"lineHeight\",\n \"gap\",\n]);\n\nexport const MOTION_ANIMATE_PROPS = new Set([\n \"animate\",\n \"initial\",\n \"exit\",\n \"whileHover\",\n \"whileTap\",\n \"whileFocus\",\n \"whileDrag\",\n \"whileInView\",\n]);\n\nexport const HEAVY_LIBRARIES = new Set([\n \"@monaco-editor/react\",\n \"monaco-editor\",\n \"recharts\",\n \"@react-pdf/renderer\",\n \"react-quill\",\n \"@codemirror/view\",\n \"@codemirror/state\",\n \"chart.js\",\n \"react-chartjs-2\",\n \"@toast-ui/editor\",\n \"draft-js\",\n]);\n\nexport const FETCH_CALLEE_NAMES = new Set([\"fetch\"]);\nexport const FETCH_MEMBER_OBJECTS = new Set([\"axios\", \"ky\", \"got\"]);\nexport const INDEX_PARAMETER_NAMES = new Set([\"index\", \"idx\", \"i\"]);\nexport const BARREL_INDEX_SUFFIXES = [\n \"/index\",\n \"/index.js\",\n \"/index.ts\",\n \"/index.tsx\",\n \"/index.mjs\",\n];\nexport const PASSIVE_EVENT_NAMES = new Set([\n \"scroll\",\n \"wheel\",\n \"touchstart\",\n \"touchmove\",\n \"touchend\",\n]);\n\nexport const LOOP_TYPES = [\n \"ForStatement\",\n \"ForInStatement\",\n \"ForOfStatement\",\n \"WhileStatement\",\n \"DoWhileStatement\",\n];\n\nexport const AUTH_FUNCTION_NAMES = new Set([\n \"auth\",\n \"getSession\",\n \"getServerSession\",\n \"getUser\",\n \"requireAuth\",\n \"checkAuth\",\n \"verifyAuth\",\n \"authenticate\",\n \"currentUser\",\n \"getAuth\",\n \"validateSession\",\n]);\n\nexport const SECRET_PATTERNS = [\n /^sk_live_/,\n /^sk_test_/,\n /^AKIA[0-9A-Z]{16}$/,\n /^ghp_[a-zA-Z0-9]{36}$/,\n /^gho_[a-zA-Z0-9]{36}$/,\n /^github_pat_/,\n /^glpat-/,\n /^xox[bporas]-/,\n /^sk-[a-zA-Z0-9]{32,}$/,\n];\n\nexport const SECRET_VARIABLE_PATTERN = /(?:api_?key|secret|token|password|credential|auth)/i;\n\nexport const LOADING_STATE_PATTERN = /(?:loading|isLoading|isPending|isSubmitting|isFetching)/i;\n\nexport const EVENT_PROP_PATTERN = /^on[A-Z]/;\nexport const SETTER_PATTERN = /^set[A-Z]/;\nexport const RENDER_FUNCTION_PATTERN = /^render[A-Z]/;\nexport const UPPERCASE_PATTERN = /^[A-Z]/;\nexport const PAGE_FILE_PATTERN = /\\/page\\.(tsx?|jsx?)$/;\nexport const PAGE_OR_LAYOUT_FILE_PATTERN = /\\/(page|layout)\\.(tsx?|jsx?)$/;\nexport const PAGES_DIRECTORY_PATTERN = /\\/pages\\//;\nexport const SERVER_ACTION_FILE_PATTERN = /actions?\\.(tsx?|jsx?)$/;\nexport const SERVER_ACTION_DIRECTORY_PATTERN = /\\/actions\\//;\n\nexport const NEXTJS_NAVIGATION_FUNCTIONS = new Set([\n \"redirect\",\n \"permanentRedirect\",\n \"notFound\",\n \"forbidden\",\n \"unauthorized\",\n]);\n\nexport const GOOGLE_FONTS_PATTERN = /fonts\\.googleapis\\.com/;\n\nexport const POLYFILL_SCRIPT_PATTERN = /polyfill\\.io|polyfill\\.min\\.js|cdn\\.polyfill/;\n\nexport const APP_DIRECTORY_PATTERN = /\\/app\\//;\n\nexport const ROUTE_HANDLER_FILE_PATTERN = /\\/route\\.(tsx?|jsx?)$/;\n\nexport const MUTATION_METHOD_NAMES = new Set([\n \"create\",\n \"insert\",\n \"insertInto\",\n \"update\",\n \"upsert\",\n \"delete\",\n \"remove\",\n \"destroy\",\n \"set\",\n \"append\",\n]);\n\nexport const MUTATING_HTTP_METHODS = new Set([\"POST\", \"PUT\", \"DELETE\", \"PATCH\"]);\n\nexport const MUTATING_ROUTE_SEGMENTS = new Set([\n \"logout\",\n \"log-out\",\n \"signout\",\n \"sign-out\",\n \"unsubscribe\",\n \"delete\",\n \"remove\",\n \"revoke\",\n \"cancel\",\n \"deactivate\",\n]);\n\nexport const EFFECT_HOOK_NAMES = new Set([\"useEffect\", \"useLayoutEffect\"]);\nexport const HOOKS_WITH_DEPS = new Set([\"useEffect\", \"useLayoutEffect\", \"useMemo\", \"useCallback\"]);\nexport const CHAINABLE_ITERATION_METHODS = new Set([\"map\", \"filter\", \"forEach\", \"flatMap\"]);\nexport const STORAGE_OBJECTS = new Set([\"localStorage\", \"sessionStorage\"]);\n\nexport const LARGE_BLUR_THRESHOLD_PX = 10;\nexport const BLUR_VALUE_PATTERN = /blur\\((\\d+(?:\\.\\d+)?)px\\)/;\nexport const ANIMATION_CALLBACK_NAMES = new Set([\"requestAnimationFrame\", \"setInterval\"]);\nexport const MOTION_LIBRARY_PACKAGES = new Set([\"framer-motion\", \"motion\"]);\n","import { execSync } from \"node:child_process\";\nimport fs from \"node:fs\";\nimport path from \"node:path\";\nimport { MOTION_LIBRARY_PACKAGES } from \"../plugin/constants.js\";\nimport type { Diagnostic } from \"../types.js\";\nimport { readPackageJson } from \"./read-package-json.js\";\n\nconst REDUCED_MOTION_GREP_PATTERN = \"prefers-reduced-motion|useReducedMotion\";\nconst REDUCED_MOTION_FILE_GLOBS = '\"*.ts\" \"*.tsx\" \"*.js\" \"*.jsx\" \"*.css\" \"*.scss\"';\n\nconst MISSING_REDUCED_MOTION_DIAGNOSTIC: Diagnostic = {\n filePath: \"package.json\",\n plugin: \"react-doctor\",\n rule: \"require-reduced-motion\",\n severity: \"error\",\n message:\n \"Project uses a motion library but has no prefers-reduced-motion handling — required for accessibility (WCAG 2.3.3)\",\n help: \"Add `useReducedMotion()` from your animation library, or a `@media (prefers-reduced-motion: reduce)` CSS query\",\n line: 0,\n column: 0,\n category: \"Accessibility\",\n weight: 2,\n};\n\nexport const checkReducedMotion = (rootDirectory: string): Diagnostic[] => {\n const packageJsonPath = path.join(rootDirectory, \"package.json\");\n if (!fs.existsSync(packageJsonPath)) return [];\n\n let hasMotionLibrary = false;\n try {\n const packageJson = readPackageJson(packageJsonPath);\n const allDependencies = { ...packageJson.dependencies, ...packageJson.devDependencies };\n hasMotionLibrary = Object.keys(allDependencies).some((packageName) =>\n MOTION_LIBRARY_PACKAGES.has(packageName),\n );\n } catch {\n return [];\n }\n if (!hasMotionLibrary) return [];\n\n try {\n execSync(`git grep -ql -E \"${REDUCED_MOTION_GREP_PATTERN}\" -- ${REDUCED_MOTION_FILE_GLOBS}`, {\n cwd: rootDirectory,\n stdio: \"pipe\",\n });\n return [];\n } catch {\n return [MISSING_REDUCED_MOTION_DIAGNOSTIC];\n }\n};\n","import path from \"node:path\";\nimport { main } from \"knip\";\nimport { createOptions } from \"knip/session\";\nimport type { Diagnostic, KnipIssueRecords, KnipResults } from \"../types.js\";\n\nconst KNIP_CATEGORY_MAP: Record<string, string> = {\n files: \"Dead Code\",\n exports: \"Dead Code\",\n types: \"Dead Code\",\n duplicates: \"Dead Code\",\n};\n\nconst KNIP_MESSAGE_MAP: Record<string, string> = {\n files: \"Unused file\",\n exports: \"Unused export\",\n types: \"Unused type\",\n duplicates: \"Duplicate export\",\n};\n\nconst KNIP_SEVERITY_MAP: Record<string, \"error\" | \"warning\"> = {\n files: \"warning\",\n exports: \"warning\",\n types: \"warning\",\n duplicates: \"warning\",\n};\n\nconst collectIssueRecords = (\n records: KnipIssueRecords,\n issueType: string,\n rootDirectory: string,\n): Diagnostic[] => {\n const diagnostics: Diagnostic[] = [];\n\n for (const issues of Object.values(records)) {\n for (const issue of Object.values(issues)) {\n diagnostics.push({\n filePath: path.relative(rootDirectory, issue.filePath),\n plugin: \"knip\",\n rule: issueType,\n severity: KNIP_SEVERITY_MAP[issueType] ?? \"warning\",\n message: `${KNIP_MESSAGE_MAP[issueType]}: ${issue.symbol}`,\n help: \"\",\n line: 0,\n column: 0,\n category: KNIP_CATEGORY_MAP[issueType] ?? \"Dead Code\",\n weight: 1,\n });\n }\n }\n\n return diagnostics;\n};\n\n// HACK: knip triggers dotenv which logs to stdout/stderr via console methods\nconst silenced = async <T>(fn: () => Promise<T>): Promise<T> => {\n const originalLog = console.log;\n const originalInfo = console.info;\n const originalWarn = console.warn;\n console.log = () => {};\n console.info = () => {};\n console.warn = () => {};\n try {\n return await fn();\n } finally {\n console.log = originalLog;\n console.info = originalInfo;\n console.warn = originalWarn;\n }\n};\n\nexport const runKnip = async (rootDirectory: string): Promise<Diagnostic[]> => {\n const options = await silenced(() =>\n createOptions({\n cwd: rootDirectory,\n isShowProgress: false,\n }),\n );\n\n const { issues } = (await silenced(() => main(options))) as KnipResults;\n const diagnostics: Diagnostic[] = [];\n\n for (const unusedFile of issues.files) {\n diagnostics.push({\n filePath: path.relative(rootDirectory, unusedFile),\n plugin: \"knip\",\n rule: \"files\",\n severity: KNIP_SEVERITY_MAP[\"files\"],\n message: KNIP_MESSAGE_MAP[\"files\"],\n help: \"This file is not imported by any other file in the project.\",\n line: 0,\n column: 0,\n category: KNIP_CATEGORY_MAP[\"files\"],\n weight: 1,\n });\n }\n\n const recordTypes = [\"exports\", \"types\", \"duplicates\"] as const;\n\n for (const issueType of recordTypes) {\n diagnostics.push(...collectIssueRecords(issues[issueType], issueType, rootDirectory));\n }\n\n return diagnostics;\n};\n","import { createRequire } from \"node:module\";\nimport type { Framework } from \"./types.js\";\n\nconst esmRequire = createRequire(import.meta.url);\n\nconst NEXTJS_RULES: Record<string, string> = {\n \"react-doctor/nextjs-no-img-element\": \"warn\",\n \"react-doctor/nextjs-async-client-component\": \"error\",\n \"react-doctor/nextjs-no-a-element\": \"warn\",\n \"react-doctor/nextjs-no-use-search-params-without-suspense\": \"error\",\n \"react-doctor/nextjs-no-client-fetch-for-server-data\": \"warn\",\n \"react-doctor/nextjs-missing-metadata\": \"warn\",\n \"react-doctor/nextjs-no-client-side-redirect\": \"warn\",\n \"react-doctor/nextjs-no-redirect-in-try-catch\": \"warn\",\n \"react-doctor/nextjs-image-missing-sizes\": \"warn\",\n \"react-doctor/nextjs-no-native-script\": \"warn\",\n \"react-doctor/nextjs-inline-script-missing-id\": \"warn\",\n \"react-doctor/nextjs-no-font-link\": \"warn\",\n \"react-doctor/nextjs-no-css-link\": \"warn\",\n \"react-doctor/nextjs-no-polyfill-script\": \"warn\",\n \"react-doctor/nextjs-no-head-import\": \"error\",\n \"react-doctor/nextjs-no-side-effect-in-get-handler\": \"error\",\n};\n\nconst REACT_COMPILER_RULES: Record<string, string> = {\n \"react-hooks-js/set-state-in-render\": \"error\",\n \"react-hooks-js/immutability\": \"error\",\n \"react-hooks-js/refs\": \"error\",\n \"react-hooks-js/purity\": \"error\",\n \"react-hooks-js/hooks\": \"error\",\n \"react-hooks-js/set-state-in-effect\": \"error\",\n \"react-hooks-js/globals\": \"error\",\n \"react-hooks-js/error-boundaries\": \"error\",\n \"react-hooks-js/preserve-manual-memoization\": \"error\",\n \"react-hooks-js/unsupported-syntax\": \"error\",\n \"react-hooks-js/component-hook-factories\": \"error\",\n \"react-hooks-js/static-components\": \"error\",\n \"react-hooks-js/use-memo\": \"error\",\n \"react-hooks-js/void-use-memo\": \"error\",\n \"react-hooks-js/incompatible-library\": \"error\",\n \"react-hooks-js/todo\": \"error\",\n};\n\nconst REACT_PERF_RULES: Record<string, string> = {\n \"react-perf/jsx-no-new-object-as-prop\": \"warn\",\n \"react-perf/jsx-no-new-array-as-prop\": \"warn\",\n \"react-perf/jsx-no-new-function-as-prop\": \"warn\",\n \"react-perf/jsx-no-jsx-as-prop\": \"warn\",\n};\n\ninterface OxlintConfigOptions {\n pluginPath: string;\n framework: Framework;\n hasReactCompiler: boolean;\n}\n\nexport const createOxlintConfig = ({\n pluginPath,\n framework,\n hasReactCompiler,\n}: OxlintConfigOptions) => ({\n categories: {\n correctness: \"off\",\n suspicious: \"off\",\n pedantic: \"off\",\n perf: \"off\",\n restriction: \"off\",\n style: \"off\",\n nursery: \"off\",\n },\n plugins: [\"react\", \"jsx-a11y\", ...(hasReactCompiler ? [] : [\"react-perf\"])],\n jsPlugins: [\n ...(hasReactCompiler\n ? [{ name: \"react-hooks-js\", specifier: esmRequire.resolve(\"eslint-plugin-react-hooks\") }]\n : []),\n pluginPath,\n ],\n rules: {\n \"react/rules-of-hooks\": \"error\",\n \"react/no-direct-mutation-state\": \"error\",\n \"react/jsx-no-duplicate-props\": \"error\",\n \"react/jsx-key\": \"error\",\n \"react/no-children-prop\": \"warn\",\n \"react/no-danger\": \"warn\",\n \"react/jsx-no-script-url\": \"error\",\n \"react/no-render-return-value\": \"warn\",\n \"react/no-string-refs\": \"warn\",\n \"react/no-unescaped-entities\": \"warn\",\n \"react/no-is-mounted\": \"warn\",\n \"react/require-render-return\": \"error\",\n \"react/no-unknown-property\": \"warn\",\n\n \"jsx-a11y/alt-text\": \"error\",\n \"jsx-a11y/anchor-is-valid\": \"warn\",\n \"jsx-a11y/click-events-have-key-events\": \"warn\",\n \"jsx-a11y/no-static-element-interactions\": \"warn\",\n \"jsx-a11y/no-noninteractive-element-interactions\": \"warn\",\n \"jsx-a11y/role-has-required-aria-props\": \"error\",\n \"jsx-a11y/no-autofocus\": \"warn\",\n \"jsx-a11y/heading-has-content\": \"warn\",\n \"jsx-a11y/html-has-lang\": \"warn\",\n \"jsx-a11y/no-redundant-roles\": \"warn\",\n \"jsx-a11y/scope\": \"warn\",\n \"jsx-a11y/tabindex-no-positive\": \"warn\",\n \"jsx-a11y/label-has-associated-control\": \"warn\",\n \"jsx-a11y/no-distracting-elements\": \"error\",\n \"jsx-a11y/iframe-has-title\": \"warn\",\n\n ...(hasReactCompiler ? REACT_COMPILER_RULES : REACT_PERF_RULES),\n\n \"react-doctor/no-derived-state-effect\": \"error\",\n \"react-doctor/no-fetch-in-effect\": \"error\",\n \"react-doctor/no-cascading-set-state\": \"warn\",\n \"react-doctor/no-effect-event-handler\": \"warn\",\n \"react-doctor/no-derived-useState\": \"warn\",\n \"react-doctor/prefer-useReducer\": \"warn\",\n \"react-doctor/rerender-lazy-state-init\": \"warn\",\n \"react-doctor/rerender-functional-setstate\": \"warn\",\n \"react-doctor/rerender-dependencies\": \"error\",\n\n \"react-doctor/no-generic-handler-names\": \"warn\",\n \"react-doctor/no-giant-component\": \"warn\",\n \"react-doctor/no-render-in-render\": \"warn\",\n \"react-doctor/no-nested-component-definition\": \"error\",\n\n \"react-doctor/no-usememo-simple-expression\": \"warn\",\n \"react-doctor/no-layout-property-animation\": \"error\",\n \"react-doctor/rerender-memo-with-default-value\": \"warn\",\n \"react-doctor/rendering-animate-svg-wrapper\": \"warn\",\n \"react-doctor/rendering-usetransition-loading\": \"warn\",\n \"react-doctor/rendering-hydration-no-flicker\": \"warn\",\n\n \"react-doctor/no-transition-all\": \"warn\",\n \"react-doctor/no-global-css-variable-animation\": \"error\",\n \"react-doctor/no-large-animated-blur\": \"warn\",\n \"react-doctor/no-scale-from-zero\": \"warn\",\n \"react-doctor/no-permanent-will-change\": \"warn\",\n\n \"react-doctor/no-secrets-in-client-code\": \"error\",\n\n \"react-doctor/no-barrel-import\": \"warn\",\n \"react-doctor/no-full-lodash-import\": \"warn\",\n \"react-doctor/no-moment\": \"warn\",\n \"react-doctor/prefer-dynamic-import\": \"warn\",\n \"react-doctor/use-lazy-motion\": \"warn\",\n \"react-doctor/no-undeferred-third-party\": \"warn\",\n\n \"react-doctor/no-array-index-as-key\": \"warn\",\n \"react-doctor/rendering-conditional-render\": \"warn\",\n \"react-doctor/no-prevent-default\": \"warn\",\n\n \"react-doctor/server-auth-actions\": \"error\",\n \"react-doctor/server-after-nonblocking\": \"warn\",\n\n \"react-doctor/client-passive-event-listeners\": \"warn\",\n\n \"react-doctor/async-parallel\": \"warn\",\n ...(framework === \"nextjs\" ? NEXTJS_RULES : {}),\n },\n});\n","import { spawn } from \"node:child_process\";\nimport fs from \"node:fs\";\nimport { createRequire } from \"node:module\";\nimport os from \"node:os\";\nimport path from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\nimport { ERROR_PREVIEW_LENGTH_CHARS, JSX_FILE_PATTERN } from \"../constants.js\";\nimport { createOxlintConfig } from \"../oxlint-config.js\";\nimport type { CleanedDiagnostic, Diagnostic, Framework, OxlintOutput } from \"../types.js\";\n\nconst esmRequire = createRequire(import.meta.url);\n\nconst PLUGIN_CATEGORY_MAP: Record<string, string> = {\n react: \"Correctness\",\n \"react-hooks\": \"Correctness\",\n \"react-hooks-js\": \"React Compiler\",\n \"react-perf\": \"Performance\",\n \"jsx-a11y\": \"Accessibility\",\n};\n\nconst RULE_CATEGORY_MAP: Record<string, string> = {\n \"react-doctor/no-derived-state-effect\": \"State & Effects\",\n \"react-doctor/no-fetch-in-effect\": \"State & Effects\",\n \"react-doctor/no-cascading-set-state\": \"State & Effects\",\n \"react-doctor/no-effect-event-handler\": \"State & Effects\",\n \"react-doctor/no-derived-useState\": \"State & Effects\",\n \"react-doctor/prefer-useReducer\": \"State & Effects\",\n \"react-doctor/rerender-lazy-state-init\": \"Performance\",\n \"react-doctor/rerender-functional-setstate\": \"Performance\",\n \"react-doctor/rerender-dependencies\": \"State & Effects\",\n\n \"react-doctor/no-generic-handler-names\": \"Architecture\",\n \"react-doctor/no-giant-component\": \"Architecture\",\n \"react-doctor/no-render-in-render\": \"Architecture\",\n \"react-doctor/no-nested-component-definition\": \"Correctness\",\n\n \"react-doctor/no-usememo-simple-expression\": \"Performance\",\n \"react-doctor/no-layout-property-animation\": \"Performance\",\n \"react-doctor/rerender-memo-with-default-value\": \"Performance\",\n \"react-doctor/rendering-animate-svg-wrapper\": \"Performance\",\n \"react-doctor/rendering-usetransition-loading\": \"Performance\",\n \"react-doctor/rendering-hydration-no-flicker\": \"Performance\",\n\n \"react-doctor/no-transition-all\": \"Performance\",\n \"react-doctor/no-global-css-variable-animation\": \"Performance\",\n \"react-doctor/no-large-animated-blur\": \"Performance\",\n \"react-doctor/no-scale-from-zero\": \"Performance\",\n \"react-doctor/no-permanent-will-change\": \"Performance\",\n\n \"react-doctor/no-secrets-in-client-code\": \"Security\",\n\n \"react-doctor/no-barrel-import\": \"Bundle Size\",\n \"react-doctor/no-full-lodash-import\": \"Bundle Size\",\n \"react-doctor/no-moment\": \"Bundle Size\",\n \"react-doctor/prefer-dynamic-import\": \"Bundle Size\",\n \"react-doctor/use-lazy-motion\": \"Bundle Size\",\n \"react-doctor/no-undeferred-third-party\": \"Bundle Size\",\n\n \"react-doctor/no-array-index-as-key\": \"Correctness\",\n \"react-doctor/rendering-conditional-render\": \"Correctness\",\n \"react-doctor/no-prevent-default\": \"Correctness\",\n \"react-doctor/nextjs-no-img-element\": \"Next.js\",\n \"react-doctor/nextjs-async-client-component\": \"Next.js\",\n \"react-doctor/nextjs-no-a-element\": \"Next.js\",\n \"react-doctor/nextjs-no-use-search-params-without-suspense\": \"Next.js\",\n \"react-doctor/nextjs-no-client-fetch-for-server-data\": \"Next.js\",\n \"react-doctor/nextjs-missing-metadata\": \"Next.js\",\n \"react-doctor/nextjs-no-client-side-redirect\": \"Next.js\",\n \"react-doctor/nextjs-no-redirect-in-try-catch\": \"Next.js\",\n \"react-doctor/nextjs-image-missing-sizes\": \"Next.js\",\n \"react-doctor/nextjs-no-native-script\": \"Next.js\",\n \"react-doctor/nextjs-inline-script-missing-id\": \"Next.js\",\n \"react-doctor/nextjs-no-font-link\": \"Next.js\",\n \"react-doctor/nextjs-no-css-link\": \"Next.js\",\n \"react-doctor/nextjs-no-polyfill-script\": \"Next.js\",\n \"react-doctor/nextjs-no-head-import\": \"Next.js\",\n \"react-doctor/nextjs-no-side-effect-in-get-handler\": \"Security\",\n\n \"react-doctor/server-auth-actions\": \"Server\",\n \"react-doctor/server-after-nonblocking\": \"Server\",\n\n \"react-doctor/client-passive-event-listeners\": \"Performance\",\n\n \"react-doctor/async-parallel\": \"Performance\",\n};\n\nconst RULE_HELP_MAP: Record<string, string> = {\n \"no-derived-state-effect\":\n \"Compute during render: `const derived = computeFrom(dep1, dep2)` — no useEffect needed\",\n \"no-fetch-in-effect\":\n \"Use `useQuery()` from @tanstack/react-query, `useSWR()`, or fetch in a Server Component instead\",\n \"no-cascading-set-state\":\n \"Combine into useReducer: `const [state, dispatch] = useReducer(reducer, initialState)`\",\n \"no-effect-event-handler\":\n \"Move the conditional logic into onClick, onChange, or onSubmit handlers directly\",\n \"no-derived-useState\":\n \"Remove useState and compute the value inline: `const value = transform(propName)`\",\n \"prefer-useReducer\":\n \"Group related state: `const [state, dispatch] = useReducer(reducer, { field1, field2, ... })`\",\n \"rerender-lazy-state-init\":\n \"Wrap in an arrow function so it only runs once: `useState(() => expensiveComputation())`\",\n \"rerender-functional-setstate\":\n \"Use the callback form: `setState(prev => prev + 1)` to always read the latest value\",\n \"rerender-dependencies\":\n \"Extract to a useMemo, useRef, or module-level constant so the reference is stable\",\n\n \"no-generic-handler-names\":\n \"Rename to describe the action: e.g. `handleSubmit` → `saveUserProfile`, `handleClick` → `toggleSidebar`\",\n \"no-giant-component\":\n \"Extract logical sections into focused components: `<UserHeader />`, `<UserActions />`, etc.\",\n \"no-render-in-render\":\n \"Extract to a named component: `const ListItem = ({ item }) => <div>{item.name}</div>`\",\n \"no-nested-component-definition\":\n \"Move to a separate file or to module scope above the parent component\",\n\n \"no-usememo-simple-expression\":\n \"Remove useMemo — property access, math, and ternaries are already cheap without memoization\",\n \"no-layout-property-animation\":\n \"Use `transform: translateX()` or `scale()` instead — they run on the compositor and skip layout/paint\",\n \"rerender-memo-with-default-value\":\n \"Move to module scope: `const EMPTY_ITEMS: Item[] = []` then use as the default value\",\n \"rendering-animate-svg-wrapper\":\n \"Wrap the SVG: `<motion.div animate={...}><svg>...</svg></motion.div>`\",\n \"rendering-usetransition-loading\":\n \"Replace with `const [isPending, startTransition] = useTransition()` — avoids a re-render for the loading state\",\n \"rendering-hydration-no-flicker\":\n \"Use `useSyncExternalStore(subscribe, getSnapshot, getServerSnapshot)` or add `suppressHydrationWarning` to the element\",\n\n \"no-transition-all\":\n 'List specific properties: `transition: \"opacity 200ms, transform 200ms\"` — or in Tailwind use `transition-colors`, `transition-opacity`, or `transition-transform`',\n \"no-global-css-variable-animation\":\n \"Set the variable on the nearest element instead of a parent, or use `@property` with `inherits: false` to prevent cascade. Better yet, use targeted `element.style.transform` updates\",\n \"no-large-animated-blur\":\n \"Keep blur radius under 10px, or apply blur to a smaller element. Large blurs multiply GPU memory usage with layer size\",\n \"no-scale-from-zero\":\n \"Use `initial={{ scale: 0.95, opacity: 0 }}` — elements should deflate like a balloon, not vanish into a point\",\n \"no-permanent-will-change\":\n \"Add will-change on animation start (`onMouseEnter`) and remove on end (`onAnimationEnd`). Permanent promotion wastes GPU memory and can degrade performance\",\n\n \"no-secrets-in-client-code\":\n \"Move to server-side `process.env.SECRET_NAME`. Only `NEXT_PUBLIC_*` vars are safe for the client (and should not contain secrets)\",\n\n \"no-barrel-import\":\n \"Import from the direct path: `import { Button } from './components/Button'` instead of `./components`\",\n \"no-full-lodash-import\":\n \"Import the specific function: `import debounce from 'lodash/debounce'` — saves ~70kb\",\n \"no-moment\":\n \"Replace with `import { format } from 'date-fns'` (tree-shakeable) or `import dayjs from 'dayjs'` (2kb)\",\n \"prefer-dynamic-import\":\n \"Use `const Component = dynamic(() => import('library'), { ssr: false })` from next/dynamic or React.lazy()\",\n \"use-lazy-motion\":\n 'Use `import { LazyMotion, m } from \"framer-motion\"` with `domAnimation` features — saves ~30kb',\n \"no-undeferred-third-party\":\n 'Use `next/script` with `strategy=\"lazyOnload\"` or add the `defer` attribute',\n\n \"no-array-index-as-key\":\n \"Use a stable unique identifier: `key={item.id}` or `key={item.slug}` — index keys break on reorder/filter\",\n \"rendering-conditional-render\":\n \"Change to `{items.length > 0 && <List />}` or use a ternary: `{items.length ? <List /> : null}`\",\n \"no-prevent-default\":\n \"Use `<form action={serverAction}>` (works without JS) or `<button>` instead of `<a>` with preventDefault\",\n\n \"nextjs-no-img-element\":\n \"`import Image from 'next/image'` — provides automatic WebP/AVIF, lazy loading, and responsive srcset\",\n \"nextjs-async-client-component\":\n \"Fetch data in a parent Server Component and pass it as props, or use useQuery/useSWR in the client component\",\n \"nextjs-no-a-element\":\n \"`import Link from 'next/link'` — enables client-side navigation, prefetching, and preserves scroll position\",\n \"nextjs-no-use-search-params-without-suspense\":\n \"Wrap the component using useSearchParams: `<Suspense fallback={<Skeleton />}><SearchComponent /></Suspense>`\",\n \"nextjs-no-client-fetch-for-server-data\":\n \"Remove 'use client' and fetch directly in the Server Component — no API round-trip, secrets stay on server\",\n \"nextjs-missing-metadata\":\n \"Add `export const metadata = { title: '...', description: '...' }` or `export async function generateMetadata()`\",\n \"nextjs-no-client-side-redirect\":\n \"Use `redirect('/path')` from 'next/navigation' in a Server Component, or handle in middleware\",\n \"nextjs-no-redirect-in-try-catch\":\n \"Move the redirect/notFound call outside the try block, or add `unstable_rethrow(error)` in the catch\",\n \"nextjs-image-missing-sizes\":\n 'Add sizes for responsive behavior: `sizes=\"(max-width: 768px) 100vw, 50vw\"` matching your layout breakpoints',\n \"nextjs-no-native-script\":\n '`import Script from \"next/script\"` — use `strategy=\"afterInteractive\"` for analytics or `\"lazyOnload\"` for widgets',\n \"nextjs-inline-script-missing-id\":\n 'Add `id=\"descriptive-name\"` so Next.js can track, deduplicate, and re-execute the script correctly',\n \"nextjs-no-font-link\":\n '`import { Inter } from \"next/font/google\"` — self-hosted, zero layout shift, no render-blocking requests',\n \"nextjs-no-css-link\":\n \"Import CSS directly: `import './styles.css'` or use CSS Modules: `import styles from './Button.module.css'`\",\n \"nextjs-no-polyfill-script\":\n \"Next.js includes polyfills for fetch, Promise, Object.assign, Array.from, and 50+ others automatically\",\n \"nextjs-no-head-import\":\n \"Use the Metadata API instead: `export const metadata = { title: '...' }` or `export async function generateMetadata()`\",\n \"nextjs-no-side-effect-in-get-handler\":\n \"Move the side effect to a POST handler and use a <form> or fetch with method POST — GET requests can be triggered by prefetching and are vulnerable to CSRF\",\n\n \"server-auth-actions\":\n \"Add `const session = await auth()` at the top and throw/redirect if unauthorized before any data access\",\n \"server-after-nonblocking\":\n \"`import { after } from 'next/server'` then wrap: `after(() => analytics.track(...))` — response isn't blocked\",\n\n \"client-passive-event-listeners\":\n \"Add `{ passive: true }` as the third argument: `addEventListener('scroll', handler, { passive: true })`\",\n\n \"async-parallel\":\n \"Use `const [a, b] = await Promise.all([fetchA(), fetchB()])` to run independent operations concurrently\",\n};\n\nconst FILEPATH_WITH_LOCATION_PATTERN = /\\S+\\.\\w+:\\d+:\\d+[\\s\\S]*$/;\n\nconst REACT_COMPILER_MESSAGE = \"React Compiler can't optimize this code\";\n\nconst cleanDiagnosticMessage = (\n message: string,\n help: string,\n plugin: string,\n rule: string,\n): CleanedDiagnostic => {\n if (plugin === \"react-hooks-js\") {\n const rawMessage = message.replace(FILEPATH_WITH_LOCATION_PATTERN, \"\").trim();\n return { message: REACT_COMPILER_MESSAGE, help: rawMessage || help };\n }\n const cleaned = message.replace(FILEPATH_WITH_LOCATION_PATTERN, \"\").trim();\n return { message: cleaned || message, help: help || RULE_HELP_MAP[rule] || \"\" };\n};\n\nconst parseRuleCode = (code: string): { plugin: string; rule: string } => {\n const match = code.match(/^(.+)\\((.+)\\)$/);\n if (!match) return { plugin: \"unknown\", rule: code };\n return { plugin: match[1].replace(/^eslint-plugin-/, \"\"), rule: match[2] };\n};\n\nconst resolveOxlintBinary = (): string => {\n const oxlintMainPath = esmRequire.resolve(\"oxlint\");\n const oxlintPackageDirectory = path.resolve(path.dirname(oxlintMainPath), \"..\");\n return path.join(oxlintPackageDirectory, \"bin\", \"oxlint\");\n};\n\nconst resolvePluginPath = (): string => {\n const currentDirectory = path.dirname(fileURLToPath(import.meta.url));\n const pluginPath = path.join(currentDirectory, \"react-doctor-plugin.js\");\n if (fs.existsSync(pluginPath)) return pluginPath;\n\n const distPluginPath = path.resolve(currentDirectory, \"../../dist/react-doctor-plugin.js\");\n if (fs.existsSync(distPluginPath)) return distPluginPath;\n\n return pluginPath;\n};\n\nconst resolveDiagnosticCategory = (plugin: string, rule: string): string => {\n const ruleKey = `${plugin}/${rule}`;\n return RULE_CATEGORY_MAP[ruleKey] ?? PLUGIN_CATEGORY_MAP[plugin] ?? \"Other\";\n};\n\nexport const runOxlint = async (\n rootDirectory: string,\n hasTypeScript: boolean,\n framework: Framework,\n hasReactCompiler: boolean,\n): Promise<Diagnostic[]> => {\n const configPath = path.join(os.tmpdir(), `react-doctor-oxlintrc-${process.pid}.json`);\n const pluginPath = resolvePluginPath();\n const config = createOxlintConfig({ pluginPath, framework, hasReactCompiler });\n\n try {\n fs.writeFileSync(configPath, JSON.stringify(config, null, 2));\n\n const oxlintBinary = resolveOxlintBinary();\n const args = [oxlintBinary, \"-c\", configPath, \"--format\", \"json\"];\n\n if (hasTypeScript) {\n args.push(\"--tsconfig\", \"./tsconfig.json\");\n }\n\n args.push(\".\");\n\n const stdout = await new Promise<string>((resolve, reject) => {\n const child = spawn(process.execPath, args, {\n cwd: rootDirectory,\n });\n\n const stdoutBuffers: Buffer[] = [];\n const stderrBuffers: Buffer[] = [];\n\n child.stdout.on(\"data\", (buffer: Buffer) => stdoutBuffers.push(buffer));\n child.stderr.on(\"data\", (buffer: Buffer) => stderrBuffers.push(buffer));\n\n child.on(\"error\", (error) => reject(new Error(`Failed to run oxlint: ${error.message}`)));\n child.on(\"close\", () => {\n const output = Buffer.concat(stdoutBuffers).toString(\"utf-8\").trim();\n if (!output) {\n const stderrOutput = Buffer.concat(stderrBuffers).toString(\"utf-8\").trim();\n if (stderrOutput) {\n reject(new Error(`Failed to run oxlint: ${stderrOutput}`));\n return;\n }\n }\n resolve(output);\n });\n });\n\n if (!stdout) {\n return [];\n }\n\n let output: OxlintOutput;\n try {\n output = JSON.parse(stdout) as OxlintOutput;\n } catch {\n throw new Error(\n `Failed to parse oxlint output: ${stdout.slice(0, ERROR_PREVIEW_LENGTH_CHARS)}`,\n );\n }\n\n return output.diagnostics\n .filter((diagnostic) => JSX_FILE_PATTERN.test(diagnostic.filename))\n .map((diagnostic) => {\n const { plugin, rule } = parseRuleCode(diagnostic.code);\n const primaryLabel = diagnostic.labels[0];\n\n const cleaned = cleanDiagnosticMessage(diagnostic.message, diagnostic.help, plugin, rule);\n\n return {\n filePath: diagnostic.filename,\n plugin,\n rule,\n severity: diagnostic.severity,\n message: cleaned.message,\n help: cleaned.help,\n line: primaryLabel?.span.line ?? 0,\n column: primaryLabel?.span.column ?? 0,\n category: resolveDiagnosticCategory(plugin, rule),\n };\n });\n } finally {\n if (fs.existsSync(configPath)) {\n fs.unlinkSync(configPath);\n }\n }\n};\n","import ora from \"ora\";\n\nlet sharedInstance: ReturnType<typeof ora> | null = null;\nlet activeCount = 0;\nconst pendingTexts = new Set<string>();\n\nconst finalize = (method: \"succeed\" | \"fail\", originalText: string, displayText: string) => {\n pendingTexts.delete(originalText);\n activeCount--;\n\n if (activeCount <= 0 || !sharedInstance) {\n sharedInstance?.[method](displayText);\n sharedInstance = null;\n activeCount = 0;\n return;\n }\n\n sharedInstance.stop();\n ora(displayText).start()[method](displayText);\n\n const [remainingText] = pendingTexts;\n if (remainingText) {\n sharedInstance.text = remainingText;\n }\n sharedInstance.start();\n};\n\nexport const spinner = (text: string) => ({\n start() {\n activeCount++;\n pendingTexts.add(text);\n\n if (!sharedInstance) {\n sharedInstance = ora({ text }).start();\n } else {\n sharedInstance.text = text;\n }\n\n return {\n succeed: (displayText: string) => finalize(\"succeed\", text, displayText),\n fail: (displayText: string) => finalize(\"fail\", text, displayText),\n };\n },\n});\n","import { randomUUID } from \"node:crypto\";\nimport { mkdirSync, writeFileSync } from \"node:fs\";\nimport { tmpdir } from \"node:os\";\nimport { join } from \"node:path\";\nimport { performance } from \"node:perf_hooks\";\nimport {\n MILLISECONDS_PER_SECOND,\n OFFLINE_MESSAGE,\n PERFECT_SCORE,\n SCORE_BAR_WIDTH_CHARS,\n SCORE_GOOD_THRESHOLD,\n SCORE_OK_THRESHOLD,\n SEPARATOR_LENGTH_CHARS,\n} from \"./constants.js\";\nimport type { Diagnostic, ScanOptions, ScoreResult } from \"./types.js\";\nimport { calculateScore } from \"./utils/calculate-score.js\";\nimport { discoverProject, formatFrameworkName } from \"./utils/discover-project.js\";\nimport { groupBy } from \"./utils/group-by.js\";\nimport { highlighter } from \"./utils/highlighter.js\";\nimport { logger } from \"./utils/logger.js\";\nimport { checkReducedMotion } from \"./utils/check-reduced-motion.js\";\nimport { runKnip } from \"./utils/run-knip.js\";\nimport { runOxlint } from \"./utils/run-oxlint.js\";\nimport { spinner } from \"./utils/spinner.js\";\n\nconst SEVERITY_ORDER: Record<Diagnostic[\"severity\"], number> = {\n error: 0,\n warning: 1,\n};\n\nconst sortBySeverity = (diagnosticGroups: [string, Diagnostic[]][]): [string, Diagnostic[]][] =>\n diagnosticGroups.toSorted(([, diagnosticsA], [, diagnosticsB]) => {\n const severityA = SEVERITY_ORDER[diagnosticsA[0].severity];\n const severityB = SEVERITY_ORDER[diagnosticsB[0].severity];\n return severityA - severityB;\n });\n\nconst collectAffectedFiles = (diagnostics: Diagnostic[]): Set<string> =>\n new Set(diagnostics.map((diagnostic) => diagnostic.filePath));\n\nconst buildFileLineMap = (diagnostics: Diagnostic[]): Map<string, number[]> => {\n const fileLines = new Map<string, number[]>();\n for (const diagnostic of diagnostics) {\n const lines = fileLines.get(diagnostic.filePath) ?? [];\n if (diagnostic.line > 0) {\n lines.push(diagnostic.line);\n }\n fileLines.set(diagnostic.filePath, lines);\n }\n return fileLines;\n};\n\nconst printDiagnostics = (diagnostics: Diagnostic[], isVerbose: boolean): void => {\n const ruleGroups = groupBy(\n diagnostics,\n (diagnostic) => `${diagnostic.plugin}/${diagnostic.rule}`,\n );\n\n const sortedRuleGroups = sortBySeverity([...ruleGroups.entries()]);\n\n for (const [, ruleDiagnostics] of sortedRuleGroups) {\n const firstDiagnostic = ruleDiagnostics[0];\n const icon =\n firstDiagnostic.severity === \"error\" ? highlighter.error(\"✗\") : highlighter.warn(\"⚠\");\n const count = ruleDiagnostics.length;\n const countLabel = count > 1 ? ` (${count})` : \"\";\n\n logger.log(` ${icon} ${firstDiagnostic.message}${countLabel}`);\n if (firstDiagnostic.help) {\n logger.dim(` ${firstDiagnostic.help}`);\n }\n\n if (isVerbose) {\n const fileLines = buildFileLineMap(ruleDiagnostics);\n\n for (const [filePath, lines] of fileLines) {\n const lineLabel = lines.length > 0 ? `: ${lines.join(\", \")}` : \"\";\n logger.dim(` ${filePath}${lineLabel}`);\n }\n }\n\n logger.break();\n }\n};\n\nconst formatElapsedTime = (elapsedMilliseconds: number): string => {\n if (elapsedMilliseconds < MILLISECONDS_PER_SECOND) {\n return `${Math.round(elapsedMilliseconds)}ms`;\n }\n return `${(elapsedMilliseconds / MILLISECONDS_PER_SECOND).toFixed(1)}s`;\n};\n\nconst formatRuleSummary = (ruleKey: string, ruleDiagnostics: Diagnostic[]): string => {\n const firstDiagnostic = ruleDiagnostics[0];\n const fileLines = buildFileLineMap(ruleDiagnostics);\n\n const sections = [\n `Rule: ${ruleKey}`,\n `Severity: ${firstDiagnostic.severity}`,\n `Category: ${firstDiagnostic.category}`,\n `Count: ${ruleDiagnostics.length}`,\n \"\",\n firstDiagnostic.message,\n ];\n\n if (firstDiagnostic.help) {\n sections.push(\"\", `Suggestion: ${firstDiagnostic.help}`);\n }\n\n sections.push(\"\", \"Files:\");\n for (const [filePath, lines] of fileLines) {\n const lineLabel = lines.length > 0 ? `: ${lines.join(\", \")}` : \"\";\n sections.push(` ${filePath}${lineLabel}`);\n }\n\n return sections.join(\"\\n\") + \"\\n\";\n};\n\nconst writeDiagnosticsDirectory = (diagnostics: Diagnostic[]): string => {\n const outputDirectory = join(tmpdir(), `react-doctor-${randomUUID()}`);\n mkdirSync(outputDirectory);\n\n const ruleGroups = groupBy(\n diagnostics,\n (diagnostic) => `${diagnostic.plugin}/${diagnostic.rule}`,\n );\n const sortedRuleGroups = sortBySeverity([...ruleGroups.entries()]);\n\n for (const [ruleKey, ruleDiagnostics] of sortedRuleGroups) {\n const fileName = ruleKey.replace(/\\//g, \"--\") + \".txt\";\n writeFileSync(join(outputDirectory, fileName), formatRuleSummary(ruleKey, ruleDiagnostics));\n }\n\n writeFileSync(join(outputDirectory, \"diagnostics.json\"), JSON.stringify(diagnostics, null, 2));\n\n return outputDirectory;\n};\n\nconst colorizeByScore = (text: string, score: number): string => {\n if (score >= SCORE_GOOD_THRESHOLD) return highlighter.success(text);\n if (score >= SCORE_OK_THRESHOLD) return highlighter.warn(text);\n return highlighter.error(text);\n};\n\nconst buildScoreBar = (score: number): string => {\n const filledCount = Math.round((score / PERFECT_SCORE) * SCORE_BAR_WIDTH_CHARS);\n const emptyCount = SCORE_BAR_WIDTH_CHARS - filledCount;\n const filled = \"█\".repeat(filledCount);\n const empty = \"░\".repeat(emptyCount);\n return colorizeByScore(filled, score) + highlighter.dim(empty);\n};\n\nconst printScoreGauge = (score: number, label: string): void => {\n const scoreDisplay = colorizeByScore(`${score}`, score);\n const labelDisplay = colorizeByScore(label, score);\n logger.log(` ${scoreDisplay} / ${PERFECT_SCORE} ${labelDisplay}`);\n logger.break();\n logger.log(` ${buildScoreBar(score)}`);\n logger.break();\n};\n\nconst getDoctorFace = (score: number): string[] => {\n if (score >= SCORE_GOOD_THRESHOLD) return [\"◠ ◠\", \" ▽ \"];\n if (score >= SCORE_OK_THRESHOLD) return [\"• •\", \" ─ \"];\n return [\"x x\", \" ▽ \"];\n};\n\nconst printBranding = (score?: number): void => {\n if (score !== undefined) {\n const [eyes, mouth] = getDoctorFace(score);\n const colorize = (text: string) => colorizeByScore(text, score);\n logger.log(colorize(\" ┌─────┐\"));\n logger.log(colorize(` │ ${eyes} │`));\n logger.log(colorize(` │ ${mouth} │`));\n logger.log(colorize(\" └─────┘\"));\n }\n logger.log(` React Doctor ${highlighter.dim(\"(www.react.doctor)\")}`);\n logger.break();\n};\n\nconst printSummary = (\n diagnostics: Diagnostic[],\n elapsedMilliseconds: number,\n scoreResult: ScoreResult | null,\n): void => {\n const errorCount = diagnostics.filter((diagnostic) => diagnostic.severity === \"error\").length;\n const warningCount = diagnostics.filter((diagnostic) => diagnostic.severity === \"warning\").length;\n const affectedFileCount = collectAffectedFiles(diagnostics).size;\n const elapsed = formatElapsedTime(elapsedMilliseconds);\n\n logger.log(\"─\".repeat(SEPARATOR_LENGTH_CHARS));\n logger.break();\n\n printBranding(scoreResult?.score);\n\n if (scoreResult) {\n printScoreGauge(scoreResult.score, scoreResult.label);\n } else {\n logger.dim(` ${OFFLINE_MESSAGE}`);\n logger.break();\n }\n\n const parts: string[] = [];\n if (errorCount > 0) {\n parts.push(highlighter.error(`${errorCount} error${errorCount === 1 ? \"\" : \"s\"}`));\n }\n if (warningCount > 0) {\n parts.push(highlighter.warn(`${warningCount} warning${warningCount === 1 ? \"\" : \"s\"}`));\n }\n parts.push(\n highlighter.dim(`across ${affectedFileCount} file${affectedFileCount === 1 ? \"\" : \"s\"}`),\n );\n parts.push(highlighter.dim(`in ${elapsed}`));\n\n logger.log(` ${parts.join(\" \")}`);\n\n try {\n const diagnosticsDirectory = writeDiagnosticsDirectory(diagnostics);\n logger.break();\n logger.dim(` Full diagnostics written to ${diagnosticsDirectory}`);\n } catch {\n logger.break();\n }\n};\n\nexport const scan = async (directory: string, options: ScanOptions): Promise<void> => {\n const startTime = performance.now();\n const projectInfo = discoverProject(directory);\n\n if (!projectInfo.reactVersion) {\n throw new Error(\"No React dependency found in package.json\");\n }\n\n if (!options.scoreOnly) {\n const frameworkLabel = formatFrameworkName(projectInfo.framework);\n const languageLabel = projectInfo.hasTypeScript ? \"TypeScript\" : \"JavaScript\";\n\n const completeStep = (message: string) => {\n spinner(message).start().succeed(message);\n };\n\n completeStep(`Detecting framework. Found ${highlighter.info(frameworkLabel)}.`);\n completeStep(\n `Detecting React version. Found ${highlighter.info(`React ${projectInfo.reactVersion}`)}.`,\n );\n completeStep(`Detecting language. Found ${highlighter.info(languageLabel)}.`);\n completeStep(\n `Detecting React Compiler. ${projectInfo.hasReactCompiler ? highlighter.info(\"Found React Compiler.\") : \"Not found.\"}`,\n );\n completeStep(`Found ${highlighter.info(`${projectInfo.sourceFileCount}`)} source files.`);\n\n logger.break();\n }\n\n const lintPromise = options.lint\n ? (async () => {\n const lintSpinner = options.scoreOnly ? null : spinner(\"Running lint checks...\").start();\n try {\n const lintDiagnostics = await runOxlint(\n directory,\n projectInfo.hasTypeScript,\n projectInfo.framework,\n projectInfo.hasReactCompiler,\n );\n lintSpinner?.succeed(\"Running lint checks.\");\n return lintDiagnostics;\n } catch {\n lintSpinner?.fail(\"Lint checks failed (non-fatal, skipping).\");\n return [];\n }\n })()\n : Promise.resolve<Diagnostic[]>([]);\n\n const deadCodePromise = options.deadCode\n ? (async () => {\n const deadCodeSpinner = options.scoreOnly\n ? null\n : spinner(\"Detecting dead code...\").start();\n try {\n const knipDiagnostics = await runKnip(directory);\n deadCodeSpinner?.succeed(\"Detecting dead code.\");\n return knipDiagnostics;\n } catch {\n deadCodeSpinner?.fail(\"Dead code detection failed (non-fatal, skipping).\");\n return [];\n }\n })()\n : Promise.resolve<Diagnostic[]>([]);\n\n const [lintDiagnostics, deadCodeDiagnostics] = await Promise.all([lintPromise, deadCodePromise]);\n const diagnostics = [\n ...lintDiagnostics,\n ...deadCodeDiagnostics,\n ...checkReducedMotion(directory),\n ];\n\n const elapsedMilliseconds = performance.now() - startTime;\n\n const scoreResult = await calculateScore(diagnostics);\n\n if (options.scoreOnly) {\n if (scoreResult) {\n logger.log(`${scoreResult.score}`);\n } else {\n logger.dim(OFFLINE_MESSAGE);\n }\n return;\n }\n\n if (diagnostics.length === 0) {\n logger.success(\"No issues found!\");\n logger.break();\n if (scoreResult) {\n printBranding(scoreResult.score);\n printScoreGauge(scoreResult.score, scoreResult.label);\n } else {\n logger.dim(` ${OFFLINE_MESSAGE}`);\n }\n return;\n }\n\n printDiagnostics(diagnostics, options.verbose);\n\n printSummary(diagnostics, elapsedMilliseconds, scoreResult);\n};\n","import basePrompts, { type PromptObject, type Answers } from \"prompts\";\nimport { logger } from \"./logger.js\";\n\nconst onCancel = () => {\n logger.break();\n logger.log(\"Cancelled.\");\n logger.break();\n process.exit(0);\n};\n\nexport const prompts = <T extends string = string>(\n questions: PromptObject<T> | PromptObject<T>[],\n): Promise<Answers<T>> => {\n return basePrompts(questions, { onCancel });\n};\n","import path from \"node:path\";\nimport type { WorkspacePackage } from \"../types.js\";\nimport { discoverReactSubprojects, listWorkspacePackages } from \"./discover-project.js\";\nimport { highlighter } from \"./highlighter.js\";\nimport { logger } from \"./logger.js\";\nimport { prompts } from \"./prompts.js\";\n\nexport const selectProjects = async (\n rootDirectory: string,\n projectFlag: string | undefined,\n skipPrompts: boolean,\n): Promise<string[]> => {\n let packages = listWorkspacePackages(rootDirectory);\n if (packages.length === 0) {\n packages = discoverReactSubprojects(rootDirectory);\n }\n\n if (packages.length === 0) return [rootDirectory];\n\n if (projectFlag) return resolveProjectFlag(projectFlag, packages);\n\n if (skipPrompts) {\n printDiscoveredProjects(packages);\n process.exit(0);\n }\n\n return promptProjectSelection(packages, rootDirectory);\n};\n\nconst resolveProjectFlag = (\n projectFlag: string,\n workspacePackages: WorkspacePackage[],\n): string[] => {\n const requestedNames = projectFlag.split(\",\").map((name) => name.trim());\n const resolvedDirectories: string[] = [];\n\n for (const requestedName of requestedNames) {\n const matched = workspacePackages.find(\n (workspacePackage) =>\n workspacePackage.name === requestedName ||\n path.basename(workspacePackage.directory) === requestedName,\n );\n\n if (!matched) {\n const availableNames = workspacePackages\n .map((workspacePackage) => workspacePackage.name)\n .join(\", \");\n throw new Error(`Project \"${requestedName}\" not found. Available: ${availableNames}`);\n }\n\n resolvedDirectories.push(matched.directory);\n }\n\n return resolvedDirectories;\n};\n\nconst printDiscoveredProjects = (packages: WorkspacePackage[]): void => {\n logger.log(\n `${highlighter.success(\"✔\")} Found ${highlighter.info(`${packages.length}`)} React projects:`,\n );\n logger.break();\n\n for (const workspacePackage of packages) {\n logger.log(` ${highlighter.dim(\"─\")} ${workspacePackage.directory}`);\n }\n\n logger.break();\n logger.dim(`Run with a specific path to scan a project:`);\n logger.dim(` npx -y react-doctor@latest <path>`);\n logger.break();\n};\n\nconst promptProjectSelection = async (\n workspacePackages: WorkspacePackage[],\n rootDirectory: string,\n): Promise<string[]> => {\n const { selectedDirectories } = await prompts({\n type: \"multiselect\",\n name: \"selectedDirectories\",\n message: \"Select projects to scan\",\n choices: workspacePackages.map((workspacePackage) => ({\n title: workspacePackage.name,\n description: path.relative(rootDirectory, workspacePackage.directory),\n value: workspacePackage.directory,\n })),\n min: 1,\n });\n\n return selectedDirectories;\n};\n","import { execSync } from \"node:child_process\";\nimport { existsSync, mkdirSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { join } from \"node:path\";\nimport { highlighter } from \"./highlighter.js\";\nimport { logger } from \"./logger.js\";\nimport { prompts } from \"./prompts.js\";\n\nconst CONFIG_DIRECTORY = join(homedir(), \".react-doctor\");\nconst CONFIG_FILE = join(CONFIG_DIRECTORY, \"config.json\");\nconst SKILL_REPO = \"aidenybai/react-doctor\";\n\ninterface UserConfig {\n skillPromptDismissed?: boolean;\n}\n\nconst readConfig = (): UserConfig => {\n try {\n if (!existsSync(CONFIG_FILE)) return {};\n return JSON.parse(readFileSync(CONFIG_FILE, \"utf-8\"));\n } catch {\n return {};\n }\n};\n\nconst writeConfig = (config: UserConfig): void => {\n try {\n if (!existsSync(CONFIG_DIRECTORY)) {\n mkdirSync(CONFIG_DIRECTORY, { recursive: true });\n }\n writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2));\n } catch {}\n};\n\nconst installSkill = (): boolean => {\n try {\n execSync(`npx -y skills add ${SKILL_REPO}`, { stdio: \"inherit\" });\n return true;\n } catch {\n return false;\n }\n};\n\nexport const maybePromptSkillInstall = async (shouldSkipPrompts: boolean): Promise<void> => {\n const config = readConfig();\n if (config.skillPromptDismissed) return;\n if (shouldSkipPrompts) return;\n\n logger.break();\n logger.log(\n `${highlighter.info(\"💡\")} Install the ${highlighter.info(\"react-doctor\")} skill for your coding agent?`,\n );\n logger.dim(\" Adds React best practices to Cursor, Claude Code, Copilot, and more.\");\n logger.break();\n\n const { shouldInstall } = await prompts({\n type: \"confirm\",\n name: \"shouldInstall\",\n message: \"Install skill?\",\n initial: true,\n });\n\n if (shouldInstall) {\n logger.break();\n const didInstall = installSkill();\n if (didInstall) {\n logger.break();\n logger.success(\"Skill installed!\");\n } else {\n logger.break();\n logger.dim(\"Skill install failed. You can install manually:\");\n logger.dim(` npx skills add ${SKILL_REPO}`);\n }\n }\n\n writeConfig({ ...config, skillPromptDismissed: true });\n};\n","import { execSync } from \"node:child_process\";\nimport path from \"node:path\";\nimport { Command } from \"commander\";\nimport type { ScanOptions } from \"./types.js\";\nimport { handleError } from \"./utils/handle-error.js\";\nimport { highlighter } from \"./utils/highlighter.js\";\nimport { logger } from \"./utils/logger.js\";\nimport { scan } from \"./scan.js\";\nimport { selectProjects } from \"./utils/select-projects.js\";\nimport { maybePromptSkillInstall } from \"./utils/skill-prompt.js\";\n\nconst VERSION = process.env.VERSION ?? \"0.0.0\";\n\ninterface CliFlags {\n lint: boolean;\n deadCode: boolean;\n verbose: boolean;\n score: boolean;\n fix: boolean;\n yes: boolean;\n project?: string;\n}\n\nprocess.on(\"SIGINT\", () => process.exit(0));\nprocess.on(\"SIGTERM\", () => process.exit(0));\n\nconst program = new Command()\n .name(\"react-doctor\")\n .description(\"Diagnose React codebase health\")\n .version(VERSION, \"-v, --version\", \"display the version number\")\n .argument(\"[directory]\", \"project directory to scan\", \".\")\n .option(\"--no-lint\", \"skip linting\")\n .option(\"--no-dead-code\", \"skip dead code detection\")\n .option(\"--verbose\", \"show file details per rule\")\n .option(\"--score\", \"output only the score\")\n .option(\"-y, --yes\", \"skip prompts, scan all workspace projects\")\n .option(\"--project <name>\", \"select workspace project (comma-separated for multiple)\")\n .option(\"--fix\", \"open Ami to auto-fix all issues\")\n .action(async (directory: string, flags: CliFlags) => {\n try {\n const resolvedDirectory = path.resolve(directory);\n const isScoreOnly = flags.score;\n\n if (!isScoreOnly) {\n logger.log(`react-doctor v${VERSION}`);\n logger.break();\n }\n\n const scanOptions: ScanOptions = {\n lint: flags.lint,\n deadCode: flags.deadCode,\n verbose: Boolean(flags.verbose),\n scoreOnly: isScoreOnly,\n };\n\n const shouldSkipPrompts =\n flags.yes ||\n Boolean(process.env.CI) ||\n Boolean(process.env.CLAUDECODE) ||\n Boolean(process.env.AMI) ||\n !process.stdin.isTTY;\n const projectDirectories = await selectProjects(\n resolvedDirectory,\n flags.project,\n shouldSkipPrompts,\n );\n\n for (const projectDirectory of projectDirectories) {\n if (!isScoreOnly) {\n logger.dim(`Scanning ${projectDirectory}...`);\n logger.break();\n }\n await scan(projectDirectory, scanOptions);\n if (!isScoreOnly) {\n logger.break();\n }\n }\n\n if (flags.fix) {\n openAmiToFix(resolvedDirectory);\n }\n\n if (!isScoreOnly) {\n await maybePromptSkillInstall(shouldSkipPrompts);\n }\n } catch (error) {\n handleError(error);\n }\n })\n .addHelpText(\n \"after\",\n `\n${highlighter.dim(\"Learn more:\")}\n ${highlighter.info(\"https://github.com/aidenybai/react-doctor\")}\n`,\n );\n\nconst AMI_INSTALL_URL = \"https://ami.dev/install.sh\";\nconst AMI_FIX_PROMPT =\n \"Run npx -y react-doctor@latest . --verbose, read every diagnostic, then fix all issues one by one. After fixing, re-run react-doctor to verify the score improved.\";\nconst OPEN_PROJECT_DELAY_S = 2;\n\nconst isAmiInstalled = (): boolean => {\n try {\n execSync(\"ls /Applications/Ami.app\", { stdio: \"ignore\" });\n return true;\n } catch {\n return false;\n }\n};\n\nconst installAmi = (): void => {\n logger.log(\"Ami not found. Installing...\");\n logger.break();\n try {\n execSync(`curl -fsSL ${AMI_INSTALL_URL} | bash`, { stdio: \"inherit\" });\n } catch {\n logger.error(\"Failed to install Ami. Visit https://ami.dev to install manually.\");\n process.exit(1);\n }\n logger.break();\n};\n\nconst openAmiToFix = (directory: string): void => {\n const resolvedDirectory = path.resolve(directory);\n\n if (!isAmiInstalled()) {\n installAmi();\n }\n\n logger.log(\"Opening Ami to fix react-doctor issues...\");\n\n const encodedDirectory = encodeURIComponent(resolvedDirectory);\n const encodedPrompt = encodeURIComponent(AMI_FIX_PROMPT);\n const openProjectDeeplink = `ami://open-project?cwd=${encodedDirectory}`;\n const newChatDeeplink = `ami://new-chat?prompt=${encodedPrompt}&mode=agent&send=true`;\n\n try {\n execSync(\n `open \"${openProjectDeeplink}\" && sleep ${OPEN_PROJECT_DELAY_S} && open \"${newChatDeeplink}\"`,\n { stdio: \"ignore\" },\n );\n logger.success(\"Opened Ami with react-doctor fix prompt.\");\n } catch {\n logger.break();\n logger.dim(\"Could not open Ami automatically. Open these URLs manually:\");\n logger.info(openProjectDeeplink);\n logger.info(newChatDeeplink);\n }\n};\n\nconst fixAction = (directory: string) => {\n try {\n openAmiToFix(directory);\n } catch (error) {\n handleError(error);\n }\n};\n\nconst fixCommand = new Command(\"fix\")\n .description(\"Open Ami to auto-fix react-doctor issues\")\n .argument(\"[directory]\", \"project directory\", \".\")\n .action(fixAction);\n\nconst installAmiCommand = new Command(\"install-ami\")\n .description(\"Open Ami to auto-fix react-doctor issues\")\n .argument(\"[directory]\", \"project directory\", \".\")\n .action(fixAction);\n\nprogram.addCommand(fixCommand);\nprogram.addCommand(installAmiCommand);\n\nconst main = async () => {\n await program.parseAsync();\n};\n\nmain();\n"],"mappings":"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"}
1
+ {"version":3,"file":"cli.js","names":["esmRequire","main"],"sources":["../src/utils/highlighter.ts","../src/utils/logger.ts","../src/utils/handle-error.ts","../src/constants.ts","../src/utils/calculate-score.ts","../src/utils/read-package-json.ts","../src/utils/discover-project.ts","../src/utils/group-by.ts","../src/plugin/constants.ts","../src/utils/check-reduced-motion.ts","../src/utils/run-knip.ts","../src/oxlint-config.ts","../src/utils/run-oxlint.ts","../src/utils/spinner.ts","../src/scan.ts","../src/utils/prompts.ts","../src/utils/select-projects.ts","../src/utils/skill-prompt.ts","../src/cli.ts"],"sourcesContent":["import pc from \"picocolors\";\n\nexport const highlighter = {\n error: pc.red,\n warn: pc.yellow,\n info: pc.cyan,\n success: pc.green,\n dim: pc.dim,\n};\n","import { highlighter } from \"./highlighter.js\";\n\nexport const logger = {\n error(...args: unknown[]) {\n console.log(highlighter.error(args.join(\" \")));\n },\n warn(...args: unknown[]) {\n console.log(highlighter.warn(args.join(\" \")));\n },\n info(...args: unknown[]) {\n console.log(highlighter.info(args.join(\" \")));\n },\n success(...args: unknown[]) {\n console.log(highlighter.success(args.join(\" \")));\n },\n dim(...args: unknown[]) {\n console.log(highlighter.dim(args.join(\" \")));\n },\n log(...args: unknown[]) {\n console.log(args.join(\" \"));\n },\n break() {\n console.log(\"\");\n },\n};\n","import { logger } from \"./logger.js\";\n\nexport const handleError = (error: unknown): void => {\n logger.break();\n logger.error(\"Something went wrong. Please check the error below for more details.\");\n logger.error(\"If the problem persists, please open an issue on GitHub.\");\n logger.error(\"\");\n if (error instanceof Error) {\n logger.error(error.message);\n }\n logger.break();\n process.exit(1);\n};\n","export const SOURCE_FILE_PATTERN = /\\.(tsx?|jsx?)$/;\n\nexport const JSX_FILE_PATTERN = /\\.(tsx|jsx)$/;\n\nexport const MILLISECONDS_PER_SECOND = 1000;\n\nexport const SEPARATOR_LENGTH_CHARS = 62;\n\nexport const ERROR_PREVIEW_LENGTH_CHARS = 200;\n\nexport const PERFECT_SCORE = 100;\n\nexport const SCORE_GOOD_THRESHOLD = 75;\n\nexport const SCORE_OK_THRESHOLD = 50;\n\nexport const SCORE_BAR_WIDTH_CHARS = 50;\n\nexport const SCORE_API_URL = \"https://www.react.doctor/api/score\";\n\nexport const SHARE_BASE_URL = \"https://www.react.doctor/share\";\n\nexport const OFFLINE_MESSAGE =\n \"You are offline, could not calculate score. Reconnect to calculate.\";\n","import { SCORE_API_URL } from \"../constants.js\";\nimport type { Diagnostic, ScoreResult } from \"../types.js\";\n\nexport const calculateScore = async (diagnostics: Diagnostic[]): Promise<ScoreResult | null> => {\n const payload = diagnostics.map((diagnostic) => ({\n plugin: diagnostic.plugin,\n rule: diagnostic.rule,\n severity: diagnostic.severity,\n }));\n\n try {\n const response = await fetch(SCORE_API_URL, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ diagnostics: payload }),\n });\n\n if (!response.ok) return null;\n\n return (await response.json()) as ScoreResult;\n } catch {\n return null;\n }\n};\n","import fs from \"node:fs\";\nimport type { PackageJson } from \"../types.js\";\n\nexport const readPackageJson = (packageJsonPath: string): PackageJson =>\n JSON.parse(fs.readFileSync(packageJsonPath, \"utf-8\"));\n","import fs from \"node:fs\";\nimport path from \"node:path\";\nimport { spawnSync } from \"node:child_process\";\nimport { SOURCE_FILE_PATTERN } from \"../constants.js\";\nimport type {\n DependencyInfo,\n Framework,\n PackageJson,\n ProjectInfo,\n WorkspacePackage,\n} from \"../types.js\";\nimport { readPackageJson } from \"./read-package-json.js\";\n\nconst REACT_COMPILER_PACKAGES = new Set([\n \"babel-plugin-react-compiler\",\n \"react-compiler-runtime\",\n \"eslint-plugin-react-compiler\",\n]);\n\nconst NEXT_CONFIG_FILENAMES = [\n \"next.config.js\",\n \"next.config.mjs\",\n \"next.config.ts\",\n \"next.config.cjs\",\n];\n\nconst BABEL_CONFIG_FILENAMES = [\n \".babelrc\",\n \".babelrc.json\",\n \"babel.config.js\",\n \"babel.config.json\",\n \"babel.config.cjs\",\n \"babel.config.mjs\",\n];\n\nconst VITE_CONFIG_FILENAMES = [\n \"vite.config.js\",\n \"vite.config.ts\",\n \"vite.config.mjs\",\n \"vite.config.cjs\",\n];\n\nconst REACT_COMPILER_CONFIG_PATTERN = /react-compiler|reactCompiler/;\n\nconst FRAMEWORK_PACKAGES: Record<string, Framework> = {\n next: \"nextjs\",\n vite: \"vite\",\n \"react-scripts\": \"cra\",\n \"@remix-run/react\": \"remix\",\n gatsby: \"gatsby\",\n};\n\nconst FRAMEWORK_DISPLAY_NAMES: Record<Framework, string> = {\n nextjs: \"Next.js\",\n vite: \"Vite\",\n cra: \"Create React App\",\n remix: \"Remix\",\n gatsby: \"Gatsby\",\n unknown: \"React\",\n};\n\nexport const formatFrameworkName = (framework: Framework): string =>\n FRAMEWORK_DISPLAY_NAMES[framework];\n\nconst countSourceFiles = (rootDirectory: string): number => {\n const result = spawnSync(\"git\", [\"ls-files\", \"--cached\", \"--others\", \"--exclude-standard\"], {\n cwd: rootDirectory,\n encoding: \"utf-8\",\n });\n\n if (result.error || result.status !== 0) {\n return 0;\n }\n\n return result.stdout\n .split(\"\\n\")\n .filter((filePath) => filePath.length > 0 && SOURCE_FILE_PATTERN.test(filePath)).length;\n};\n\nconst collectAllDependencies = (packageJson: PackageJson): Record<string, string> => ({\n ...packageJson.dependencies,\n ...packageJson.devDependencies,\n});\n\nconst detectFramework = (dependencies: Record<string, string>): Framework => {\n for (const [packageName, frameworkName] of Object.entries(FRAMEWORK_PACKAGES)) {\n if (dependencies[packageName]) {\n return frameworkName;\n }\n }\n return \"unknown\";\n};\n\nconst extractDependencyInfo = (packageJson: PackageJson): DependencyInfo => {\n const allDependencies = collectAllDependencies(packageJson);\n return {\n reactVersion: allDependencies.react ?? null,\n framework: detectFramework(allDependencies),\n };\n};\n\nconst parsePnpmWorkspacePatterns = (rootDirectory: string): string[] => {\n const workspacePath = path.join(rootDirectory, \"pnpm-workspace.yaml\");\n if (!fs.existsSync(workspacePath)) return [];\n\n const content = fs.readFileSync(workspacePath, \"utf-8\");\n const patterns: string[] = [];\n let isInsidePackagesBlock = false;\n\n for (const line of content.split(\"\\n\")) {\n const trimmed = line.trim();\n if (trimmed === \"packages:\") {\n isInsidePackagesBlock = true;\n continue;\n }\n if (isInsidePackagesBlock && trimmed.startsWith(\"-\")) {\n patterns.push(trimmed.replace(/^-\\s*/, \"\").replace(/[\"']/g, \"\"));\n } else if (isInsidePackagesBlock && trimmed.length > 0 && !trimmed.startsWith(\"#\")) {\n isInsidePackagesBlock = false;\n }\n }\n\n return patterns;\n};\n\nconst getWorkspacePatterns = (rootDirectory: string, packageJson: PackageJson): string[] => {\n const pnpmPatterns = parsePnpmWorkspacePatterns(rootDirectory);\n if (pnpmPatterns.length > 0) return pnpmPatterns;\n\n if (Array.isArray(packageJson.workspaces)) {\n return packageJson.workspaces;\n }\n\n if (packageJson.workspaces?.packages) {\n return packageJson.workspaces.packages;\n }\n\n return [];\n};\n\nconst resolveWorkspaceDirectories = (rootDirectory: string, pattern: string): string[] => {\n const cleanPattern = pattern.replace(/[\"']/g, \"\").replace(/\\/\\*\\*$/, \"/*\");\n\n if (!cleanPattern.includes(\"*\")) {\n const directoryPath = path.join(rootDirectory, cleanPattern);\n if (fs.existsSync(directoryPath) && fs.existsSync(path.join(directoryPath, \"package.json\"))) {\n return [directoryPath];\n }\n return [];\n }\n\n const baseDirectory = path.join(rootDirectory, cleanPattern.slice(0, cleanPattern.indexOf(\"*\")));\n\n if (!fs.existsSync(baseDirectory) || !fs.statSync(baseDirectory).isDirectory()) {\n return [];\n }\n\n return fs\n .readdirSync(baseDirectory)\n .map((entry) => path.join(baseDirectory, entry))\n .filter(\n (entryPath) =>\n fs.statSync(entryPath).isDirectory() && fs.existsSync(path.join(entryPath, \"package.json\")),\n );\n};\n\nconst findDependencyInfoFromAncestors = (startDirectory: string): DependencyInfo => {\n let currentDirectory = path.dirname(startDirectory);\n const result: DependencyInfo = { reactVersion: null, framework: \"unknown\" };\n\n while (currentDirectory !== path.dirname(currentDirectory)) {\n const packageJsonPath = path.join(currentDirectory, \"package.json\");\n if (fs.existsSync(packageJsonPath)) {\n const packageJson = readPackageJson(packageJsonPath);\n const info = extractDependencyInfo(packageJson);\n\n if (!result.reactVersion && info.reactVersion) {\n result.reactVersion = info.reactVersion;\n }\n if (result.framework === \"unknown\" && info.framework !== \"unknown\") {\n result.framework = info.framework;\n }\n\n if (result.reactVersion && result.framework !== \"unknown\") {\n return result;\n }\n }\n\n currentDirectory = path.dirname(currentDirectory);\n }\n\n return result;\n};\n\nconst findReactInWorkspaces = (rootDirectory: string, packageJson: PackageJson): DependencyInfo => {\n const patterns = getWorkspacePatterns(rootDirectory, packageJson);\n const result: DependencyInfo = { reactVersion: null, framework: \"unknown\" };\n\n for (const pattern of patterns) {\n const directories = resolveWorkspaceDirectories(rootDirectory, pattern);\n\n for (const workspaceDirectory of directories) {\n const workspacePackageJson = readPackageJson(path.join(workspaceDirectory, \"package.json\"));\n const info = extractDependencyInfo(workspacePackageJson);\n\n if (info.reactVersion && !result.reactVersion) {\n result.reactVersion = info.reactVersion;\n }\n if (info.framework !== \"unknown\" && result.framework === \"unknown\") {\n result.framework = info.framework;\n }\n\n if (result.reactVersion && result.framework !== \"unknown\") {\n return result;\n }\n }\n }\n\n return result;\n};\n\nconst hasReactDependency = (packageJson: PackageJson): boolean => {\n const allDependencies = collectAllDependencies(packageJson);\n return Object.keys(allDependencies).some(\n (packageName) => packageName === \"next\" || packageName.includes(\"react\"),\n );\n};\n\nexport const discoverReactSubprojects = (rootDirectory: string): WorkspacePackage[] => {\n if (!fs.existsSync(rootDirectory) || !fs.statSync(rootDirectory).isDirectory()) return [];\n\n const entries = fs.readdirSync(rootDirectory, { withFileTypes: true });\n const packages: WorkspacePackage[] = [];\n\n for (const entry of entries) {\n if (!entry.isDirectory() || entry.name.startsWith(\".\") || entry.name === \"node_modules\") {\n continue;\n }\n\n const subdirectory = path.join(rootDirectory, entry.name);\n const packageJsonPath = path.join(subdirectory, \"package.json\");\n if (!fs.existsSync(packageJsonPath)) continue;\n\n const packageJson = readPackageJson(packageJsonPath);\n if (!hasReactDependency(packageJson)) continue;\n\n const name = packageJson.name ?? entry.name;\n packages.push({ name, directory: subdirectory });\n }\n\n return packages;\n};\n\nexport const listWorkspacePackages = (rootDirectory: string): WorkspacePackage[] => {\n const packageJsonPath = path.join(rootDirectory, \"package.json\");\n if (!fs.existsSync(packageJsonPath)) return [];\n\n const packageJson = readPackageJson(packageJsonPath);\n const patterns = getWorkspacePatterns(rootDirectory, packageJson);\n if (patterns.length === 0) return [];\n\n const packages: WorkspacePackage[] = [];\n\n for (const pattern of patterns) {\n const directories = resolveWorkspaceDirectories(rootDirectory, pattern);\n for (const workspaceDirectory of directories) {\n const workspacePackageJson = readPackageJson(path.join(workspaceDirectory, \"package.json\"));\n\n if (!hasReactDependency(workspacePackageJson)) continue;\n\n const name = workspacePackageJson.name ?? path.basename(workspaceDirectory);\n packages.push({ name, directory: workspaceDirectory });\n }\n }\n\n return packages;\n};\n\nconst hasCompilerPackage = (packageJson: PackageJson): boolean => {\n const allDependencies = collectAllDependencies(packageJson);\n return Object.keys(allDependencies).some((packageName) =>\n REACT_COMPILER_PACKAGES.has(packageName),\n );\n};\n\nconst fileContainsPattern = (filePath: string, pattern: RegExp): boolean => {\n if (!fs.existsSync(filePath)) return false;\n const content = fs.readFileSync(filePath, \"utf-8\");\n return pattern.test(content);\n};\n\nconst hasCompilerInConfigFiles = (directory: string, filenames: string[]): boolean =>\n filenames.some((filename) =>\n fileContainsPattern(path.join(directory, filename), REACT_COMPILER_CONFIG_PATTERN),\n );\n\nconst detectReactCompiler = (directory: string, packageJson: PackageJson): boolean => {\n if (hasCompilerPackage(packageJson)) return true;\n\n if (hasCompilerInConfigFiles(directory, NEXT_CONFIG_FILENAMES)) return true;\n if (hasCompilerInConfigFiles(directory, BABEL_CONFIG_FILENAMES)) return true;\n if (hasCompilerInConfigFiles(directory, VITE_CONFIG_FILENAMES)) return true;\n\n let ancestorDirectory = path.dirname(directory);\n while (ancestorDirectory !== path.dirname(ancestorDirectory)) {\n const ancestorPackagePath = path.join(ancestorDirectory, \"package.json\");\n if (fs.existsSync(ancestorPackagePath)) {\n const ancestorPackageJson = readPackageJson(ancestorPackagePath);\n if (hasCompilerPackage(ancestorPackageJson)) return true;\n }\n ancestorDirectory = path.dirname(ancestorDirectory);\n }\n\n return false;\n};\n\nexport const discoverProject = (directory: string): ProjectInfo => {\n const packageJsonPath = path.join(directory, \"package.json\");\n if (!fs.existsSync(packageJsonPath)) {\n throw new Error(`No package.json found in ${directory}`);\n }\n\n const packageJson = readPackageJson(packageJsonPath);\n let { reactVersion, framework } = extractDependencyInfo(packageJson);\n\n if (!reactVersion || framework === \"unknown\") {\n const workspaceInfo = findReactInWorkspaces(directory, packageJson);\n if (!reactVersion && workspaceInfo.reactVersion) {\n reactVersion = workspaceInfo.reactVersion;\n }\n if (framework === \"unknown\" && workspaceInfo.framework !== \"unknown\") {\n framework = workspaceInfo.framework;\n }\n }\n\n if (!reactVersion || framework === \"unknown\") {\n const ancestorInfo = findDependencyInfoFromAncestors(directory);\n if (!reactVersion) {\n reactVersion = ancestorInfo.reactVersion;\n }\n if (framework === \"unknown\") {\n framework = ancestorInfo.framework;\n }\n }\n\n const hasTypeScript = fs.existsSync(path.join(directory, \"tsconfig.json\"));\n const sourceFileCount = countSourceFiles(directory);\n\n const hasReactCompiler = detectReactCompiler(directory, packageJson);\n\n return {\n rootDirectory: directory,\n reactVersion,\n framework,\n hasTypeScript,\n hasReactCompiler,\n sourceFileCount,\n };\n};\n","export const groupBy = <T>(items: T[], keyFn: (item: T) => string): Map<string, T[]> => {\n const groups = new Map<string, T[]>();\n\n for (const item of items) {\n const key = keyFn(item);\n const existing = groups.get(key) ?? [];\n existing.push(item);\n groups.set(key, existing);\n }\n\n return groups;\n};\n","export const GIANT_COMPONENT_LINE_THRESHOLD = 200;\nexport const CASCADING_SET_STATE_THRESHOLD = 2;\nexport const RELATED_USE_STATE_THRESHOLD = 3;\nexport const DEEP_NESTING_THRESHOLD = 3;\nexport const DUPLICATE_STORAGE_READ_THRESHOLD = 2;\nexport const SEQUENTIAL_AWAIT_THRESHOLD = 2;\nexport const SECRET_MIN_LENGTH_CHARS = 8;\nexport const AUTH_CHECK_LOOKAHEAD_STATEMENTS = 3;\n\nexport const LAYOUT_PROPERTIES = new Set([\n \"width\",\n \"height\",\n \"top\",\n \"left\",\n \"right\",\n \"bottom\",\n \"padding\",\n \"paddingTop\",\n \"paddingRight\",\n \"paddingBottom\",\n \"paddingLeft\",\n \"margin\",\n \"marginTop\",\n \"marginRight\",\n \"marginBottom\",\n \"marginLeft\",\n \"borderWidth\",\n \"fontSize\",\n \"lineHeight\",\n \"gap\",\n]);\n\nexport const MOTION_ANIMATE_PROPS = new Set([\n \"animate\",\n \"initial\",\n \"exit\",\n \"whileHover\",\n \"whileTap\",\n \"whileFocus\",\n \"whileDrag\",\n \"whileInView\",\n]);\n\nexport const HEAVY_LIBRARIES = new Set([\n \"@monaco-editor/react\",\n \"monaco-editor\",\n \"recharts\",\n \"@react-pdf/renderer\",\n \"react-quill\",\n \"@codemirror/view\",\n \"@codemirror/state\",\n \"chart.js\",\n \"react-chartjs-2\",\n \"@toast-ui/editor\",\n \"draft-js\",\n]);\n\nexport const FETCH_CALLEE_NAMES = new Set([\"fetch\"]);\nexport const FETCH_MEMBER_OBJECTS = new Set([\"axios\", \"ky\", \"got\"]);\nexport const INDEX_PARAMETER_NAMES = new Set([\"index\", \"idx\", \"i\"]);\nexport const BARREL_INDEX_SUFFIXES = [\n \"/index\",\n \"/index.js\",\n \"/index.ts\",\n \"/index.tsx\",\n \"/index.mjs\",\n];\nexport const PASSIVE_EVENT_NAMES = new Set([\n \"scroll\",\n \"wheel\",\n \"touchstart\",\n \"touchmove\",\n \"touchend\",\n]);\n\nexport const LOOP_TYPES = [\n \"ForStatement\",\n \"ForInStatement\",\n \"ForOfStatement\",\n \"WhileStatement\",\n \"DoWhileStatement\",\n];\n\nexport const AUTH_FUNCTION_NAMES = new Set([\n \"auth\",\n \"getSession\",\n \"getServerSession\",\n \"getUser\",\n \"requireAuth\",\n \"checkAuth\",\n \"verifyAuth\",\n \"authenticate\",\n \"currentUser\",\n \"getAuth\",\n \"validateSession\",\n]);\n\nexport const SECRET_PATTERNS = [\n /^sk_live_/,\n /^sk_test_/,\n /^AKIA[0-9A-Z]{16}$/,\n /^ghp_[a-zA-Z0-9]{36}$/,\n /^gho_[a-zA-Z0-9]{36}$/,\n /^github_pat_/,\n /^glpat-/,\n /^xox[bporas]-/,\n /^sk-[a-zA-Z0-9]{32,}$/,\n];\n\nexport const SECRET_VARIABLE_PATTERN = /(?:api_?key|secret|token|password|credential|auth)/i;\n\nexport const LOADING_STATE_PATTERN = /(?:loading|isLoading|isPending|isSubmitting|isFetching)/i;\n\nexport const EVENT_PROP_PATTERN = /^on[A-Z]/;\nexport const SETTER_PATTERN = /^set[A-Z]/;\nexport const RENDER_FUNCTION_PATTERN = /^render[A-Z]/;\nexport const UPPERCASE_PATTERN = /^[A-Z]/;\nexport const PAGE_FILE_PATTERN = /\\/page\\.(tsx?|jsx?)$/;\nexport const PAGE_OR_LAYOUT_FILE_PATTERN = /\\/(page|layout)\\.(tsx?|jsx?)$/;\nexport const PAGES_DIRECTORY_PATTERN = /\\/pages\\//;\nexport const SERVER_ACTION_FILE_PATTERN = /actions?\\.(tsx?|jsx?)$/;\nexport const SERVER_ACTION_DIRECTORY_PATTERN = /\\/actions\\//;\n\nexport const NEXTJS_NAVIGATION_FUNCTIONS = new Set([\n \"redirect\",\n \"permanentRedirect\",\n \"notFound\",\n \"forbidden\",\n \"unauthorized\",\n]);\n\nexport const GOOGLE_FONTS_PATTERN = /fonts\\.googleapis\\.com/;\n\nexport const POLYFILL_SCRIPT_PATTERN = /polyfill\\.io|polyfill\\.min\\.js|cdn\\.polyfill/;\n\nexport const APP_DIRECTORY_PATTERN = /\\/app\\//;\n\nexport const ROUTE_HANDLER_FILE_PATTERN = /\\/route\\.(tsx?|jsx?)$/;\n\nexport const MUTATION_METHOD_NAMES = new Set([\n \"create\",\n \"insert\",\n \"insertInto\",\n \"update\",\n \"upsert\",\n \"delete\",\n \"remove\",\n \"destroy\",\n \"set\",\n \"append\",\n]);\n\nexport const MUTATING_HTTP_METHODS = new Set([\"POST\", \"PUT\", \"DELETE\", \"PATCH\"]);\n\nexport const MUTATING_ROUTE_SEGMENTS = new Set([\n \"logout\",\n \"log-out\",\n \"signout\",\n \"sign-out\",\n \"unsubscribe\",\n \"delete\",\n \"remove\",\n \"revoke\",\n \"cancel\",\n \"deactivate\",\n]);\n\nexport const EFFECT_HOOK_NAMES = new Set([\"useEffect\", \"useLayoutEffect\"]);\nexport const HOOKS_WITH_DEPS = new Set([\"useEffect\", \"useLayoutEffect\", \"useMemo\", \"useCallback\"]);\nexport const CHAINABLE_ITERATION_METHODS = new Set([\"map\", \"filter\", \"forEach\", \"flatMap\"]);\nexport const STORAGE_OBJECTS = new Set([\"localStorage\", \"sessionStorage\"]);\n\nexport const LARGE_BLUR_THRESHOLD_PX = 10;\nexport const BLUR_VALUE_PATTERN = /blur\\((\\d+(?:\\.\\d+)?)px\\)/;\nexport const ANIMATION_CALLBACK_NAMES = new Set([\"requestAnimationFrame\", \"setInterval\"]);\nexport const MOTION_LIBRARY_PACKAGES = new Set([\"framer-motion\", \"motion\"]);\n","import { execSync } from \"node:child_process\";\nimport fs from \"node:fs\";\nimport path from \"node:path\";\nimport { MOTION_LIBRARY_PACKAGES } from \"../plugin/constants.js\";\nimport type { Diagnostic } from \"../types.js\";\nimport { readPackageJson } from \"./read-package-json.js\";\n\nconst REDUCED_MOTION_GREP_PATTERN = \"prefers-reduced-motion|useReducedMotion\";\nconst REDUCED_MOTION_FILE_GLOBS = '\"*.ts\" \"*.tsx\" \"*.js\" \"*.jsx\" \"*.css\" \"*.scss\"';\n\nconst MISSING_REDUCED_MOTION_DIAGNOSTIC: Diagnostic = {\n filePath: \"package.json\",\n plugin: \"react-doctor\",\n rule: \"require-reduced-motion\",\n severity: \"error\",\n message:\n \"Project uses a motion library but has no prefers-reduced-motion handling — required for accessibility (WCAG 2.3.3)\",\n help: \"Add `useReducedMotion()` from your animation library, or a `@media (prefers-reduced-motion: reduce)` CSS query\",\n line: 0,\n column: 0,\n category: \"Accessibility\",\n weight: 2,\n};\n\nexport const checkReducedMotion = (rootDirectory: string): Diagnostic[] => {\n const packageJsonPath = path.join(rootDirectory, \"package.json\");\n if (!fs.existsSync(packageJsonPath)) return [];\n\n let hasMotionLibrary = false;\n try {\n const packageJson = readPackageJson(packageJsonPath);\n const allDependencies = { ...packageJson.dependencies, ...packageJson.devDependencies };\n hasMotionLibrary = Object.keys(allDependencies).some((packageName) =>\n MOTION_LIBRARY_PACKAGES.has(packageName),\n );\n } catch {\n return [];\n }\n if (!hasMotionLibrary) return [];\n\n try {\n execSync(`git grep -ql -E \"${REDUCED_MOTION_GREP_PATTERN}\" -- ${REDUCED_MOTION_FILE_GLOBS}`, {\n cwd: rootDirectory,\n stdio: \"pipe\",\n });\n return [];\n } catch {\n return [MISSING_REDUCED_MOTION_DIAGNOSTIC];\n }\n};\n","import path from \"node:path\";\nimport { main } from \"knip\";\nimport { createOptions } from \"knip/session\";\nimport type { Diagnostic, KnipIssueRecords, KnipResults } from \"../types.js\";\n\nconst KNIP_CATEGORY_MAP: Record<string, string> = {\n files: \"Dead Code\",\n exports: \"Dead Code\",\n types: \"Dead Code\",\n duplicates: \"Dead Code\",\n};\n\nconst KNIP_MESSAGE_MAP: Record<string, string> = {\n files: \"Unused file\",\n exports: \"Unused export\",\n types: \"Unused type\",\n duplicates: \"Duplicate export\",\n};\n\nconst KNIP_SEVERITY_MAP: Record<string, \"error\" | \"warning\"> = {\n files: \"warning\",\n exports: \"warning\",\n types: \"warning\",\n duplicates: \"warning\",\n};\n\nconst collectIssueRecords = (\n records: KnipIssueRecords,\n issueType: string,\n rootDirectory: string,\n): Diagnostic[] => {\n const diagnostics: Diagnostic[] = [];\n\n for (const issues of Object.values(records)) {\n for (const issue of Object.values(issues)) {\n diagnostics.push({\n filePath: path.relative(rootDirectory, issue.filePath),\n plugin: \"knip\",\n rule: issueType,\n severity: KNIP_SEVERITY_MAP[issueType] ?? \"warning\",\n message: `${KNIP_MESSAGE_MAP[issueType]}: ${issue.symbol}`,\n help: \"\",\n line: 0,\n column: 0,\n category: KNIP_CATEGORY_MAP[issueType] ?? \"Dead Code\",\n weight: 1,\n });\n }\n }\n\n return diagnostics;\n};\n\n// HACK: knip triggers dotenv which logs to stdout/stderr via console methods\nconst silenced = async <T>(fn: () => Promise<T>): Promise<T> => {\n const originalLog = console.log;\n const originalInfo = console.info;\n const originalWarn = console.warn;\n console.log = () => {};\n console.info = () => {};\n console.warn = () => {};\n try {\n return await fn();\n } finally {\n console.log = originalLog;\n console.info = originalInfo;\n console.warn = originalWarn;\n }\n};\n\nexport const runKnip = async (rootDirectory: string): Promise<Diagnostic[]> => {\n const options = await silenced(() =>\n createOptions({\n cwd: rootDirectory,\n isShowProgress: false,\n }),\n );\n\n const { issues } = (await silenced(() => main(options))) as KnipResults;\n const diagnostics: Diagnostic[] = [];\n\n for (const unusedFile of issues.files) {\n diagnostics.push({\n filePath: path.relative(rootDirectory, unusedFile),\n plugin: \"knip\",\n rule: \"files\",\n severity: KNIP_SEVERITY_MAP[\"files\"],\n message: KNIP_MESSAGE_MAP[\"files\"],\n help: \"This file is not imported by any other file in the project.\",\n line: 0,\n column: 0,\n category: KNIP_CATEGORY_MAP[\"files\"],\n weight: 1,\n });\n }\n\n const recordTypes = [\"exports\", \"types\", \"duplicates\"] as const;\n\n for (const issueType of recordTypes) {\n diagnostics.push(...collectIssueRecords(issues[issueType], issueType, rootDirectory));\n }\n\n return diagnostics;\n};\n","import { createRequire } from \"node:module\";\nimport type { Framework } from \"./types.js\";\n\nconst esmRequire = createRequire(import.meta.url);\n\nconst NEXTJS_RULES: Record<string, string> = {\n \"react-doctor/nextjs-no-img-element\": \"warn\",\n \"react-doctor/nextjs-async-client-component\": \"error\",\n \"react-doctor/nextjs-no-a-element\": \"warn\",\n \"react-doctor/nextjs-no-use-search-params-without-suspense\": \"error\",\n \"react-doctor/nextjs-no-client-fetch-for-server-data\": \"warn\",\n \"react-doctor/nextjs-missing-metadata\": \"warn\",\n \"react-doctor/nextjs-no-client-side-redirect\": \"warn\",\n \"react-doctor/nextjs-no-redirect-in-try-catch\": \"warn\",\n \"react-doctor/nextjs-image-missing-sizes\": \"warn\",\n \"react-doctor/nextjs-no-native-script\": \"warn\",\n \"react-doctor/nextjs-inline-script-missing-id\": \"warn\",\n \"react-doctor/nextjs-no-font-link\": \"warn\",\n \"react-doctor/nextjs-no-css-link\": \"warn\",\n \"react-doctor/nextjs-no-polyfill-script\": \"warn\",\n \"react-doctor/nextjs-no-head-import\": \"error\",\n \"react-doctor/nextjs-no-side-effect-in-get-handler\": \"error\",\n};\n\nconst REACT_COMPILER_RULES: Record<string, string> = {\n \"react-hooks-js/set-state-in-render\": \"error\",\n \"react-hooks-js/immutability\": \"error\",\n \"react-hooks-js/refs\": \"error\",\n \"react-hooks-js/purity\": \"error\",\n \"react-hooks-js/hooks\": \"error\",\n \"react-hooks-js/set-state-in-effect\": \"error\",\n \"react-hooks-js/globals\": \"error\",\n \"react-hooks-js/error-boundaries\": \"error\",\n \"react-hooks-js/preserve-manual-memoization\": \"error\",\n \"react-hooks-js/unsupported-syntax\": \"error\",\n \"react-hooks-js/component-hook-factories\": \"error\",\n \"react-hooks-js/static-components\": \"error\",\n \"react-hooks-js/use-memo\": \"error\",\n \"react-hooks-js/void-use-memo\": \"error\",\n \"react-hooks-js/incompatible-library\": \"error\",\n \"react-hooks-js/todo\": \"error\",\n};\n\nconst REACT_PERF_RULES: Record<string, string> = {\n \"react-perf/jsx-no-new-object-as-prop\": \"warn\",\n \"react-perf/jsx-no-new-array-as-prop\": \"warn\",\n \"react-perf/jsx-no-new-function-as-prop\": \"warn\",\n \"react-perf/jsx-no-jsx-as-prop\": \"warn\",\n};\n\ninterface OxlintConfigOptions {\n pluginPath: string;\n framework: Framework;\n hasReactCompiler: boolean;\n}\n\nexport const createOxlintConfig = ({\n pluginPath,\n framework,\n hasReactCompiler,\n}: OxlintConfigOptions) => ({\n categories: {\n correctness: \"off\",\n suspicious: \"off\",\n pedantic: \"off\",\n perf: \"off\",\n restriction: \"off\",\n style: \"off\",\n nursery: \"off\",\n },\n plugins: [\"react\", \"jsx-a11y\", ...(hasReactCompiler ? [] : [\"react-perf\"])],\n jsPlugins: [\n ...(hasReactCompiler\n ? [{ name: \"react-hooks-js\", specifier: esmRequire.resolve(\"eslint-plugin-react-hooks\") }]\n : []),\n pluginPath,\n ],\n rules: {\n \"react/rules-of-hooks\": \"error\",\n \"react/no-direct-mutation-state\": \"error\",\n \"react/jsx-no-duplicate-props\": \"error\",\n \"react/jsx-key\": \"error\",\n \"react/no-children-prop\": \"warn\",\n \"react/no-danger\": \"warn\",\n \"react/jsx-no-script-url\": \"error\",\n \"react/no-render-return-value\": \"warn\",\n \"react/no-string-refs\": \"warn\",\n \"react/no-unescaped-entities\": \"warn\",\n \"react/no-is-mounted\": \"warn\",\n \"react/require-render-return\": \"error\",\n \"react/no-unknown-property\": \"warn\",\n\n \"jsx-a11y/alt-text\": \"error\",\n \"jsx-a11y/anchor-is-valid\": \"warn\",\n \"jsx-a11y/click-events-have-key-events\": \"warn\",\n \"jsx-a11y/no-static-element-interactions\": \"warn\",\n \"jsx-a11y/no-noninteractive-element-interactions\": \"warn\",\n \"jsx-a11y/role-has-required-aria-props\": \"error\",\n \"jsx-a11y/no-autofocus\": \"warn\",\n \"jsx-a11y/heading-has-content\": \"warn\",\n \"jsx-a11y/html-has-lang\": \"warn\",\n \"jsx-a11y/no-redundant-roles\": \"warn\",\n \"jsx-a11y/scope\": \"warn\",\n \"jsx-a11y/tabindex-no-positive\": \"warn\",\n \"jsx-a11y/label-has-associated-control\": \"warn\",\n \"jsx-a11y/no-distracting-elements\": \"error\",\n \"jsx-a11y/iframe-has-title\": \"warn\",\n\n ...(hasReactCompiler ? REACT_COMPILER_RULES : REACT_PERF_RULES),\n\n \"react-doctor/no-derived-state-effect\": \"error\",\n \"react-doctor/no-fetch-in-effect\": \"error\",\n \"react-doctor/no-cascading-set-state\": \"warn\",\n \"react-doctor/no-effect-event-handler\": \"warn\",\n \"react-doctor/no-derived-useState\": \"warn\",\n \"react-doctor/prefer-useReducer\": \"warn\",\n \"react-doctor/rerender-lazy-state-init\": \"warn\",\n \"react-doctor/rerender-functional-setstate\": \"warn\",\n \"react-doctor/rerender-dependencies\": \"error\",\n\n \"react-doctor/no-generic-handler-names\": \"warn\",\n \"react-doctor/no-giant-component\": \"warn\",\n \"react-doctor/no-render-in-render\": \"warn\",\n \"react-doctor/no-nested-component-definition\": \"error\",\n\n \"react-doctor/no-usememo-simple-expression\": \"warn\",\n \"react-doctor/no-layout-property-animation\": \"error\",\n \"react-doctor/rerender-memo-with-default-value\": \"warn\",\n \"react-doctor/rendering-animate-svg-wrapper\": \"warn\",\n \"react-doctor/rendering-usetransition-loading\": \"warn\",\n \"react-doctor/rendering-hydration-no-flicker\": \"warn\",\n\n \"react-doctor/no-transition-all\": \"warn\",\n \"react-doctor/no-global-css-variable-animation\": \"error\",\n \"react-doctor/no-large-animated-blur\": \"warn\",\n \"react-doctor/no-scale-from-zero\": \"warn\",\n \"react-doctor/no-permanent-will-change\": \"warn\",\n\n \"react-doctor/no-secrets-in-client-code\": \"error\",\n\n \"react-doctor/no-barrel-import\": \"warn\",\n \"react-doctor/no-full-lodash-import\": \"warn\",\n \"react-doctor/no-moment\": \"warn\",\n \"react-doctor/prefer-dynamic-import\": \"warn\",\n \"react-doctor/use-lazy-motion\": \"warn\",\n \"react-doctor/no-undeferred-third-party\": \"warn\",\n\n \"react-doctor/no-array-index-as-key\": \"warn\",\n \"react-doctor/rendering-conditional-render\": \"warn\",\n \"react-doctor/no-prevent-default\": \"warn\",\n\n \"react-doctor/server-auth-actions\": \"error\",\n \"react-doctor/server-after-nonblocking\": \"warn\",\n\n \"react-doctor/client-passive-event-listeners\": \"warn\",\n\n \"react-doctor/async-parallel\": \"warn\",\n ...(framework === \"nextjs\" ? NEXTJS_RULES : {}),\n },\n});\n","import { spawn } from \"node:child_process\";\nimport fs from \"node:fs\";\nimport { createRequire } from \"node:module\";\nimport os from \"node:os\";\nimport path from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\nimport { ERROR_PREVIEW_LENGTH_CHARS, JSX_FILE_PATTERN } from \"../constants.js\";\nimport { createOxlintConfig } from \"../oxlint-config.js\";\nimport type { CleanedDiagnostic, Diagnostic, Framework, OxlintOutput } from \"../types.js\";\n\nconst esmRequire = createRequire(import.meta.url);\n\nconst PLUGIN_CATEGORY_MAP: Record<string, string> = {\n react: \"Correctness\",\n \"react-hooks\": \"Correctness\",\n \"react-hooks-js\": \"React Compiler\",\n \"react-perf\": \"Performance\",\n \"jsx-a11y\": \"Accessibility\",\n};\n\nconst RULE_CATEGORY_MAP: Record<string, string> = {\n \"react-doctor/no-derived-state-effect\": \"State & Effects\",\n \"react-doctor/no-fetch-in-effect\": \"State & Effects\",\n \"react-doctor/no-cascading-set-state\": \"State & Effects\",\n \"react-doctor/no-effect-event-handler\": \"State & Effects\",\n \"react-doctor/no-derived-useState\": \"State & Effects\",\n \"react-doctor/prefer-useReducer\": \"State & Effects\",\n \"react-doctor/rerender-lazy-state-init\": \"Performance\",\n \"react-doctor/rerender-functional-setstate\": \"Performance\",\n \"react-doctor/rerender-dependencies\": \"State & Effects\",\n\n \"react-doctor/no-generic-handler-names\": \"Architecture\",\n \"react-doctor/no-giant-component\": \"Architecture\",\n \"react-doctor/no-render-in-render\": \"Architecture\",\n \"react-doctor/no-nested-component-definition\": \"Correctness\",\n\n \"react-doctor/no-usememo-simple-expression\": \"Performance\",\n \"react-doctor/no-layout-property-animation\": \"Performance\",\n \"react-doctor/rerender-memo-with-default-value\": \"Performance\",\n \"react-doctor/rendering-animate-svg-wrapper\": \"Performance\",\n \"react-doctor/rendering-usetransition-loading\": \"Performance\",\n \"react-doctor/rendering-hydration-no-flicker\": \"Performance\",\n\n \"react-doctor/no-transition-all\": \"Performance\",\n \"react-doctor/no-global-css-variable-animation\": \"Performance\",\n \"react-doctor/no-large-animated-blur\": \"Performance\",\n \"react-doctor/no-scale-from-zero\": \"Performance\",\n \"react-doctor/no-permanent-will-change\": \"Performance\",\n\n \"react-doctor/no-secrets-in-client-code\": \"Security\",\n\n \"react-doctor/no-barrel-import\": \"Bundle Size\",\n \"react-doctor/no-full-lodash-import\": \"Bundle Size\",\n \"react-doctor/no-moment\": \"Bundle Size\",\n \"react-doctor/prefer-dynamic-import\": \"Bundle Size\",\n \"react-doctor/use-lazy-motion\": \"Bundle Size\",\n \"react-doctor/no-undeferred-third-party\": \"Bundle Size\",\n\n \"react-doctor/no-array-index-as-key\": \"Correctness\",\n \"react-doctor/rendering-conditional-render\": \"Correctness\",\n \"react-doctor/no-prevent-default\": \"Correctness\",\n \"react-doctor/nextjs-no-img-element\": \"Next.js\",\n \"react-doctor/nextjs-async-client-component\": \"Next.js\",\n \"react-doctor/nextjs-no-a-element\": \"Next.js\",\n \"react-doctor/nextjs-no-use-search-params-without-suspense\": \"Next.js\",\n \"react-doctor/nextjs-no-client-fetch-for-server-data\": \"Next.js\",\n \"react-doctor/nextjs-missing-metadata\": \"Next.js\",\n \"react-doctor/nextjs-no-client-side-redirect\": \"Next.js\",\n \"react-doctor/nextjs-no-redirect-in-try-catch\": \"Next.js\",\n \"react-doctor/nextjs-image-missing-sizes\": \"Next.js\",\n \"react-doctor/nextjs-no-native-script\": \"Next.js\",\n \"react-doctor/nextjs-inline-script-missing-id\": \"Next.js\",\n \"react-doctor/nextjs-no-font-link\": \"Next.js\",\n \"react-doctor/nextjs-no-css-link\": \"Next.js\",\n \"react-doctor/nextjs-no-polyfill-script\": \"Next.js\",\n \"react-doctor/nextjs-no-head-import\": \"Next.js\",\n \"react-doctor/nextjs-no-side-effect-in-get-handler\": \"Security\",\n\n \"react-doctor/server-auth-actions\": \"Server\",\n \"react-doctor/server-after-nonblocking\": \"Server\",\n\n \"react-doctor/client-passive-event-listeners\": \"Performance\",\n\n \"react-doctor/async-parallel\": \"Performance\",\n};\n\nconst RULE_HELP_MAP: Record<string, string> = {\n \"no-derived-state-effect\":\n \"Compute during render: `const derived = computeFrom(dep1, dep2)` — no useEffect needed\",\n \"no-fetch-in-effect\":\n \"Use `useQuery()` from @tanstack/react-query, `useSWR()`, or fetch in a Server Component instead\",\n \"no-cascading-set-state\":\n \"Combine into useReducer: `const [state, dispatch] = useReducer(reducer, initialState)`\",\n \"no-effect-event-handler\":\n \"Move the conditional logic into onClick, onChange, or onSubmit handlers directly\",\n \"no-derived-useState\":\n \"Remove useState and compute the value inline: `const value = transform(propName)`\",\n \"prefer-useReducer\":\n \"Group related state: `const [state, dispatch] = useReducer(reducer, { field1, field2, ... })`\",\n \"rerender-lazy-state-init\":\n \"Wrap in an arrow function so it only runs once: `useState(() => expensiveComputation())`\",\n \"rerender-functional-setstate\":\n \"Use the callback form: `setState(prev => prev + 1)` to always read the latest value\",\n \"rerender-dependencies\":\n \"Extract to a useMemo, useRef, or module-level constant so the reference is stable\",\n\n \"no-generic-handler-names\":\n \"Rename to describe the action: e.g. `handleSubmit` → `saveUserProfile`, `handleClick` → `toggleSidebar`\",\n \"no-giant-component\":\n \"Extract logical sections into focused components: `<UserHeader />`, `<UserActions />`, etc.\",\n \"no-render-in-render\":\n \"Extract to a named component: `const ListItem = ({ item }) => <div>{item.name}</div>`\",\n \"no-nested-component-definition\":\n \"Move to a separate file or to module scope above the parent component\",\n\n \"no-usememo-simple-expression\":\n \"Remove useMemo — property access, math, and ternaries are already cheap without memoization\",\n \"no-layout-property-animation\":\n \"Use `transform: translateX()` or `scale()` instead — they run on the compositor and skip layout/paint\",\n \"rerender-memo-with-default-value\":\n \"Move to module scope: `const EMPTY_ITEMS: Item[] = []` then use as the default value\",\n \"rendering-animate-svg-wrapper\":\n \"Wrap the SVG: `<motion.div animate={...}><svg>...</svg></motion.div>`\",\n \"rendering-usetransition-loading\":\n \"Replace with `const [isPending, startTransition] = useTransition()` — avoids a re-render for the loading state\",\n \"rendering-hydration-no-flicker\":\n \"Use `useSyncExternalStore(subscribe, getSnapshot, getServerSnapshot)` or add `suppressHydrationWarning` to the element\",\n\n \"no-transition-all\":\n 'List specific properties: `transition: \"opacity 200ms, transform 200ms\"` — or in Tailwind use `transition-colors`, `transition-opacity`, or `transition-transform`',\n \"no-global-css-variable-animation\":\n \"Set the variable on the nearest element instead of a parent, or use `@property` with `inherits: false` to prevent cascade. Better yet, use targeted `element.style.transform` updates\",\n \"no-large-animated-blur\":\n \"Keep blur radius under 10px, or apply blur to a smaller element. Large blurs multiply GPU memory usage with layer size\",\n \"no-scale-from-zero\":\n \"Use `initial={{ scale: 0.95, opacity: 0 }}` — elements should deflate like a balloon, not vanish into a point\",\n \"no-permanent-will-change\":\n \"Add will-change on animation start (`onMouseEnter`) and remove on end (`onAnimationEnd`). Permanent promotion wastes GPU memory and can degrade performance\",\n\n \"no-secrets-in-client-code\":\n \"Move to server-side `process.env.SECRET_NAME`. Only `NEXT_PUBLIC_*` vars are safe for the client (and should not contain secrets)\",\n\n \"no-barrel-import\":\n \"Import from the direct path: `import { Button } from './components/Button'` instead of `./components`\",\n \"no-full-lodash-import\":\n \"Import the specific function: `import debounce from 'lodash/debounce'` — saves ~70kb\",\n \"no-moment\":\n \"Replace with `import { format } from 'date-fns'` (tree-shakeable) or `import dayjs from 'dayjs'` (2kb)\",\n \"prefer-dynamic-import\":\n \"Use `const Component = dynamic(() => import('library'), { ssr: false })` from next/dynamic or React.lazy()\",\n \"use-lazy-motion\":\n 'Use `import { LazyMotion, m } from \"framer-motion\"` with `domAnimation` features — saves ~30kb',\n \"no-undeferred-third-party\":\n 'Use `next/script` with `strategy=\"lazyOnload\"` or add the `defer` attribute',\n\n \"no-array-index-as-key\":\n \"Use a stable unique identifier: `key={item.id}` or `key={item.slug}` — index keys break on reorder/filter\",\n \"rendering-conditional-render\":\n \"Change to `{items.length > 0 && <List />}` or use a ternary: `{items.length ? <List /> : null}`\",\n \"no-prevent-default\":\n \"Use `<form action={serverAction}>` (works without JS) or `<button>` instead of `<a>` with preventDefault\",\n\n \"nextjs-no-img-element\":\n \"`import Image from 'next/image'` — provides automatic WebP/AVIF, lazy loading, and responsive srcset\",\n \"nextjs-async-client-component\":\n \"Fetch data in a parent Server Component and pass it as props, or use useQuery/useSWR in the client component\",\n \"nextjs-no-a-element\":\n \"`import Link from 'next/link'` — enables client-side navigation, prefetching, and preserves scroll position\",\n \"nextjs-no-use-search-params-without-suspense\":\n \"Wrap the component using useSearchParams: `<Suspense fallback={<Skeleton />}><SearchComponent /></Suspense>`\",\n \"nextjs-no-client-fetch-for-server-data\":\n \"Remove 'use client' and fetch directly in the Server Component — no API round-trip, secrets stay on server\",\n \"nextjs-missing-metadata\":\n \"Add `export const metadata = { title: '...', description: '...' }` or `export async function generateMetadata()`\",\n \"nextjs-no-client-side-redirect\":\n \"Use `redirect('/path')` from 'next/navigation' in a Server Component, or handle in middleware\",\n \"nextjs-no-redirect-in-try-catch\":\n \"Move the redirect/notFound call outside the try block, or add `unstable_rethrow(error)` in the catch\",\n \"nextjs-image-missing-sizes\":\n 'Add sizes for responsive behavior: `sizes=\"(max-width: 768px) 100vw, 50vw\"` matching your layout breakpoints',\n \"nextjs-no-native-script\":\n '`import Script from \"next/script\"` — use `strategy=\"afterInteractive\"` for analytics or `\"lazyOnload\"` for widgets',\n \"nextjs-inline-script-missing-id\":\n 'Add `id=\"descriptive-name\"` so Next.js can track, deduplicate, and re-execute the script correctly',\n \"nextjs-no-font-link\":\n '`import { Inter } from \"next/font/google\"` — self-hosted, zero layout shift, no render-blocking requests',\n \"nextjs-no-css-link\":\n \"Import CSS directly: `import './styles.css'` or use CSS Modules: `import styles from './Button.module.css'`\",\n \"nextjs-no-polyfill-script\":\n \"Next.js includes polyfills for fetch, Promise, Object.assign, Array.from, and 50+ others automatically\",\n \"nextjs-no-head-import\":\n \"Use the Metadata API instead: `export const metadata = { title: '...' }` or `export async function generateMetadata()`\",\n \"nextjs-no-side-effect-in-get-handler\":\n \"Move the side effect to a POST handler and use a <form> or fetch with method POST — GET requests can be triggered by prefetching and are vulnerable to CSRF\",\n\n \"server-auth-actions\":\n \"Add `const session = await auth()` at the top and throw/redirect if unauthorized before any data access\",\n \"server-after-nonblocking\":\n \"`import { after } from 'next/server'` then wrap: `after(() => analytics.track(...))` — response isn't blocked\",\n\n \"client-passive-event-listeners\":\n \"Add `{ passive: true }` as the third argument: `addEventListener('scroll', handler, { passive: true })`\",\n\n \"async-parallel\":\n \"Use `const [a, b] = await Promise.all([fetchA(), fetchB()])` to run independent operations concurrently\",\n};\n\nconst FILEPATH_WITH_LOCATION_PATTERN = /\\S+\\.\\w+:\\d+:\\d+[\\s\\S]*$/;\n\nconst REACT_COMPILER_MESSAGE = \"React Compiler can't optimize this code\";\n\nconst cleanDiagnosticMessage = (\n message: string,\n help: string,\n plugin: string,\n rule: string,\n): CleanedDiagnostic => {\n if (plugin === \"react-hooks-js\") {\n const rawMessage = message.replace(FILEPATH_WITH_LOCATION_PATTERN, \"\").trim();\n return { message: REACT_COMPILER_MESSAGE, help: rawMessage || help };\n }\n const cleaned = message.replace(FILEPATH_WITH_LOCATION_PATTERN, \"\").trim();\n return { message: cleaned || message, help: help || RULE_HELP_MAP[rule] || \"\" };\n};\n\nconst parseRuleCode = (code: string): { plugin: string; rule: string } => {\n const match = code.match(/^(.+)\\((.+)\\)$/);\n if (!match) return { plugin: \"unknown\", rule: code };\n return { plugin: match[1].replace(/^eslint-plugin-/, \"\"), rule: match[2] };\n};\n\nconst resolveOxlintBinary = (): string => {\n const oxlintMainPath = esmRequire.resolve(\"oxlint\");\n const oxlintPackageDirectory = path.resolve(path.dirname(oxlintMainPath), \"..\");\n return path.join(oxlintPackageDirectory, \"bin\", \"oxlint\");\n};\n\nconst resolvePluginPath = (): string => {\n const currentDirectory = path.dirname(fileURLToPath(import.meta.url));\n const pluginPath = path.join(currentDirectory, \"react-doctor-plugin.js\");\n if (fs.existsSync(pluginPath)) return pluginPath;\n\n const distPluginPath = path.resolve(currentDirectory, \"../../dist/react-doctor-plugin.js\");\n if (fs.existsSync(distPluginPath)) return distPluginPath;\n\n return pluginPath;\n};\n\nconst resolveDiagnosticCategory = (plugin: string, rule: string): string => {\n const ruleKey = `${plugin}/${rule}`;\n return RULE_CATEGORY_MAP[ruleKey] ?? PLUGIN_CATEGORY_MAP[plugin] ?? \"Other\";\n};\n\nexport const runOxlint = async (\n rootDirectory: string,\n hasTypeScript: boolean,\n framework: Framework,\n hasReactCompiler: boolean,\n): Promise<Diagnostic[]> => {\n const configPath = path.join(os.tmpdir(), `react-doctor-oxlintrc-${process.pid}.json`);\n const pluginPath = resolvePluginPath();\n const config = createOxlintConfig({ pluginPath, framework, hasReactCompiler });\n\n try {\n fs.writeFileSync(configPath, JSON.stringify(config, null, 2));\n\n const oxlintBinary = resolveOxlintBinary();\n const args = [oxlintBinary, \"-c\", configPath, \"--format\", \"json\"];\n\n if (hasTypeScript) {\n args.push(\"--tsconfig\", \"./tsconfig.json\");\n }\n\n args.push(\".\");\n\n const stdout = await new Promise<string>((resolve, reject) => {\n const child = spawn(process.execPath, args, {\n cwd: rootDirectory,\n });\n\n const stdoutBuffers: Buffer[] = [];\n const stderrBuffers: Buffer[] = [];\n\n child.stdout.on(\"data\", (buffer: Buffer) => stdoutBuffers.push(buffer));\n child.stderr.on(\"data\", (buffer: Buffer) => stderrBuffers.push(buffer));\n\n child.on(\"error\", (error) => reject(new Error(`Failed to run oxlint: ${error.message}`)));\n child.on(\"close\", () => {\n const output = Buffer.concat(stdoutBuffers).toString(\"utf-8\").trim();\n if (!output) {\n const stderrOutput = Buffer.concat(stderrBuffers).toString(\"utf-8\").trim();\n if (stderrOutput) {\n reject(new Error(`Failed to run oxlint: ${stderrOutput}`));\n return;\n }\n }\n resolve(output);\n });\n });\n\n if (!stdout) {\n return [];\n }\n\n let output: OxlintOutput;\n try {\n output = JSON.parse(stdout) as OxlintOutput;\n } catch {\n throw new Error(\n `Failed to parse oxlint output: ${stdout.slice(0, ERROR_PREVIEW_LENGTH_CHARS)}`,\n );\n }\n\n return output.diagnostics\n .filter((diagnostic) => JSX_FILE_PATTERN.test(diagnostic.filename))\n .map((diagnostic) => {\n const { plugin, rule } = parseRuleCode(diagnostic.code);\n const primaryLabel = diagnostic.labels[0];\n\n const cleaned = cleanDiagnosticMessage(diagnostic.message, diagnostic.help, plugin, rule);\n\n return {\n filePath: diagnostic.filename,\n plugin,\n rule,\n severity: diagnostic.severity,\n message: cleaned.message,\n help: cleaned.help,\n line: primaryLabel?.span.line ?? 0,\n column: primaryLabel?.span.column ?? 0,\n category: resolveDiagnosticCategory(plugin, rule),\n };\n });\n } finally {\n if (fs.existsSync(configPath)) {\n fs.unlinkSync(configPath);\n }\n }\n};\n","import ora from \"ora\";\n\nlet sharedInstance: ReturnType<typeof ora> | null = null;\nlet activeCount = 0;\nconst pendingTexts = new Set<string>();\n\nconst finalize = (method: \"succeed\" | \"fail\", originalText: string, displayText: string) => {\n pendingTexts.delete(originalText);\n activeCount--;\n\n if (activeCount <= 0 || !sharedInstance) {\n sharedInstance?.[method](displayText);\n sharedInstance = null;\n activeCount = 0;\n return;\n }\n\n sharedInstance.stop();\n ora(displayText).start()[method](displayText);\n\n const [remainingText] = pendingTexts;\n if (remainingText) {\n sharedInstance.text = remainingText;\n }\n sharedInstance.start();\n};\n\nexport const spinner = (text: string) => ({\n start() {\n activeCount++;\n pendingTexts.add(text);\n\n if (!sharedInstance) {\n sharedInstance = ora({ text }).start();\n } else {\n sharedInstance.text = text;\n }\n\n return {\n succeed: (displayText: string) => finalize(\"succeed\", text, displayText),\n fail: (displayText: string) => finalize(\"fail\", text, displayText),\n };\n },\n});\n","import { randomUUID } from \"node:crypto\";\nimport { mkdirSync, writeFileSync } from \"node:fs\";\nimport { tmpdir } from \"node:os\";\nimport { join } from \"node:path\";\nimport { performance } from \"node:perf_hooks\";\nimport {\n MILLISECONDS_PER_SECOND,\n OFFLINE_MESSAGE,\n PERFECT_SCORE,\n SCORE_BAR_WIDTH_CHARS,\n SCORE_GOOD_THRESHOLD,\n SCORE_OK_THRESHOLD,\n SEPARATOR_LENGTH_CHARS,\n SHARE_BASE_URL,\n} from \"./constants.js\";\nimport type { Diagnostic, ScanOptions, ScoreResult } from \"./types.js\";\nimport { calculateScore } from \"./utils/calculate-score.js\";\nimport { discoverProject, formatFrameworkName } from \"./utils/discover-project.js\";\nimport { groupBy } from \"./utils/group-by.js\";\nimport { highlighter } from \"./utils/highlighter.js\";\nimport { logger } from \"./utils/logger.js\";\nimport { checkReducedMotion } from \"./utils/check-reduced-motion.js\";\nimport { runKnip } from \"./utils/run-knip.js\";\nimport { runOxlint } from \"./utils/run-oxlint.js\";\nimport { spinner } from \"./utils/spinner.js\";\n\nconst SEVERITY_ORDER: Record<Diagnostic[\"severity\"], number> = {\n error: 0,\n warning: 1,\n};\n\nconst sortBySeverity = (diagnosticGroups: [string, Diagnostic[]][]): [string, Diagnostic[]][] =>\n diagnosticGroups.toSorted(([, diagnosticsA], [, diagnosticsB]) => {\n const severityA = SEVERITY_ORDER[diagnosticsA[0].severity];\n const severityB = SEVERITY_ORDER[diagnosticsB[0].severity];\n return severityA - severityB;\n });\n\nconst collectAffectedFiles = (diagnostics: Diagnostic[]): Set<string> =>\n new Set(diagnostics.map((diagnostic) => diagnostic.filePath));\n\nconst buildFileLineMap = (diagnostics: Diagnostic[]): Map<string, number[]> => {\n const fileLines = new Map<string, number[]>();\n for (const diagnostic of diagnostics) {\n const lines = fileLines.get(diagnostic.filePath) ?? [];\n if (diagnostic.line > 0) {\n lines.push(diagnostic.line);\n }\n fileLines.set(diagnostic.filePath, lines);\n }\n return fileLines;\n};\n\nconst printDiagnostics = (diagnostics: Diagnostic[], isVerbose: boolean): void => {\n const ruleGroups = groupBy(\n diagnostics,\n (diagnostic) => `${diagnostic.plugin}/${diagnostic.rule}`,\n );\n\n const sortedRuleGroups = sortBySeverity([...ruleGroups.entries()]);\n\n for (const [, ruleDiagnostics] of sortedRuleGroups) {\n const firstDiagnostic = ruleDiagnostics[0];\n const icon =\n firstDiagnostic.severity === \"error\" ? highlighter.error(\"✗\") : highlighter.warn(\"⚠\");\n const count = ruleDiagnostics.length;\n const countLabel = count > 1 ? ` (${count})` : \"\";\n\n logger.log(` ${icon} ${firstDiagnostic.message}${countLabel}`);\n if (firstDiagnostic.help) {\n logger.dim(` ${firstDiagnostic.help}`);\n }\n\n if (isVerbose) {\n const fileLines = buildFileLineMap(ruleDiagnostics);\n\n for (const [filePath, lines] of fileLines) {\n const lineLabel = lines.length > 0 ? `: ${lines.join(\", \")}` : \"\";\n logger.dim(` ${filePath}${lineLabel}`);\n }\n }\n\n logger.break();\n }\n};\n\nconst formatElapsedTime = (elapsedMilliseconds: number): string => {\n if (elapsedMilliseconds < MILLISECONDS_PER_SECOND) {\n return `${Math.round(elapsedMilliseconds)}ms`;\n }\n return `${(elapsedMilliseconds / MILLISECONDS_PER_SECOND).toFixed(1)}s`;\n};\n\nconst formatRuleSummary = (ruleKey: string, ruleDiagnostics: Diagnostic[]): string => {\n const firstDiagnostic = ruleDiagnostics[0];\n const fileLines = buildFileLineMap(ruleDiagnostics);\n\n const sections = [\n `Rule: ${ruleKey}`,\n `Severity: ${firstDiagnostic.severity}`,\n `Category: ${firstDiagnostic.category}`,\n `Count: ${ruleDiagnostics.length}`,\n \"\",\n firstDiagnostic.message,\n ];\n\n if (firstDiagnostic.help) {\n sections.push(\"\", `Suggestion: ${firstDiagnostic.help}`);\n }\n\n sections.push(\"\", \"Files:\");\n for (const [filePath, lines] of fileLines) {\n const lineLabel = lines.length > 0 ? `: ${lines.join(\", \")}` : \"\";\n sections.push(` ${filePath}${lineLabel}`);\n }\n\n return sections.join(\"\\n\") + \"\\n\";\n};\n\nconst writeDiagnosticsDirectory = (diagnostics: Diagnostic[]): string => {\n const outputDirectory = join(tmpdir(), `react-doctor-${randomUUID()}`);\n mkdirSync(outputDirectory);\n\n const ruleGroups = groupBy(\n diagnostics,\n (diagnostic) => `${diagnostic.plugin}/${diagnostic.rule}`,\n );\n const sortedRuleGroups = sortBySeverity([...ruleGroups.entries()]);\n\n for (const [ruleKey, ruleDiagnostics] of sortedRuleGroups) {\n const fileName = ruleKey.replace(/\\//g, \"--\") + \".txt\";\n writeFileSync(join(outputDirectory, fileName), formatRuleSummary(ruleKey, ruleDiagnostics));\n }\n\n writeFileSync(join(outputDirectory, \"diagnostics.json\"), JSON.stringify(diagnostics, null, 2));\n\n return outputDirectory;\n};\n\nconst colorizeByScore = (text: string, score: number): string => {\n if (score >= SCORE_GOOD_THRESHOLD) return highlighter.success(text);\n if (score >= SCORE_OK_THRESHOLD) return highlighter.warn(text);\n return highlighter.error(text);\n};\n\nconst buildScoreBar = (score: number): string => {\n const filledCount = Math.round((score / PERFECT_SCORE) * SCORE_BAR_WIDTH_CHARS);\n const emptyCount = SCORE_BAR_WIDTH_CHARS - filledCount;\n const filled = \"█\".repeat(filledCount);\n const empty = \"░\".repeat(emptyCount);\n return colorizeByScore(filled, score) + highlighter.dim(empty);\n};\n\nconst printScoreGauge = (score: number, label: string): void => {\n const scoreDisplay = colorizeByScore(`${score}`, score);\n const labelDisplay = colorizeByScore(label, score);\n logger.log(` ${scoreDisplay} / ${PERFECT_SCORE} ${labelDisplay}`);\n logger.break();\n logger.log(` ${buildScoreBar(score)}`);\n logger.break();\n};\n\nconst getDoctorFace = (score: number): string[] => {\n if (score >= SCORE_GOOD_THRESHOLD) return [\"◠ ◠\", \" ▽ \"];\n if (score >= SCORE_OK_THRESHOLD) return [\"• •\", \" ─ \"];\n return [\"x x\", \" ▽ \"];\n};\n\nconst printBranding = (score?: number): void => {\n if (score !== undefined) {\n const [eyes, mouth] = getDoctorFace(score);\n const colorize = (text: string) => colorizeByScore(text, score);\n logger.log(colorize(\" ┌─────┐\"));\n logger.log(colorize(` │ ${eyes} │`));\n logger.log(colorize(` │ ${mouth} │`));\n logger.log(colorize(\" └─────┘\"));\n }\n logger.log(` React Doctor ${highlighter.dim(\"(www.react.doctor)\")}`);\n logger.break();\n};\n\nconst buildShareUrl = (diagnostics: Diagnostic[], scoreResult: ScoreResult | null): string => {\n const errorCount = diagnostics.filter((diagnostic) => diagnostic.severity === \"error\").length;\n const warningCount = diagnostics.filter((diagnostic) => diagnostic.severity === \"warning\").length;\n const affectedFileCount = collectAffectedFiles(diagnostics).size;\n\n const params = new URLSearchParams();\n if (scoreResult) params.set(\"s\", String(scoreResult.score));\n if (errorCount > 0) params.set(\"e\", String(errorCount));\n if (warningCount > 0) params.set(\"w\", String(warningCount));\n if (affectedFileCount > 0) params.set(\"f\", String(affectedFileCount));\n\n return `${SHARE_BASE_URL}?${params.toString()}`;\n};\n\nconst printSummary = (\n diagnostics: Diagnostic[],\n elapsedMilliseconds: number,\n scoreResult: ScoreResult | null,\n): void => {\n const errorCount = diagnostics.filter((diagnostic) => diagnostic.severity === \"error\").length;\n const warningCount = diagnostics.filter((diagnostic) => diagnostic.severity === \"warning\").length;\n const affectedFileCount = collectAffectedFiles(diagnostics).size;\n const elapsed = formatElapsedTime(elapsedMilliseconds);\n\n logger.log(\"─\".repeat(SEPARATOR_LENGTH_CHARS));\n logger.break();\n\n printBranding(scoreResult?.score);\n\n if (scoreResult) {\n printScoreGauge(scoreResult.score, scoreResult.label);\n } else {\n logger.dim(` ${OFFLINE_MESSAGE}`);\n logger.break();\n }\n\n const parts: string[] = [];\n if (errorCount > 0) {\n parts.push(highlighter.error(`${errorCount} error${errorCount === 1 ? \"\" : \"s\"}`));\n }\n if (warningCount > 0) {\n parts.push(highlighter.warn(`${warningCount} warning${warningCount === 1 ? \"\" : \"s\"}`));\n }\n parts.push(\n highlighter.dim(`across ${affectedFileCount} file${affectedFileCount === 1 ? \"\" : \"s\"}`),\n );\n parts.push(highlighter.dim(`in ${elapsed}`));\n\n logger.log(` ${parts.join(\" \")}`);\n\n try {\n const diagnosticsDirectory = writeDiagnosticsDirectory(diagnostics);\n logger.break();\n logger.dim(` Full diagnostics written to ${diagnosticsDirectory}`);\n } catch {\n logger.break();\n }\n\n const shareUrl = buildShareUrl(diagnostics, scoreResult);\n logger.break();\n logger.dim(` Share your results: ${highlighter.info(shareUrl)}`);\n};\n\nexport const scan = async (directory: string, options: ScanOptions): Promise<void> => {\n const startTime = performance.now();\n const projectInfo = discoverProject(directory);\n\n if (!projectInfo.reactVersion) {\n throw new Error(\"No React dependency found in package.json\");\n }\n\n if (!options.scoreOnly) {\n const frameworkLabel = formatFrameworkName(projectInfo.framework);\n const languageLabel = projectInfo.hasTypeScript ? \"TypeScript\" : \"JavaScript\";\n\n const completeStep = (message: string) => {\n spinner(message).start().succeed(message);\n };\n\n completeStep(`Detecting framework. Found ${highlighter.info(frameworkLabel)}.`);\n completeStep(\n `Detecting React version. Found ${highlighter.info(`React ${projectInfo.reactVersion}`)}.`,\n );\n completeStep(`Detecting language. Found ${highlighter.info(languageLabel)}.`);\n completeStep(\n `Detecting React Compiler. ${projectInfo.hasReactCompiler ? highlighter.info(\"Found React Compiler.\") : \"Not found.\"}`,\n );\n completeStep(`Found ${highlighter.info(`${projectInfo.sourceFileCount}`)} source files.`);\n\n logger.break();\n }\n\n const lintPromise = options.lint\n ? (async () => {\n const lintSpinner = options.scoreOnly ? null : spinner(\"Running lint checks...\").start();\n try {\n const lintDiagnostics = await runOxlint(\n directory,\n projectInfo.hasTypeScript,\n projectInfo.framework,\n projectInfo.hasReactCompiler,\n );\n lintSpinner?.succeed(\"Running lint checks.\");\n return lintDiagnostics;\n } catch {\n lintSpinner?.fail(\"Lint checks failed (non-fatal, skipping).\");\n return [];\n }\n })()\n : Promise.resolve<Diagnostic[]>([]);\n\n const deadCodePromise = options.deadCode\n ? (async () => {\n const deadCodeSpinner = options.scoreOnly\n ? null\n : spinner(\"Detecting dead code...\").start();\n try {\n const knipDiagnostics = await runKnip(directory);\n deadCodeSpinner?.succeed(\"Detecting dead code.\");\n return knipDiagnostics;\n } catch {\n deadCodeSpinner?.fail(\"Dead code detection failed (non-fatal, skipping).\");\n return [];\n }\n })()\n : Promise.resolve<Diagnostic[]>([]);\n\n const [lintDiagnostics, deadCodeDiagnostics] = await Promise.all([lintPromise, deadCodePromise]);\n const diagnostics = [\n ...lintDiagnostics,\n ...deadCodeDiagnostics,\n ...checkReducedMotion(directory),\n ];\n\n const elapsedMilliseconds = performance.now() - startTime;\n\n const scoreResult = await calculateScore(diagnostics);\n\n if (options.scoreOnly) {\n if (scoreResult) {\n logger.log(`${scoreResult.score}`);\n } else {\n logger.dim(OFFLINE_MESSAGE);\n }\n return;\n }\n\n if (diagnostics.length === 0) {\n logger.success(\"No issues found!\");\n logger.break();\n if (scoreResult) {\n printBranding(scoreResult.score);\n printScoreGauge(scoreResult.score, scoreResult.label);\n } else {\n logger.dim(` ${OFFLINE_MESSAGE}`);\n }\n return;\n }\n\n printDiagnostics(diagnostics, options.verbose);\n\n printSummary(diagnostics, elapsedMilliseconds, scoreResult);\n};\n","import basePrompts, { type PromptObject, type Answers } from \"prompts\";\nimport { logger } from \"./logger.js\";\n\nconst onCancel = () => {\n logger.break();\n logger.log(\"Cancelled.\");\n logger.break();\n process.exit(0);\n};\n\nexport const prompts = <T extends string = string>(\n questions: PromptObject<T> | PromptObject<T>[],\n): Promise<Answers<T>> => {\n return basePrompts(questions, { onCancel });\n};\n","import path from \"node:path\";\nimport type { WorkspacePackage } from \"../types.js\";\nimport { discoverReactSubprojects, listWorkspacePackages } from \"./discover-project.js\";\nimport { highlighter } from \"./highlighter.js\";\nimport { logger } from \"./logger.js\";\nimport { prompts } from \"./prompts.js\";\n\nexport const selectProjects = async (\n rootDirectory: string,\n projectFlag: string | undefined,\n skipPrompts: boolean,\n): Promise<string[]> => {\n let packages = listWorkspacePackages(rootDirectory);\n if (packages.length === 0) {\n packages = discoverReactSubprojects(rootDirectory);\n }\n\n if (packages.length === 0) return [rootDirectory];\n\n if (projectFlag) return resolveProjectFlag(projectFlag, packages);\n\n if (skipPrompts) {\n printDiscoveredProjects(packages);\n process.exit(0);\n }\n\n return promptProjectSelection(packages, rootDirectory);\n};\n\nconst resolveProjectFlag = (\n projectFlag: string,\n workspacePackages: WorkspacePackage[],\n): string[] => {\n const requestedNames = projectFlag.split(\",\").map((name) => name.trim());\n const resolvedDirectories: string[] = [];\n\n for (const requestedName of requestedNames) {\n const matched = workspacePackages.find(\n (workspacePackage) =>\n workspacePackage.name === requestedName ||\n path.basename(workspacePackage.directory) === requestedName,\n );\n\n if (!matched) {\n const availableNames = workspacePackages\n .map((workspacePackage) => workspacePackage.name)\n .join(\", \");\n throw new Error(`Project \"${requestedName}\" not found. Available: ${availableNames}`);\n }\n\n resolvedDirectories.push(matched.directory);\n }\n\n return resolvedDirectories;\n};\n\nconst printDiscoveredProjects = (packages: WorkspacePackage[]): void => {\n logger.log(\n `${highlighter.success(\"✔\")} Found ${highlighter.info(`${packages.length}`)} React projects:`,\n );\n logger.break();\n\n for (const workspacePackage of packages) {\n logger.log(` ${highlighter.dim(\"─\")} ${workspacePackage.directory}`);\n }\n\n logger.break();\n logger.dim(`Run with a specific path to scan a project:`);\n logger.dim(` npx -y react-doctor@latest <path>`);\n logger.break();\n};\n\nconst promptProjectSelection = async (\n workspacePackages: WorkspacePackage[],\n rootDirectory: string,\n): Promise<string[]> => {\n const { selectedDirectories } = await prompts({\n type: \"multiselect\",\n name: \"selectedDirectories\",\n message: \"Select projects to scan\",\n choices: workspacePackages.map((workspacePackage) => ({\n title: workspacePackage.name,\n description: path.relative(rootDirectory, workspacePackage.directory),\n value: workspacePackage.directory,\n })),\n min: 1,\n });\n\n return selectedDirectories;\n};\n","import { execSync } from \"node:child_process\";\nimport { existsSync, mkdirSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { join } from \"node:path\";\nimport { highlighter } from \"./highlighter.js\";\nimport { logger } from \"./logger.js\";\nimport { prompts } from \"./prompts.js\";\n\nconst CONFIG_DIRECTORY = join(homedir(), \".react-doctor\");\nconst CONFIG_FILE = join(CONFIG_DIRECTORY, \"config.json\");\nconst SKILL_REPO = \"aidenybai/react-doctor\";\n\ninterface UserConfig {\n skillPromptDismissed?: boolean;\n}\n\nconst readConfig = (): UserConfig => {\n try {\n if (!existsSync(CONFIG_FILE)) return {};\n return JSON.parse(readFileSync(CONFIG_FILE, \"utf-8\"));\n } catch {\n return {};\n }\n};\n\nconst writeConfig = (config: UserConfig): void => {\n try {\n if (!existsSync(CONFIG_DIRECTORY)) {\n mkdirSync(CONFIG_DIRECTORY, { recursive: true });\n }\n writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2));\n } catch {}\n};\n\nconst installSkill = (): boolean => {\n try {\n execSync(`npx -y skills add ${SKILL_REPO}`, { stdio: \"inherit\" });\n return true;\n } catch {\n return false;\n }\n};\n\nexport const maybePromptSkillInstall = async (shouldSkipPrompts: boolean): Promise<void> => {\n const config = readConfig();\n if (config.skillPromptDismissed) return;\n if (shouldSkipPrompts) return;\n\n logger.break();\n logger.log(`${highlighter.info(\"💡\")} Have your coding agent fix these issues automatically?`);\n logger.dim(\n ` Install the ${highlighter.info(\"react-doctor\")} skill to teach Cursor, Claude Code, Copilot,`,\n );\n logger.dim(\" Ami, and other AI agents how to diagnose and fix these React issues.\");\n logger.break();\n\n const { shouldInstall } = await prompts({\n type: \"confirm\",\n name: \"shouldInstall\",\n message: \"Install skill?\",\n initial: true,\n });\n\n if (shouldInstall) {\n logger.break();\n const didInstall = installSkill();\n if (didInstall) {\n logger.break();\n logger.success(\"Skill installed!\");\n } else {\n logger.break();\n logger.dim(\"Skill install failed. You can install manually:\");\n logger.dim(` npx skills add ${SKILL_REPO}`);\n }\n }\n\n writeConfig({ ...config, skillPromptDismissed: true });\n};\n","import { execSync } from \"node:child_process\";\nimport path from \"node:path\";\nimport { Command } from \"commander\";\nimport type { ScanOptions } from \"./types.js\";\nimport { handleError } from \"./utils/handle-error.js\";\nimport { highlighter } from \"./utils/highlighter.js\";\nimport { logger } from \"./utils/logger.js\";\nimport { scan } from \"./scan.js\";\nimport { selectProjects } from \"./utils/select-projects.js\";\nimport { prompts } from \"./utils/prompts.js\";\nimport { maybePromptSkillInstall } from \"./utils/skill-prompt.js\";\n\nconst VERSION = process.env.VERSION ?? \"0.0.0\";\n\ninterface CliFlags {\n lint: boolean;\n deadCode: boolean;\n verbose: boolean;\n score: boolean;\n fix: boolean;\n yes: boolean;\n project?: string;\n}\n\nprocess.on(\"SIGINT\", () => process.exit(0));\nprocess.on(\"SIGTERM\", () => process.exit(0));\n\nconst program = new Command()\n .name(\"react-doctor\")\n .description(\"Diagnose React codebase health\")\n .version(VERSION, \"-v, --version\", \"display the version number\")\n .argument(\"[directory]\", \"project directory to scan\", \".\")\n .option(\"--no-lint\", \"skip linting\")\n .option(\"--no-dead-code\", \"skip dead code detection\")\n .option(\"--verbose\", \"show file details per rule\")\n .option(\"--score\", \"output only the score\")\n .option(\"-y, --yes\", \"skip prompts, scan all workspace projects\")\n .option(\"--project <name>\", \"select workspace project (comma-separated for multiple)\")\n .option(\"--fix\", \"open Ami to auto-fix all issues\")\n .action(async (directory: string, flags: CliFlags) => {\n try {\n const resolvedDirectory = path.resolve(directory);\n const isScoreOnly = flags.score;\n\n if (!isScoreOnly) {\n logger.log(`react-doctor v${VERSION}`);\n logger.break();\n }\n\n const scanOptions: ScanOptions = {\n lint: flags.lint,\n deadCode: flags.deadCode,\n verbose: Boolean(flags.verbose),\n scoreOnly: isScoreOnly,\n };\n\n const isAutomatedEnvironment = [\n process.env.CI,\n process.env.CLAUDECODE,\n process.env.CURSOR_TRACE_ID,\n process.env.CURSOR_AGENT,\n process.env.CODEX_CI,\n process.env.OPENCODE,\n process.env.AMP_HOME,\n process.env.AMI,\n ].some(Boolean);\n const shouldSkipPrompts = flags.yes || isAutomatedEnvironment || !process.stdin.isTTY;\n const projectDirectories = await selectProjects(\n resolvedDirectory,\n flags.project,\n shouldSkipPrompts,\n );\n\n for (const projectDirectory of projectDirectories) {\n if (!isScoreOnly) {\n logger.dim(`Scanning ${projectDirectory}...`);\n logger.break();\n }\n await scan(projectDirectory, scanOptions);\n if (!isScoreOnly) {\n logger.break();\n }\n }\n\n if (flags.fix) {\n openAmiToFix(resolvedDirectory);\n }\n\n if (!isScoreOnly) {\n await maybePromptSkillInstall(shouldSkipPrompts);\n if (!shouldSkipPrompts && !flags.fix) {\n await maybePromptAmiFix(resolvedDirectory);\n }\n }\n } catch (error) {\n handleError(error);\n }\n })\n .addHelpText(\n \"after\",\n `\n${highlighter.dim(\"Learn more:\")}\n ${highlighter.info(\"https://github.com/aidenybai/react-doctor\")}\n`,\n );\n\nconst AMI_INSTALL_URL = \"https://ami.dev/install.sh\";\nconst AMI_FIX_PROMPT =\n \"Run npx -y react-doctor@latest . --verbose, read every diagnostic, then fix all issues one by one. After fixing, re-run react-doctor to verify the score improved.\";\n\nconst isAmiInstalled = (): boolean => {\n try {\n execSync(\"ls /Applications/Ami.app\", { stdio: \"ignore\" });\n return true;\n } catch {\n return false;\n }\n};\n\nconst installAmi = (): void => {\n logger.log(\"Ami not found. Installing...\");\n logger.break();\n try {\n execSync(`curl -fsSL ${AMI_INSTALL_URL} | bash`, { stdio: \"inherit\" });\n } catch {\n logger.error(\"Failed to install Ami. Visit https://ami.dev to install manually.\");\n process.exit(1);\n }\n logger.break();\n};\n\nconst openAmiToFix = (directory: string): void => {\n const resolvedDirectory = path.resolve(directory);\n\n if (!isAmiInstalled()) {\n installAmi();\n }\n\n logger.log(\"Opening Ami to fix react-doctor issues...\");\n\n const encodedDirectory = encodeURIComponent(resolvedDirectory);\n const encodedPrompt = encodeURIComponent(AMI_FIX_PROMPT);\n const deeplink = `ami://open-project?cwd=${encodedDirectory}&prompt=${encodedPrompt}&mode=agent`;\n\n try {\n execSync(`open \"${deeplink}\"`, { stdio: \"ignore\" });\n logger.success(\"Opened Ami with react-doctor fix prompt.\");\n } catch {\n logger.break();\n logger.dim(\"Could not open Ami automatically. Open this URL manually:\");\n logger.info(deeplink);\n }\n};\n\nconst maybePromptAmiFix = async (directory: string): Promise<void> => {\n logger.break();\n logger.log(`Fix these issues with ${highlighter.info(\"Ami\")}?`);\n logger.dim(\" Ami is a coding agent built to understand your codebase and fix issues\");\n logger.dim(` automatically. Learn more at ${highlighter.info(\"https://ami.dev\")}`);\n logger.break();\n\n const { shouldFix } = await prompts({\n type: \"confirm\",\n name: \"shouldFix\",\n message: \"Open Ami to fix?\",\n initial: true,\n });\n\n if (shouldFix) {\n openAmiToFix(directory);\n }\n};\n\nconst fixAction = (directory: string) => {\n try {\n openAmiToFix(directory);\n } catch (error) {\n handleError(error);\n }\n};\n\nconst fixCommand = new Command(\"fix\")\n .description(\"Open Ami to auto-fix react-doctor issues\")\n .argument(\"[directory]\", \"project directory\", \".\")\n .action(fixAction);\n\nconst installAmiCommand = new Command(\"install-ami\")\n .description(\"Install Ami and open it to auto-fix issues\")\n .argument(\"[directory]\", \"project directory\", \".\")\n .action(fixAction);\n\nprogram.addCommand(fixCommand);\nprogram.addCommand(installAmiCommand);\n\nconst main = async () => {\n await program.parseAsync();\n};\n\nmain();\n"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "react-doctor",
3
- "version": "0.0.7",
3
+ "version": "0.0.8",
4
4
  "bin": {
5
5
  "react-doctor": "./dist/cli.js"
6
6
  },