UNPKG

@nlabs/lex

Version:
429 lines (420 loc) 43.5 kB
import { execa } from "execa"; import { existsSync, readFileSync } from "fs"; import { sync as globSync } from "glob"; import { resolve as pathResolve } from "path"; import { LexConfig, getTypeScriptConfigPath } from "../../LexConfig.js"; import { createSpinner } from "../../utils/app.js"; import { getDirName, resolveBinaryPath } from "../../utils/file.js"; import { log } from "../../utils/log.js"; import { aiFunction } from "../ai/ai.js"; const detectESM = (cwd) => { const packageJsonPath = pathResolve(cwd, "package.json"); if (existsSync(packageJsonPath)) { try { const packageJsonContent = readFileSync(packageJsonPath, "utf8"); const packageJson = JSON.parse(packageJsonContent); return packageJson.type === "module"; } catch (_error) { return false; } } return false; }; const getTestFilePatterns = (testPathPattern) => { const defaultPatterns = ["**/*.test.*", "**/*.spec.*", "**/*.integration.*"]; if (!testPathPattern) { return defaultPatterns; } return [testPathPattern]; }; const findUncoveredSourceFiles = () => { const sourceFiles = globSync("src/**/*.{ts,tsx,js,jsx}", { cwd: process.cwd(), ignore: ["**/node_modules/**", "**/dist/**", "**/lib/**", "**/*.test.*", "**/*.spec.*"] }); const testFiles = globSync("**/*.{test,spec}.{ts,tsx,js,jsx}", { cwd: process.cwd(), ignore: ["**/node_modules/**", "**/dist/**", "**/lib/**"] }); return sourceFiles.filter((sourceFile) => { const baseName = sourceFile.replace(/\.[^/.]+$/, ""); return !testFiles.some((testFile) => testFile.includes(baseName)); }); }; const processTestResults = (outputFile) => { if (!outputFile) { return null; } try { const content = readFileSync(outputFile, "utf-8"); return JSON.parse(content); } catch (_error) { return null; } }; const test = async (options, args, callback = process.exit) => { const { analyze = false, aiAnalyze = false, aiDebug = false, aiGenerate = false, bail, changedFilesWithAncestor, changedSince, ci, cliName = "Lex", collectCoverageFrom, colors, config, debug = false, debugTests = false, detectOpenHandles, env, errorOnDeprecated, expand, forceExit, generate = false, json, lastCommit, listTests, logHeapUsage, maxWorkers, noStackTrace, notify, onlyChanged, outputFile, passWithNoTests, quiet, removeCache, runInBand, setup, showConfig, silent, testLocationInResults, testNamePattern, testPathPattern, update, useStderr, verbose, watch, watchAll } = options; const useGenerate = generate || aiGenerate; const useAnalyze = analyze || aiAnalyze; const useDebug = debugTests || aiDebug; log(`${cliName} testing...`, "info", quiet); const spinner = createSpinner(quiet); await LexConfig.parseConfig(options); const { useTypescript } = LexConfig.config; if (useTypescript) { const testConfigPath = getTypeScriptConfigPath("tsconfig.test.json"); if (existsSync(testConfigPath)) { log("Using tsconfig.test.json for testing...", "info", quiet); } else { LexConfig.checkTestTypescriptConfig(); } } if (useGenerate) { spinner.start("AI is analyzing code to generate test cases..."); try { const uncoveredFiles = findUncoveredSourceFiles(); if (uncoveredFiles.length > 0) { const targetFile = uncoveredFiles[0]; await aiFunction({ context: true, file: targetFile, prompt: `Generate Jest unit tests for this file: ${targetFile} ${readFileSync(targetFile, "utf-8")} Please create comprehensive tests that cover the main functionality. Include test fixtures and mocks where necessary.`, quiet, task: "test" }); spinner.succeed(`AI test generation suggestions provided for ${targetFile}`); } else { spinner.succeed("All source files appear to have corresponding test files"); } } catch (aiError) { spinner.fail("Could not generate AI test suggestions"); if (!quiet) { console.error("AI test generation error:", aiError); } } } const dirName = getDirName(); const projectJestBin = pathResolve(process.cwd(), "node_modules/.bin/jest"); let jestPath; if (existsSync(projectJestBin)) { jestPath = projectJestBin; } else { jestPath = resolveBinaryPath("jest"); } if (!jestPath) { log(` ${cliName} Error: Jest binary not found in Lex's node_modules or monorepo root`, "error", quiet); log("Please reinstall Lex or check your installation.", "info", quiet); return 1; } let jestConfigFile; let projectJestConfig = null; if (config) { jestConfigFile = config; } else { const projectJestConfigPath = pathResolve(process.cwd(), "jest.config.js"); const projectJestConfigCjsPath = pathResolve(process.cwd(), "jest.config.cjs"); const projectJestConfigMjsPath = pathResolve(process.cwd(), "jest.config.mjs"); const projectJestConfigJsonPath = pathResolve(process.cwd(), "jest.config.json"); if (existsSync(projectJestConfigPath)) { jestConfigFile = projectJestConfigPath; if (debug) { log(`Using project Jest config file: ${jestConfigFile}`, "info", quiet); } } else if (existsSync(projectJestConfigCjsPath)) { jestConfigFile = projectJestConfigCjsPath; if (debug) { log(`Using project Jest config file (CJS): ${jestConfigFile}`, "info", quiet); } } else if (existsSync(projectJestConfigMjsPath)) { jestConfigFile = projectJestConfigMjsPath; if (debug) { log(`Using project Jest config file (MJS): ${jestConfigFile}`, "info", quiet); } } else if (existsSync(projectJestConfigJsonPath)) { jestConfigFile = projectJestConfigJsonPath; if (debug) { log(`Using project Jest config file (JSON): ${jestConfigFile}`, "info", quiet); } } else { projectJestConfig = LexConfig.config.jest; const lexDir = LexConfig.getLexDir(); const lexJestConfig = pathResolve(lexDir, "jest.config.mjs"); if (debug) { log(`Looking for Jest config at: ${lexJestConfig}`, "info", quiet); log(`File exists: ${existsSync(lexJestConfig)}`, "info", quiet); } if (existsSync(lexJestConfig)) { jestConfigFile = lexJestConfig; if (projectJestConfig && Object.keys(projectJestConfig).length > 0) { if (debug) { log(`Using Lex Jest config with project Jest config from lex.config.cjs: ${jestConfigFile}`, "info", quiet); } } else { if (debug) { log(`Using Lex Jest config (no project Jest config found): ${jestConfigFile}`, "info", quiet); } } } else { if (debug) { log("No Jest config found in project or Lex", "warn", quiet); } jestConfigFile = ""; } } } const jestSetupFile = setup || pathResolve(process.cwd(), "jest.setup.js"); const jestOptions = ["--no-cache"]; const isESM = detectESM(process.cwd()); let nodeOptions = process.env.NODE_OPTIONS || ""; if (isESM) { if (!nodeOptions.includes("--experimental-vm-modules")) { nodeOptions = `${nodeOptions} --experimental-vm-modules`.trim(); } log("ESM project detected, using --experimental-vm-modules in NODE_OPTIONS", "info", quiet); } if (jestConfigFile) { jestOptions.push("--config", jestConfigFile); } if (bail) { jestOptions.push("--bail"); } if (changedFilesWithAncestor) { jestOptions.push("--changedFilesWithAncestor"); } if (changedSince) { jestOptions.push("--changedSince"); } if (ci) { jestOptions.push("--ci"); } if (collectCoverageFrom) { jestOptions.push("--collectCoverageFrom", collectCoverageFrom); } if (colors) { jestOptions.push("--colors"); } if (debug) { jestOptions.push("--debug"); } if (detectOpenHandles) { jestOptions.push("--detectOpenHandles"); } if (env) { jestOptions.push("--env"); } if (errorOnDeprecated) { jestOptions.push("--errorOnDeprecated"); } if (expand) { jestOptions.push("--expand"); } if (forceExit) { jestOptions.push("--forceExit"); } if (json) { jestOptions.push("--json"); } if (lastCommit) { jestOptions.push("--lastCommit"); } if (listTests) { jestOptions.push("--listTests"); } if (logHeapUsage) { jestOptions.push("--logHeapUsage"); } if (maxWorkers) { jestOptions.push("--maxWorkers", maxWorkers); } if (noStackTrace) { jestOptions.push("--noStackTrace"); } if (notify) { jestOptions.push("--notify"); } if (onlyChanged) { jestOptions.push("--onlyChanged"); } let tempOutputFile = outputFile; if ((useAnalyze || useDebug) && !outputFile) { tempOutputFile = ".lex-test-results.json"; jestOptions.push("--json", "--outputFile", tempOutputFile); } else if (outputFile) { jestOptions.push("--outputFile", outputFile); } if (passWithNoTests) { jestOptions.push("--passWithNoTests"); } if (runInBand) { jestOptions.push("--runInBand"); } if (showConfig) { jestOptions.push("--showConfig"); } if (silent) { jestOptions.push("--silent"); } if (testLocationInResults) { jestOptions.push("--testLocationInResults"); } if (testNamePattern) { jestOptions.push("--testNamePattern", testNamePattern); } if (testPathPattern) { jestOptions.push("--testPathPattern", testPathPattern); } if (useStderr) { jestOptions.push("--useStderr"); } if (verbose) { jestOptions.push("--verbose"); } if (watchAll) { jestOptions.push("--watchAll"); } if (removeCache) { jestOptions.push("--no-cache"); } if (jestSetupFile && existsSync(jestSetupFile)) { jestOptions.push(`--setupFilesAfterEnv=${jestSetupFile}`); } if (update) { jestOptions.push("--updateSnapshot"); } if (watch) { jestOptions.push("--watch", watch); } if (args) { jestOptions.push(...args); } if (debug) { log(`Jest options: ${jestOptions.join(" ")}`, "info", quiet); log(`NODE_OPTIONS: ${nodeOptions}`, "info", quiet); } try { const env2 = { ...process.env, NODE_OPTIONS: nodeOptions }; await execa(jestPath, jestOptions, { encoding: "utf8", stdio: "inherit", env: env2 }); spinner.succeed("Testing completed!"); if (useAnalyze) { spinner.start("AI is analyzing test coverage and suggesting improvements..."); try { const testResults = processTestResults(tempOutputFile); const filePatterns = getTestFilePatterns(testPathPattern); await aiFunction({ prompt: `Analyze these Jest test results and suggest test coverage improvements: ${JSON.stringify(testResults, null, 2)} Test patterns: ${filePatterns.join(", ")} Please provide: 1. Analysis of current coverage gaps 2. Suggestions for improving test cases 3. Recommendations for additional integration test scenarios 4. Best practices for increasing test effectiveness`, task: "optimize", context: true, quiet }); spinner.succeed("AI test analysis complete"); } catch (aiError) { spinner.fail("Could not generate AI test analysis"); if (!quiet) { console.error("AI analysis error:", aiError); } } } callback(0); return 0; } catch (error) { log(` ${cliName} Error: Check for unit test errors and/or coverage.`, "error", quiet); spinner.fail("Testing failed!"); if (useDebug) { spinner.start("AI is analyzing test failures..."); try { const testResults = processTestResults(tempOutputFile); await aiFunction({ context: true, prompt: `Debug these failed Jest tests and suggest fixes: ${JSON.stringify(error.message, null, 2)} Test results: ${JSON.stringify(testResults, null, 2)} Please provide: 1. Analysis of why the tests are failing 2. Specific suggestions to fix each failing test 3. Any potential issues with test fixtures or mocks 4. Code examples for solutions`, quiet, task: "help" }); spinner.succeed("AI debugging assistance complete"); } catch (aiError) { spinner.fail("Could not generate AI debugging assistance"); if (!quiet) { console.error("AI debugging error:", aiError); } } } callback(1); return 1; } }; var test_default = test; export { test_default as default, getTestFilePatterns, test }; //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/commands/test/test.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2018-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {execa} from 'execa';\nimport {existsSync, readFileSync} from 'fs';\nimport {sync as globSync} from 'glob';\nimport {resolve as pathResolve} from 'path';\n\nimport {LexConfig, getTypeScriptConfigPath} from '../../LexConfig.js';\nimport {createSpinner} from '../../utils/app.js';\nimport {getDirName, resolveBinaryPath} from '../../utils/file.js';\nimport {log} from '../../utils/log.js';\nimport {aiFunction} from '../ai/ai.js';\n\nconst detectESM = (cwd: string): boolean => {\n  const packageJsonPath = pathResolve(cwd, 'package.json');\n\n  if(existsSync(packageJsonPath)) {\n    try {\n      const packageJsonContent = readFileSync(packageJsonPath, 'utf8');\n      const packageJson = JSON.parse(packageJsonContent);\n      return packageJson.type === 'module';\n    } catch (_error) {\n      return false;\n    }\n  }\n\n  return false;\n};\n\nexport interface TestOptions {\n  readonly analyze?: boolean;\n  readonly aiDebug?: boolean;\n  readonly aiGenerate?: boolean;\n  readonly aiAnalyze?: boolean;\n  readonly bail?: boolean;\n  readonly changedFilesWithAncestor?: boolean;\n  readonly changedSince?: string;\n  readonly ci?: boolean;\n  readonly cliName?: string;\n  readonly collectCoverageFrom?: string;\n  readonly colors?: boolean;\n  readonly config?: string;\n  readonly debug?: boolean;\n  readonly debugTests?: boolean;\n  readonly detectOpenHandles?: boolean;\n  readonly env?: string;\n  readonly errorOnDeprecated?: boolean;\n  readonly expand?: boolean;\n  readonly forceExit?: boolean;\n  readonly generate?: boolean;\n  readonly json?: boolean;\n  readonly lastCommit?: boolean;\n  readonly listTests?: boolean;\n  readonly logHeapUsage?: boolean;\n  readonly maxWorkers?: string;\n  readonly noStackTrace?: boolean;\n  readonly notify?: boolean;\n  readonly onlyChanged?: boolean;\n  readonly outputFile?: string;\n  readonly passWithNoTests?: boolean;\n  readonly quiet?: boolean;\n  readonly removeCache?: boolean;\n  readonly runInBand?: boolean;\n  readonly setup?: string;\n  readonly showConfig?: boolean;\n  readonly silent?: boolean;\n  readonly testLocationInResults?: boolean;\n  readonly testNamePattern?: string;\n  readonly testPathPattern?: string;\n  readonly update?: boolean;\n  readonly useStderr?: boolean;\n  readonly verbose?: boolean;\n  readonly watch?: string;\n  readonly watchAll?: boolean;\n}\n\nexport type TestCallback = typeof process.exit;\n\nexport const getTestFilePatterns = (testPathPattern?: string): string[] => {\n  const defaultPatterns = ['**/*.test.*', '**/*.spec.*', '**/*.integration.*'];\n\n  if(!testPathPattern) {\n    return defaultPatterns;\n  }\n\n  return [testPathPattern];\n};\n\nconst findUncoveredSourceFiles = (): string[] => {\n  const sourceFiles = globSync('src/**/*.{ts,tsx,js,jsx}', {\n    cwd: process.cwd(),\n    ignore: ['**/node_modules/**', '**/dist/**', '**/lib/**', '**/*.test.*', '**/*.spec.*']\n  });\n\n  const testFiles = globSync('**/*.{test,spec}.{ts,tsx,js,jsx}', {\n    cwd: process.cwd(),\n    ignore: ['**/node_modules/**', '**/dist/**', '**/lib/**']\n  });\n\n  return sourceFiles.filter((sourceFile) => {\n    const baseName = sourceFile.replace(/\\.[^/.]+$/, '');\n    return !testFiles.some((testFile) => testFile.includes(baseName));\n  });\n};\n\nconst processTestResults = (outputFile?: string): any => {\n  if(!outputFile) {\n    return null;\n  }\n\n  try {\n    const content = readFileSync(outputFile, 'utf-8');\n    return JSON.parse(content);\n  } catch (_error) {\n    return null;\n  }\n};\n\nexport const test = async (\n  options: TestOptions,\n  args: string[],\n  callback: TestCallback = process.exit\n): Promise<number> => {\n  const {\n    analyze = false,\n    aiAnalyze = false,\n    aiDebug = false,\n    aiGenerate = false,\n    bail,\n    changedFilesWithAncestor,\n    changedSince,\n    ci,\n    cliName = 'Lex',\n    collectCoverageFrom,\n    colors,\n    config,\n    debug = false,\n    debugTests = false,\n    detectOpenHandles,\n    env,\n    errorOnDeprecated,\n    expand,\n    forceExit,\n    generate = false,\n    json,\n    lastCommit,\n    listTests,\n    logHeapUsage,\n    maxWorkers,\n    noStackTrace,\n    notify,\n    onlyChanged,\n    outputFile,\n    passWithNoTests,\n    quiet,\n    removeCache,\n    runInBand,\n    setup,\n    showConfig,\n    silent,\n    testLocationInResults,\n    testNamePattern,\n    testPathPattern,\n    update,\n    useStderr,\n    verbose,\n    watch,\n    watchAll\n  } = options;\n\n  const useGenerate = generate || aiGenerate;\n  const useAnalyze = analyze || aiAnalyze;\n  const useDebug = debugTests || aiDebug;\n\n  log(`${cliName} testing...`, 'info', quiet);\n\n  const spinner = createSpinner(quiet);\n\n  await LexConfig.parseConfig(options);\n\n  const {useTypescript} = LexConfig.config;\n\n  if(useTypescript) {\n    const testConfigPath = getTypeScriptConfigPath('tsconfig.test.json');\n    if(existsSync(testConfigPath)) {\n      log('Using tsconfig.test.json for testing...', 'info', quiet);\n    } else {\n      LexConfig.checkTestTypescriptConfig();\n    }\n  }\n\n  if(useGenerate) {\n    spinner.start('AI is analyzing code to generate test cases...');\n\n    try {\n      const uncoveredFiles = findUncoveredSourceFiles();\n\n      if(uncoveredFiles.length > 0) {\n        const targetFile = uncoveredFiles[0];\n\n        await aiFunction({\n          context: true,\n          file: targetFile,\n          prompt: `Generate Jest unit tests for this file: ${targetFile}\\n\\n${readFileSync(targetFile, 'utf-8')}\\n\\nPlease create comprehensive tests that cover the main functionality. Include test fixtures and mocks where necessary.`,\n          quiet,\n          task: 'test'\n        });\n\n        spinner.succeed(`AI test generation suggestions provided for ${targetFile}`);\n      } else {\n        spinner.succeed('All source files appear to have corresponding test files');\n      }\n    } catch (aiError) {\n      spinner.fail('Could not generate AI test suggestions');\n      if(!quiet) {\n        // eslint-disable-next-line no-console\n        console.error('AI test generation error:', aiError);\n      }\n    }\n  }\n\n  const dirName = getDirName();\n\n  const projectJestBin = pathResolve(process.cwd(), 'node_modules/.bin/jest');\n  let jestPath: string;\n\n  if(existsSync(projectJestBin)) {\n    jestPath = projectJestBin;\n  } else {\n    jestPath = resolveBinaryPath('jest');\n  }\n\n  if(!jestPath) {\n    log(`\\n${cliName} Error: Jest binary not found in Lex's node_modules or monorepo root`, 'error', quiet);\n    log('Please reinstall Lex or check your installation.', 'info', quiet);\n    return 1;\n  }\n\n  let jestConfigFile: string;\n  let projectJestConfig: any = null;\n\n  if(config) {\n    jestConfigFile = config;\n  } else {\n    const projectJestConfigPath = pathResolve(process.cwd(), 'jest.config.js');\n    const projectJestConfigCjsPath = pathResolve(process.cwd(), 'jest.config.cjs');\n    const projectJestConfigMjsPath = pathResolve(process.cwd(), 'jest.config.mjs');\n    const projectJestConfigJsonPath = pathResolve(process.cwd(), 'jest.config.json');\n\n    if(existsSync(projectJestConfigPath)) {\n      jestConfigFile = projectJestConfigPath;\n      if(debug) {\n        log(`Using project Jest config file: ${jestConfigFile}`, 'info', quiet);\n      }\n    } else if(existsSync(projectJestConfigCjsPath)) {\n      jestConfigFile = projectJestConfigCjsPath;\n      if(debug) {\n        log(`Using project Jest config file (CJS): ${jestConfigFile}`, 'info', quiet);\n      }\n    } else if(existsSync(projectJestConfigMjsPath)) {\n      jestConfigFile = projectJestConfigMjsPath;\n      if(debug) {\n        log(`Using project Jest config file (MJS): ${jestConfigFile}`, 'info', quiet);\n      }\n    } else if(existsSync(projectJestConfigJsonPath)) {\n      jestConfigFile = projectJestConfigJsonPath;\n      if(debug) {\n        log(`Using project Jest config file (JSON): ${jestConfigFile}`, 'info', quiet);\n      }\n    } else {\n      // No Jest config file exists in the project\n      // Check if there's a Jest config in lex.config.cjs\n      projectJestConfig = LexConfig.config.jest;\n\n      const lexDir = LexConfig.getLexDir();\n      const lexJestConfig = pathResolve(lexDir, 'jest.config.mjs');\n\n      if(debug) {\n        log(`Looking for Jest config at: ${lexJestConfig}`, 'info', quiet);\n        log(`File exists: ${existsSync(lexJestConfig)}`, 'info', quiet);\n      }\n\n      if(existsSync(lexJestConfig)) {\n        jestConfigFile = lexJestConfig;\n        if(projectJestConfig && Object.keys(projectJestConfig).length > 0) {\n          if(debug) {\n            log(`Using Lex Jest config with project Jest config from lex.config.cjs: ${jestConfigFile}`, 'info', quiet);\n          }\n        } else {\n          if(debug) {\n            log(`Using Lex Jest config (no project Jest config found): ${jestConfigFile}`, 'info', quiet);\n          }\n        }\n      } else {\n        if(debug) {\n          log('No Jest config found in project or Lex', 'warn', quiet);\n        }\n        jestConfigFile = '';\n      }\n    }\n  }\n\n  const jestSetupFile: string = setup || pathResolve(process.cwd(), 'jest.setup.js');\n  const jestOptions: string[] = ['--no-cache'];\n\n  const isESM = detectESM(process.cwd());\n  let nodeOptions = process.env.NODE_OPTIONS || '';\n  if(isESM) {\n    if(!nodeOptions.includes('--experimental-vm-modules')) {\n      nodeOptions = `${nodeOptions} --experimental-vm-modules`.trim();\n    }\n    log('ESM project detected, using --experimental-vm-modules in NODE_OPTIONS', 'info', quiet);\n  }\n\n  if(jestConfigFile) {\n    jestOptions.push('--config', jestConfigFile);\n  }\n\n  if(bail) {\n    jestOptions.push('--bail');\n  }\n\n  if(changedFilesWithAncestor) {\n    jestOptions.push('--changedFilesWithAncestor');\n  }\n\n  if(changedSince) {\n    jestOptions.push('--changedSince');\n  }\n\n  if(ci) {\n    jestOptions.push('--ci');\n  }\n\n  if(collectCoverageFrom) {\n    jestOptions.push('--collectCoverageFrom', collectCoverageFrom);\n  }\n\n  if(colors) {\n    jestOptions.push('--colors');\n  }\n\n  if(debug) {\n    jestOptions.push('--debug');\n  }\n\n  if(detectOpenHandles) {\n    jestOptions.push('--detectOpenHandles');\n  }\n\n  if(env) {\n    jestOptions.push('--env');\n  }\n\n  if(errorOnDeprecated) {\n    jestOptions.push('--errorOnDeprecated');\n  }\n\n  if(expand) {\n    jestOptions.push('--expand');\n  }\n\n  if(forceExit) {\n    jestOptions.push('--forceExit');\n  }\n\n  if(json) {\n    jestOptions.push('--json');\n  }\n\n  if(lastCommit) {\n    jestOptions.push('--lastCommit');\n  }\n\n  if(listTests) {\n    jestOptions.push('--listTests');\n  }\n\n  if(logHeapUsage) {\n    jestOptions.push('--logHeapUsage');\n  }\n\n  if(maxWorkers) {\n    jestOptions.push('--maxWorkers', maxWorkers);\n  }\n\n  if(noStackTrace) {\n    jestOptions.push('--noStackTrace');\n  }\n\n  if(notify) {\n    jestOptions.push('--notify');\n  }\n\n  if(onlyChanged) {\n    jestOptions.push('--onlyChanged');\n  }\n\n  let tempOutputFile = outputFile;\n\n  if((useAnalyze || useDebug) && !outputFile) {\n    tempOutputFile = '.lex-test-results.json';\n    jestOptions.push('--json', '--outputFile', tempOutputFile);\n  } else if(outputFile) {\n    jestOptions.push('--outputFile', outputFile);\n  }\n\n  if(passWithNoTests) {\n    jestOptions.push('--passWithNoTests');\n  }\n\n  if(runInBand) {\n    jestOptions.push('--runInBand');\n  }\n\n  if(showConfig) {\n    jestOptions.push('--showConfig');\n  }\n\n  if(silent) {\n    jestOptions.push('--silent');\n  }\n\n  if(testLocationInResults) {\n    jestOptions.push('--testLocationInResults');\n  }\n\n  if(testNamePattern) {\n    jestOptions.push('--testNamePattern', testNamePattern);\n  }\n\n  if(testPathPattern) {\n    jestOptions.push('--testPathPattern', testPathPattern);\n  }\n\n  if(useStderr) {\n    jestOptions.push('--useStderr');\n  }\n\n  if(verbose) {\n    jestOptions.push('--verbose');\n  }\n\n  if(watchAll) {\n    jestOptions.push('--watchAll');\n  }\n\n  if(removeCache) {\n    jestOptions.push('--no-cache');\n  }\n\n  if(jestSetupFile && existsSync(jestSetupFile)) {\n    jestOptions.push(`--setupFilesAfterEnv=${jestSetupFile}`);\n  }\n\n  if(update) {\n    jestOptions.push('--updateSnapshot');\n  }\n\n  if(watch) {\n    jestOptions.push('--watch', watch);\n  }\n\n  if(args) {\n    jestOptions.push(...args);\n  }\n\n  if(debug) {\n    log(`Jest options: ${jestOptions.join(' ')}`, 'info', quiet);\n    log(`NODE_OPTIONS: ${nodeOptions}`, 'info', quiet);\n  }\n\n  try {\n    const env: Record<string, string> = {\n      ...process.env,\n      NODE_OPTIONS: nodeOptions\n    };\n\n    await execa(jestPath, jestOptions, {\n      encoding: 'utf8',\n      stdio: 'inherit',\n      env\n    });\n\n    spinner.succeed('Testing completed!');\n\n    if(useAnalyze) {\n      spinner.start('AI is analyzing test coverage and suggesting improvements...');\n\n      try {\n        const testResults = processTestResults(tempOutputFile);\n        const filePatterns = getTestFilePatterns(testPathPattern);\n\n        await aiFunction({\n          prompt: `Analyze these Jest test results and suggest test coverage improvements:\n\n${JSON.stringify(testResults, null, 2)}\n\nTest patterns: ${filePatterns.join(', ')}\n\nPlease provide:\n1. Analysis of current coverage gaps\n2. Suggestions for improving test cases\n3. Recommendations for additional integration test scenarios\n4. Best practices for increasing test effectiveness`,\n          task: 'optimize',\n          context: true,\n          quiet\n        });\n\n        spinner.succeed('AI test analysis complete');\n      } catch (aiError) {\n        spinner.fail('Could not generate AI test analysis');\n        if(!quiet) {\n          // eslint-disable-next-line no-console\n          console.error('AI analysis error:', aiError);\n        }\n      }\n    }\n\n    callback(0);\n    return 0;\n  } catch (error) {\n    log(`\\n${cliName} Error: Check for unit test errors and/or coverage.`, 'error', quiet);\n\n    spinner.fail('Testing failed!');\n\n    if(useDebug) {\n      spinner.start('AI is analyzing test failures...');\n\n      try {\n        const testResults = processTestResults(tempOutputFile);\n\n        await aiFunction({\n          context: true,\n          prompt: `Debug these failed Jest tests and suggest fixes:\n\n${JSON.stringify(error.message, null, 2)}\n\nTest results: ${JSON.stringify(testResults, null, 2)}\n\nPlease provide:\n1. Analysis of why the tests are failing\n2. Specific suggestions to fix each failing test\n3. Any potential issues with test fixtures or mocks\n4. Code examples for solutions`,\n          quiet,\n          task: 'help'\n        });\n\n        spinner.succeed('AI debugging assistance complete');\n      } catch (aiError) {\n        spinner.fail('Could not generate AI debugging assistance');\n        if(!quiet) {\n          // eslint-disable-next-line no-console\n          console.error('AI debugging error:', aiError);\n        }\n      }\n    }\n\n    callback(1);\n    return 1;\n  }\n};\n\nexport default test;"],
  "mappings": "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",
  "names": ["env"]
}
