@fragments-sdk/cli 0.11.1 → 0.12.1

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.
Files changed (86) hide show
  1. package/dist/ai-client-I6MDWNYA.js +21 -0
  2. package/dist/bin.js +275 -368
  3. package/dist/bin.js.map +1 -1
  4. package/dist/{chunk-PW7QTQA6.js → chunk-4OC7FTJB.js} +2 -2
  5. package/dist/{chunk-HRFUSSZI.js → chunk-AM4MRTMN.js} +2 -2
  6. package/dist/{chunk-5G3VZH43.js → chunk-GVDSFQ4E.js} +281 -351
  7. package/dist/chunk-GVDSFQ4E.js.map +1 -0
  8. package/dist/chunk-JJ2VRTBU.js +626 -0
  9. package/dist/chunk-JJ2VRTBU.js.map +1 -0
  10. package/dist/{chunk-D5PYOXEI.js → chunk-LVWFOLUZ.js} +148 -13
  11. package/dist/{chunk-D5PYOXEI.js.map → chunk-LVWFOLUZ.js.map} +1 -1
  12. package/dist/{chunk-WXSR2II7.js → chunk-OQKMEFOS.js} +58 -6
  13. package/dist/chunk-OQKMEFOS.js.map +1 -0
  14. package/dist/chunk-SXTKFDCR.js +104 -0
  15. package/dist/chunk-SXTKFDCR.js.map +1 -0
  16. package/dist/chunk-T5OMVL7E.js +443 -0
  17. package/dist/chunk-T5OMVL7E.js.map +1 -0
  18. package/dist/{chunk-ZM4ZQZWZ.js → chunk-TPWGL2XS.js} +39 -37
  19. package/dist/chunk-TPWGL2XS.js.map +1 -0
  20. package/dist/{chunk-OQO55NKV.js → chunk-WFS63PCW.js} +85 -11
  21. package/dist/chunk-WFS63PCW.js.map +1 -0
  22. package/dist/core/index.js +9 -1
  23. package/dist/{discovery-NEOY4MPN.js → discovery-ZJQSXF56.js} +3 -3
  24. package/dist/{generate-FBHSXR3D.js → generate-RJFS2JWA.js} +4 -4
  25. package/dist/index.js +7 -6
  26. package/dist/index.js.map +1 -1
  27. package/dist/init-ZSX3NRCZ.js +636 -0
  28. package/dist/init-ZSX3NRCZ.js.map +1 -0
  29. package/dist/mcp-bin.js +2 -2
  30. package/dist/{scan-CJF2DOQW.js → scan-3PMCJ4RB.js} +6 -6
  31. package/dist/scan-generate-SYU4PYZD.js +1115 -0
  32. package/dist/scan-generate-SYU4PYZD.js.map +1 -0
  33. package/dist/{service-TQYWY65E.js → service-VMGNJZ42.js} +3 -3
  34. package/dist/{snapshot-SV2JOFZH.js → snapshot-XOISO2IS.js} +2 -2
  35. package/dist/{static-viewer-NUBFPKWH.js → static-viewer-5GXH2MGE.js} +3 -3
  36. package/dist/static-viewer-5GXH2MGE.js.map +1 -0
  37. package/dist/{test-Z5LVO724.js → test-SI4NSHQX.js} +4 -4
  38. package/dist/{tokens-CE46OTMD.js → tokens-T6SIVUT5.js} +5 -5
  39. package/dist/{viewer-DLLJIMCK.js → viewer-7ZEAFBVN.js} +13 -13
  40. package/package.json +4 -4
  41. package/src/ai-client.ts +156 -0
  42. package/src/bin.ts +44 -2
  43. package/src/build.ts +95 -33
  44. package/src/commands/__tests__/drift-sync.test.ts +252 -0
  45. package/src/commands/__tests__/scan-generate.test.ts +497 -45
  46. package/src/commands/enhance.ts +11 -35
  47. package/src/commands/init.ts +288 -260
  48. package/src/commands/scan-generate.ts +740 -139
  49. package/src/commands/scan.ts +37 -32
  50. package/src/commands/setup.ts +143 -52
  51. package/src/commands/sync.ts +357 -0
  52. package/src/commands/validate.ts +43 -1
  53. package/src/core/component-extractor.test.ts +282 -0
  54. package/src/core/component-extractor.ts +1030 -0
  55. package/src/core/discovery.ts +93 -7
  56. package/src/service/enhance/props-extractor.ts +235 -13
  57. package/src/validators.ts +236 -0
  58. package/dist/chunk-5G3VZH43.js.map +0 -1
  59. package/dist/chunk-OQO55NKV.js.map +0 -1
  60. package/dist/chunk-WXSR2II7.js.map +0 -1
  61. package/dist/chunk-ZM4ZQZWZ.js.map +0 -1
  62. package/dist/init-UFGK5TCN.js +0 -867
  63. package/dist/init-UFGK5TCN.js.map +0 -1
  64. package/dist/scan-generate-SJAN5MVI.js +0 -691
  65. package/dist/scan-generate-SJAN5MVI.js.map +0 -1
  66. package/src/ai.ts +0 -266
  67. package/src/commands/init-framework.ts +0 -414
  68. package/src/mcp/bin.ts +0 -36
  69. package/src/migrate/bin.ts +0 -114
  70. package/src/theme/index.ts +0 -77
  71. package/src/viewer/bin.ts +0 -86
  72. package/src/viewer/cli/health.ts +0 -256
  73. package/src/viewer/cli/index.ts +0 -33
  74. package/src/viewer/cli/scan.ts +0 -124
  75. package/src/viewer/cli/utils.ts +0 -174
  76. /package/dist/{discovery-NEOY4MPN.js.map → ai-client-I6MDWNYA.js.map} +0 -0
  77. /package/dist/{chunk-PW7QTQA6.js.map → chunk-4OC7FTJB.js.map} +0 -0
  78. /package/dist/{chunk-HRFUSSZI.js.map → chunk-AM4MRTMN.js.map} +0 -0
  79. /package/dist/{scan-CJF2DOQW.js.map → discovery-ZJQSXF56.js.map} +0 -0
  80. /package/dist/{generate-FBHSXR3D.js.map → generate-RJFS2JWA.js.map} +0 -0
  81. /package/dist/{service-TQYWY65E.js.map → scan-3PMCJ4RB.js.map} +0 -0
  82. /package/dist/{static-viewer-NUBFPKWH.js.map → service-VMGNJZ42.js.map} +0 -0
  83. /package/dist/{snapshot-SV2JOFZH.js.map → snapshot-XOISO2IS.js.map} +0 -0
  84. /package/dist/{test-Z5LVO724.js.map → test-SI4NSHQX.js.map} +0 -0
  85. /package/dist/{tokens-CE46OTMD.js.map → tokens-T6SIVUT5.js.map} +0 -0
  86. /package/dist/{viewer-DLLJIMCK.js.map → viewer-7ZEAFBVN.js.map} +0 -0
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/commands/scan-generate.ts"],"sourcesContent":["/**\n * fragments init --scan - Generate fragment files from any TypeScript component library\n *\n * Phase 2 of the Universal GenUI strategy. Scans a component directory and\n * generates .fragment.tsx files with TODO markers for uncertain fields.\n *\n * Combines:\n * - Component discovery (core/discovery.ts)\n * - ComponentExtractor: persistent LanguageService-based prop + compound extraction\n * - Confidence scoring with TODO markers\n * - Composition-aware code generation\n */\n\nimport { readFile, writeFile, access, mkdir } from \"node:fs/promises\";\nimport { resolve, basename, dirname, relative, join } from \"node:path\";\nimport * as ts from \"typescript\";\nimport pc from \"picocolors\";\nimport { BRAND } from \"../core/index.js\";\nimport {\n discoverAllComponents,\n type DiscoveredComponent,\n} from \"../core/node.js\";\nimport {\n createComponentExtractor,\n type ComponentExtractor,\n type ComponentMeta,\n type PropMeta,\n type CompositionMeta,\n} from \"../core/component-extractor.js\";\n\n// ---------------------------------------------------------------------------\n// Types\n// ---------------------------------------------------------------------------\n\nexport interface ScanGenerateOptions {\n /** Path to scan for components */\n scanPath: string;\n /** Output directory (default: colocated with source) */\n outputDir?: string;\n /** Overwrite existing fragment files */\n force?: boolean;\n /** Custom component file patterns */\n patterns?: string[];\n /** Skip Storybook story extraction */\n skipStorybook?: boolean;\n /** Verbose logging */\n verbose?: boolean;\n /** Path to tsconfig.json for accurate type resolution */\n tsconfig?: string;\n /** Enable LLM enrichment of knowledge fields */\n enrich?: boolean;\n /** Show what would be enriched without calling API */\n dryRun?: boolean;\n /** AI provider override: anthropic or openai */\n provider?: 'anthropic' | 'openai';\n /** Explicit API key for enrichment */\n apiKey?: string;\n /** Override default AI model for enrichment */\n model?: string;\n}\n\nexport interface ScanGenerateResult {\n success: boolean;\n generated: Array<{\n name: string;\n path: string;\n confidence: number;\n todoCount: number;\n enriched?: boolean;\n }>;\n skipped: Array<{ name: string; reason: string }>;\n errors: Array<{ name: string; error: string }>;\n averageConfidence: number;\n enrichmentCost?: number;\n}\n\ninterface ComponentData {\n component: DiscoveredComponent;\n /** Full metadata from ComponentExtractor (props, composition, description) */\n meta: ComponentMeta | null;\n storyVariants: StoryVariant[];\n}\n\ninterface StoryVariant {\n name: string;\n args: Record<string, unknown>;\n}\n\ninterface FieldConfidence {\n score: number;\n todoFields: string[];\n}\n\nexport interface EnrichmentResult {\n when: string[];\n whenNot: string[];\n guidelines: string[];\n a11yRules: string[];\n scenarioTags: string[];\n tags: string[];\n}\n\n// ---------------------------------------------------------------------------\n// Main orchestrator\n// ---------------------------------------------------------------------------\n\nexport async function scanGenerate(\n options: ScanGenerateOptions\n): Promise<ScanGenerateResult> {\n const scanPath = resolve(options.scanPath);\n const generated: ScanGenerateResult[\"generated\"] = [];\n const skipped: ScanGenerateResult[\"skipped\"] = [];\n const errors: ScanGenerateResult[\"errors\"] = [];\n\n console.log(pc.cyan(`\\n${BRAND.name} Scan → Generate\\n`));\n console.log(pc.dim(`Scanning: ${scanPath}\\n`));\n\n // Phase 1: Discover components\n console.log(pc.dim(\"Phase 1: Discovering components...\"));\n\n // Use broader patterns than default — the scan path IS the component root,\n // not a project root with src/components/ inside it\n const defaultScanPatterns = [\n \"**/*.tsx\",\n \"**/*.ts\",\n ];\n\n const components = await discoverAllComponents(scanPath, {\n patterns: options.patterns || defaultScanPatterns,\n exclude: [\n \"**/*.test.*\",\n \"**/*.spec.*\",\n \"**/*.stories.*\",\n \"**/*.fragment.*\",\n \"**/*.d.ts\",\n \"**/__tests__/**\",\n \"**/__mocks__/**\",\n \"**/node_modules/**\",\n \"**/dist/**\",\n ],\n });\n\n if (components.length === 0) {\n console.log(\n pc.yellow(\"No components found. Check the path or file patterns.\")\n );\n return {\n success: false,\n generated: [],\n skipped: [],\n errors: [{ name: \"*\", error: \"No components found\" }],\n averageConfidence: 0,\n };\n }\n\n console.log(pc.green(` Found ${components.length} components`));\n\n // Phase 2: Extract data for each component (persistent LanguageService)\n console.log(pc.dim(\"\\nPhase 2: Extracting component metadata...\"));\n\n // Create a single extractor that reuses its LanguageService across all components\n const extractor = createComponentExtractor(options.tsconfig);\n const componentDataList: ComponentData[] = [];\n\n const extractionStart = performance.now();\n\n for (const comp of components) {\n let meta: ComponentMeta | null = null;\n try {\n meta = extractor.extract(comp.sourcePath, comp.name);\n } catch {\n // Extraction can fail for complex types — continue gracefully\n }\n\n // Fallback compound detection for prefix-based patterns (e.g., shadcn-style\n // components that export CardHeader, CardFooter etc. as separate named exports\n // rather than using Object.assign)\n if (meta && !meta.composition) {\n try {\n const subComponents = await detectCompoundComponents(comp.sourcePath, comp.name);\n if (subComponents.length > 0) {\n meta = {\n ...meta,\n composition: {\n pattern: 'compound',\n parts: subComponents.map(name => ({ name, props: {} })),\n required: [],\n },\n };\n }\n } catch {\n // Non-fatal\n }\n }\n\n let storyVariants: StoryVariant[] = [];\n if (!options.skipStorybook && comp.storyPath) {\n try {\n storyVariants = await extractStoryVariantsFromFile(comp.storyPath);\n } catch {\n // Story extraction failure is non-fatal\n }\n }\n\n componentDataList.push({\n component: comp,\n meta,\n storyVariants,\n });\n }\n\n extractor.dispose();\n\n const extractionMs = (performance.now() - extractionStart).toFixed(0);\n const propsExtracted = componentDataList.filter(\n (d) => d.meta && Object.keys(d.meta.props).length > 0\n ).length;\n const compoundCount = componentDataList.filter(\n (d) => d.meta?.composition !== null\n ).length;\n console.log(pc.green(` Extracted props for ${propsExtracted} components (${extractionMs}ms)`));\n if (compoundCount > 0) {\n console.log(pc.green(` Detected ${compoundCount} compound component(s) with sub-component props`));\n }\n\n // Phase 3: Enrich with LLM (if --enrich)\n let enrichments = new Map<string, EnrichmentResult>();\n let enrichmentCost: number | undefined;\n\n if (options.enrich) {\n console.log(pc.dim(\"\\nPhase 3: Enriching with AI...\"));\n\n const enrichResult = await enrichComponents(componentDataList, {\n dryRun: options.dryRun,\n provider: options.provider,\n apiKey: options.apiKey,\n model: options.model,\n });\n\n enrichments = enrichResult.enrichments;\n\n if (enrichResult.model && enrichResult.totalInputTokens > 0) {\n const { calculateCost } = await import('../ai-client.js');\n enrichmentCost = calculateCost(\n enrichResult.model,\n enrichResult.totalInputTokens,\n enrichResult.totalOutputTokens\n );\n }\n }\n\n // Phase 4: Generate fragment files\n console.log(pc.dim(`\\nPhase ${options.enrich ? '4' : '3'}: Generating fragment files...`));\n\n for (const data of componentDataList) {\n const comp = data.component;\n const componentDir = dirname(comp.sourcePath);\n const componentBaseName = basename(comp.sourcePath, \".tsx\");\n\n // Determine output path\n let fragmentDir: string;\n if (options.outputDir) {\n fragmentDir = resolve(options.outputDir, comp.name);\n await mkdir(fragmentDir, { recursive: true });\n } else {\n fragmentDir = componentDir;\n }\n\n const fragmentPath = join(\n fragmentDir,\n `${componentBaseName}${BRAND.fileExtension}`\n );\n\n // Check if fragment already exists\n let fragmentExists = false;\n try {\n await access(fragmentPath);\n fragmentExists = true;\n } catch {\n // Doesn't exist\n }\n\n if (fragmentExists && !options.force) {\n skipped.push({ name: comp.name, reason: \"Fragment already exists\" });\n if (options.verbose) {\n console.log(pc.dim(` Skipping ${comp.name} (fragment exists)`));\n }\n continue;\n }\n\n try {\n // Look up enrichment for this component\n const enrichment = enrichments.get(comp.name);\n\n // Calculate confidence (enrichment boosts score)\n const confidence = calculateFieldConfidence(data, enrichment);\n\n // Compute import path\n const importPath = computeImportPath(\n fragmentDir,\n comp.sourcePath,\n componentBaseName\n );\n\n // Generate the fragment file\n const content = generateFragmentWithTodos(\n comp.name,\n importPath,\n data,\n confidence,\n enrichment\n );\n\n await writeFile(fragmentPath, content, \"utf-8\");\n\n const relPath = relative(process.cwd(), fragmentPath);\n generated.push({\n name: comp.name,\n path: relPath,\n confidence: confidence.score,\n todoCount: confidence.todoFields.length,\n enriched: !!enrichment,\n });\n\n const confColor =\n confidence.score >= 70\n ? pc.green\n : confidence.score >= 40\n ? pc.yellow\n : pc.red;\n console.log(\n pc.green(` ✓ ${comp.name}`) +\n pc.dim(` (confidence: `) +\n confColor(`${confidence.score}`) +\n pc.dim(`, TODOs: ${confidence.todoFields.length})`)\n );\n } catch (e) {\n errors.push({\n name: comp.name,\n error: e instanceof Error ? e.message : String(e),\n });\n console.log(pc.red(` ✗ ${comp.name}: ${e instanceof Error ? e.message : String(e)}`));\n }\n }\n\n // Summary\n const avgConfidence =\n generated.length > 0\n ? Math.round(\n generated.reduce((sum, g) => sum + g.confidence, 0) /\n generated.length\n )\n : 0;\n\n console.log(pc.dim(\"\\n────────────────────────────────────────\"));\n console.log(pc.green(`\\n✓ Generated ${generated.length} fragment file(s)`));\n\n if (skipped.length > 0) {\n console.log(\n pc.dim(` Skipped ${skipped.length} (use --force to overwrite)`)\n );\n }\n if (errors.length > 0) {\n console.log(pc.yellow(` ${errors.length} error(s)`));\n }\n\n console.log(pc.dim(` Average confidence: ${avgConfidence}/100`));\n\n if (options.enrich) {\n const enrichedCount = generated.filter(g => g.enriched).length;\n console.log(pc.dim(` Enriched: ${enrichedCount}/${generated.length} components`));\n if (enrichmentCost !== undefined && enrichmentCost > 0) {\n console.log(pc.dim(` Estimated cost: $${enrichmentCost.toFixed(4)}`));\n }\n }\n\n const totalTodos = generated.reduce((sum, g) => sum + g.todoCount, 0);\n if (totalTodos > 0) {\n console.log(\n pc.dim(` Total TODOs: ${totalTodos}`) +\n pc.dim(` — search for \"TODO:\" in generated files`)\n );\n }\n\n console.log();\n\n return {\n success: errors.length === 0,\n generated,\n skipped,\n errors,\n averageConfidence: avgConfidence,\n enrichmentCost,\n };\n}\n\n// ---------------------------------------------------------------------------\n// JSDoc Extraction\n// ---------------------------------------------------------------------------\n\n/**\n * Extract the JSDoc description from a component's exported declaration.\n * Uses TypeScript AST — no program needed, just source file parsing.\n */\nexport async function extractComponentJSDoc(\n filePath: string,\n componentName?: string\n): Promise<string | null> {\n const content = await readFile(filePath, \"utf-8\");\n return extractComponentJSDocFromSource(content, filePath, componentName);\n}\n\nexport function extractComponentJSDocFromSource(\n source: string,\n filePath: string,\n componentName?: string\n): string | null {\n const sourceFile = ts.createSourceFile(\n filePath,\n source,\n ts.ScriptTarget.ESNext,\n true,\n filePath.endsWith(\".tsx\") ? ts.ScriptKind.TSX : ts.ScriptKind.TS\n );\n\n const targetName =\n componentName || basename(filePath).replace(/\\.(tsx?|jsx?)$/, \"\");\n\n let componentDoc: string | null = null;\n let propsInterfaceDoc: string | null = null;\n\n for (const statement of sourceFile.statements) {\n // export function ComponentName(...)\n if (\n ts.isFunctionDeclaration(statement) &&\n statement.name?.text === targetName &&\n hasExportModifier(statement)\n ) {\n const doc = getLeadingJSDoc(statement, sourceFile);\n if (doc) componentDoc = doc;\n }\n\n // export const ComponentName = ...\n if (\n ts.isVariableStatement(statement) &&\n hasExportModifier(statement)\n ) {\n for (const decl of statement.declarationList.declarations) {\n if (ts.isIdentifier(decl.name) && decl.name.text === targetName) {\n const doc = getLeadingJSDoc(statement, sourceFile);\n if (doc) componentDoc = doc;\n }\n }\n }\n\n // export default function ...\n if (\n ts.isFunctionDeclaration(statement) &&\n hasDefaultExportModifier(statement)\n ) {\n const doc = getLeadingJSDoc(statement, sourceFile);\n if (doc) componentDoc = doc;\n }\n\n // Fallback: JSDoc on the {Name}Props interface\n if (\n ts.isInterfaceDeclaration(statement) &&\n (statement.name.text === `${targetName}Props` ||\n statement.name.text === `${targetName}Properties`)\n ) {\n const doc = getLeadingJSDoc(statement, sourceFile);\n if (doc) propsInterfaceDoc = doc;\n }\n }\n\n // Prefer component-level JSDoc; fall back to props interface JSDoc\n return componentDoc || propsInterfaceDoc;\n}\n\nfunction hasExportModifier(node: ts.Node): boolean {\n const modifiers = ts.canHaveModifiers(node)\n ? ts.getModifiers(node)\n : undefined;\n return modifiers?.some((m) => m.kind === ts.SyntaxKind.ExportKeyword) ?? false;\n}\n\nfunction hasDefaultExportModifier(node: ts.Node): boolean {\n const modifiers = ts.canHaveModifiers(node)\n ? ts.getModifiers(node)\n : undefined;\n if (!modifiers) return false;\n return (\n modifiers.some((m) => m.kind === ts.SyntaxKind.ExportKeyword) &&\n modifiers.some((m) => m.kind === ts.SyntaxKind.DefaultKeyword)\n );\n}\n\nfunction getLeadingJSDoc(\n node: ts.Node,\n sourceFile: ts.SourceFile\n): string | null {\n const fullText = sourceFile.getFullText();\n const nodeStart = node.getFullStart();\n const nodePos = node.getStart(sourceFile);\n const leadingText = fullText.slice(nodeStart, nodePos);\n\n const jsDocMatch = leadingText.match(/\\/\\*\\*([\\s\\S]*?)\\*\\//);\n if (!jsDocMatch) return null;\n\n // Parse JSDoc content — extract description lines, skip tags\n const lines = jsDocMatch[1].split(\"\\n\");\n const descriptionLines: string[] = [];\n\n for (const line of lines) {\n const trimmed = line.replace(/^\\s*\\*?\\s?/, \"\").trim();\n if (trimmed.startsWith(\"@\")) break; // Stop at first tag\n if (trimmed) descriptionLines.push(trimmed);\n }\n\n return descriptionLines.length > 0\n ? descriptionLines.join(\" \")\n : null;\n}\n\n// ---------------------------------------------------------------------------\n// Compound Component Detection\n// ---------------------------------------------------------------------------\n\n/**\n * Detect compound component patterns (Object.assign) in a source file.\n * Returns the names of sub-components found.\n */\nexport async function detectCompoundComponents(\n filePath: string,\n primaryComponentName?: string\n): Promise<string[]> {\n const content = await readFile(filePath, \"utf-8\");\n return detectCompoundComponentsFromSource(content, filePath, primaryComponentName);\n}\n\nexport function detectCompoundComponentsFromSource(\n source: string,\n filePath: string,\n primaryComponentName?: string\n): string[] {\n const sourceFile = ts.createSourceFile(\n filePath,\n source,\n ts.ScriptTarget.ESNext,\n true,\n filePath.endsWith(\".tsx\") ? ts.ScriptKind.TSX : ts.ScriptKind.TS\n );\n\n const subComponents: string[] = [];\n\n function visit(node: ts.Node) {\n // Look for Object.assign(Root, { Sub1, Sub2 }) or Object.assign(Root, { Sub1: SubComponent })\n if (\n ts.isCallExpression(node) &&\n ts.isPropertyAccessExpression(node.expression) &&\n ts.isIdentifier(node.expression.expression) &&\n node.expression.expression.text === \"Object\" &&\n node.expression.name.text === \"assign\" &&\n node.arguments.length >= 2\n ) {\n const secondArg = node.arguments[1];\n if (ts.isObjectLiteralExpression(secondArg)) {\n for (const prop of secondArg.properties) {\n if (ts.isShorthandPropertyAssignment(prop)) {\n subComponents.push(prop.name.text);\n } else if (\n ts.isPropertyAssignment(prop) &&\n ts.isIdentifier(prop.name)\n ) {\n subComponents.push(prop.name.text);\n }\n }\n }\n }\n\n ts.forEachChild(node, visit);\n }\n\n ts.forEachChild(sourceFile, visit);\n\n // Fallback: detect shadcn-style compound exports.\n // Libraries like shadcn/ui export compound sub-components as separate named exports\n // that share a prefix: export { Card, CardHeader, CardContent, CardFooter }\n // Detect these by finding PascalCase exports that start with the primary component name.\n if (subComponents.length === 0 && primaryComponentName) {\n const exportedNames: string[] = [];\n\n // Find export { Name1, Name2, ... } blocks\n for (const statement of sourceFile.statements) {\n if (ts.isExportDeclaration(statement) && statement.exportClause) {\n if (ts.isNamedExports(statement.exportClause)) {\n for (const element of statement.exportClause.elements) {\n exportedNames.push(element.name.text);\n }\n }\n }\n // Also catch `export function CardHeader` / `export const CardHeader`\n if (ts.isFunctionDeclaration(statement) && statement.name) {\n const modifiers = ts.canHaveModifiers(statement) ? ts.getModifiers(statement) : undefined;\n if (modifiers?.some((m) => m.kind === ts.SyntaxKind.ExportKeyword)) {\n exportedNames.push(statement.name.text);\n }\n }\n if (ts.isVariableStatement(statement)) {\n const modifiers = ts.canHaveModifiers(statement) ? ts.getModifiers(statement) : undefined;\n if (modifiers?.some((m) => m.kind === ts.SyntaxKind.ExportKeyword)) {\n for (const decl of statement.declarationList.declarations) {\n if (ts.isIdentifier(decl.name)) {\n exportedNames.push(decl.name.text);\n }\n }\n }\n }\n }\n\n // Find exports that start with primaryComponentName but aren't the primary itself\n // e.g., \"Card\" → [\"CardHeader\", \"CardFooter\", \"CardContent\", \"CardTitle\"]\n const prefix = primaryComponentName;\n for (const name of exportedNames) {\n if (\n name !== prefix &&\n name.startsWith(prefix) &&\n /^[A-Z]/.test(name.slice(prefix.length))\n ) {\n // Strip the prefix to get the sub-component name\n const subName = name.slice(prefix.length);\n subComponents.push(subName);\n }\n }\n }\n\n return subComponents;\n}\n\n// ---------------------------------------------------------------------------\n// Story Variant Extraction (lightweight, from generate.ts patterns)\n// ---------------------------------------------------------------------------\n\nasync function extractStoryVariantsFromFile(\n storyPath: string\n): Promise<StoryVariant[]> {\n const content = await readFile(storyPath, \"utf-8\");\n const variants: StoryVariant[] = [];\n\n const exportMatches = content.matchAll(\n /export\\s+const\\s+([A-Z][a-zA-Z0-9]*)\\s*[=:]/g\n );\n\n for (const match of exportMatches) {\n const name = match[1];\n if (name === \"default\" || name.endsWith(\"Args\") || name.endsWith(\"Meta\")) {\n continue;\n }\n\n const args = extractStoryArgs(content, name);\n variants.push({ name, args });\n }\n\n return variants;\n}\n\nfunction extractStoryArgs(\n content: string,\n storyName: string\n): Record<string, unknown> {\n const storyPattern = new RegExp(\n `export\\\\s+const\\\\s+${storyName}[^=]*=\\\\s*\\\\{([\\\\s\\\\S]*?)\\\\n\\\\};`\n );\n const storyMatch = content.match(storyPattern);\n if (!storyMatch) return {};\n\n const storyBody = storyMatch[1];\n const argsStart = storyBody.indexOf(\"args:\");\n if (argsStart === -1) return {};\n\n const braceStart = storyBody.indexOf(\"{\", argsStart);\n if (braceStart === -1) return {};\n\n let depth = 0;\n let braceEnd = -1;\n for (let i = braceStart; i < storyBody.length; i++) {\n if (storyBody[i] === \"{\") depth++;\n else if (storyBody[i] === \"}\") {\n depth--;\n if (depth === 0) {\n braceEnd = i;\n break;\n }\n }\n }\n if (braceEnd === -1) return {};\n\n const argsBlock = storyBody.slice(braceStart + 1, braceEnd).trim();\n return parseArgsBlock(argsBlock);\n}\n\nfunction parseArgsBlock(argsBlock: string): Record<string, unknown> {\n const args: Record<string, unknown> = {};\n const pairPattern =\n /(\\w+)\\s*:\\s*(?:['\"]([^'\"]*?)['\"]|(true|false)|(\\d+(?:\\.\\d+)?))/g;\n let pairMatch: RegExpExecArray | null;\n\n while ((pairMatch = pairPattern.exec(argsBlock)) !== null) {\n const key = pairMatch[1];\n if (pairMatch[2] !== undefined) {\n args[key] = pairMatch[2];\n } else if (pairMatch[3] !== undefined) {\n args[key] = pairMatch[3] === \"true\";\n } else if (pairMatch[4] !== undefined) {\n args[key] = Number(pairMatch[4]);\n }\n }\n\n return args;\n}\n\n// ---------------------------------------------------------------------------\n// Enrichment: LLM-powered knowledge field generation\n// ---------------------------------------------------------------------------\n\nexport function buildEnrichmentSystemPrompt(): string {\n return `You are a senior frontend architect and design system expert.\nGiven a component's metadata (name, description, props, composition), generate knowledge fields that help AI models use the component correctly.\n\nRespond ONLY with a JSON object. No explanation, no markdown outside the JSON.\n\nThe JSON must have exactly these fields:\n{\n \"when\": [\"...\"], // 3-5 scenarios when this component should be used\n \"whenNot\": [\"...\"], // 2-4 scenarios when NOT to use this component (suggest alternatives)\n \"guidelines\": [\"...\"], // 2-3 usage guidelines or best practices\n \"a11yRules\": [\"...\"], // 2-4 accessibility rules or requirements\n \"scenarioTags\": [\"...\"], // 3-5 dot-notation scenario tags (e.g., \"form.input.text\", \"layout.container.card\")\n \"tags\": [\"...\"] // 3-5 search keywords for component discovery\n}\n\nRules:\n- \"when\" items should describe concrete UI scenarios (e.g., \"Displaying grouped content with a header, body, and footer\")\n- \"whenNot\" items should name a better alternative (e.g., \"Simple text grouping without visual separation — use a div or Stack instead\")\n- \"guidelines\" should be actionable best practices\n- \"a11yRules\" should reference WCAG or ARIA patterns where relevant\n- \"scenarioTags\" use dot notation: category.subcategory.detail\n- \"tags\" should be lowercase single words or short phrases for search\n- Keep each item to 1 sentence. Be specific, not generic.`;\n}\n\nexport function buildEnrichmentUserPrompt(\n name: string,\n data: ComponentData\n): string {\n const props = data.meta?.props ?? {};\n const localProps = Object.entries(props).filter(([_, p]) => p.source === 'local');\n const composition = data.meta?.composition ?? null;\n const description = data.meta?.description || '';\n const category = inferCategoryFromMeta(\n name,\n Object.fromEntries(localProps)\n );\n\n const lines: string[] = [\n `Component: ${name}`,\n `Category: ${category}`,\n ];\n\n if (description) {\n lines.push(`Description: ${description}`);\n }\n\n if (localProps.length > 0) {\n lines.push('');\n lines.push('Props:');\n for (const [propName, prop] of localProps) {\n let propLine = ` - ${propName}: ${prop.typeKind}`;\n if (prop.values && prop.values.length > 0) {\n propLine += ` (${prop.values.join(' | ')})`;\n }\n if (prop.required) propLine += ' [required]';\n if (prop.description) propLine += ` — ${prop.description}`;\n lines.push(propLine);\n }\n }\n\n if (composition && composition.parts.length > 0) {\n lines.push('');\n lines.push(`Composition: ${composition.pattern} pattern`);\n lines.push(`Sub-components: ${composition.parts.map(p => p.name).join(', ')}`);\n }\n\n if (data.storyVariants.length > 0) {\n lines.push('');\n lines.push(`Known variants: ${data.storyVariants.map(v => v.name).join(', ')}`);\n }\n\n return lines.join('\\n');\n}\n\nexport function parseEnrichmentResponse(text: string): EnrichmentResult {\n const empty: EnrichmentResult = {\n when: [],\n whenNot: [],\n guidelines: [],\n a11yRules: [],\n scenarioTags: [],\n tags: [],\n };\n\n try {\n // Parse JSON from response, handling ```json fences\n const jsonMatch = text.match(/```json\\n?([\\s\\S]*?)\\n?```/) || text.match(/\\{[\\s\\S]*\\}/);\n const jsonStr = jsonMatch ? (jsonMatch[1] || jsonMatch[0]) : text;\n const parsed = JSON.parse(jsonStr) as Record<string, unknown>;\n\n const getArray = (key: string, maxLen: number): string[] => {\n const val = parsed[key];\n if (!Array.isArray(val)) return [];\n return val\n .filter((item): item is string => typeof item === 'string')\n .slice(0, maxLen);\n };\n\n return {\n when: getArray('when', 5),\n whenNot: getArray('whenNot', 4),\n guidelines: getArray('guidelines', 3),\n a11yRules: getArray('a11yRules', 4),\n scenarioTags: getArray('scenarioTags', 5),\n tags: getArray('tags', 5),\n };\n } catch {\n return empty;\n }\n}\n\nasync function enrichComponents(\n componentDataList: ComponentData[],\n options: Pick<ScanGenerateOptions, 'dryRun' | 'provider' | 'apiKey' | 'model'>\n): Promise<{\n enrichments: Map<string, EnrichmentResult>;\n totalInputTokens: number;\n totalOutputTokens: number;\n model: string;\n}> {\n const enrichments = new Map<string, EnrichmentResult>();\n let totalInputTokens = 0;\n let totalOutputTokens = 0;\n\n const {\n detectProvider,\n getApiKey,\n createAIClient,\n generateCompletion,\n ENRICHMENT_MODELS,\n } = await import('../ai-client.js');\n\n const provider = detectProvider({ provider: options.provider, apiKey: options.apiKey });\n if (provider === 'none') {\n console.log(pc.yellow(' No API key found. Set ANTHROPIC_API_KEY or OPENAI_API_KEY, or use --api-key'));\n console.log(pc.yellow(' Skipping enrichment — fragment files will have TODO markers instead.\\n'));\n return { enrichments, totalInputTokens: 0, totalOutputTokens: 0, model: '' };\n }\n\n const apiKey = getApiKey(provider, options.apiKey);\n if (!apiKey) {\n console.log(pc.yellow(' API key not found for provider: ' + provider));\n console.log(pc.yellow(' Skipping enrichment.\\n'));\n return { enrichments, totalInputTokens: 0, totalOutputTokens: 0, model: '' };\n }\n\n const model = options.model || ENRICHMENT_MODELS[provider];\n\n if (options.dryRun) {\n for (const data of componentDataList) {\n console.log(pc.dim(` Would enrich: ${data.component.name}`));\n }\n return { enrichments, totalInputTokens: 0, totalOutputTokens: 0, model };\n }\n\n const client = await createAIClient(provider, apiKey);\n const systemPrompt = buildEnrichmentSystemPrompt();\n\n for (const data of componentDataList) {\n const name = data.component.name;\n process.stdout.write(pc.dim(` Enriching ${name}...`));\n\n try {\n const userPrompt = buildEnrichmentUserPrompt(name, data);\n const result = await generateCompletion(client, provider, model, systemPrompt, userPrompt, 512);\n\n totalInputTokens += result.inputTokens;\n totalOutputTokens += result.outputTokens;\n\n const enrichment = parseEnrichmentResponse(result.text);\n const fieldCount =\n enrichment.when.length +\n enrichment.whenNot.length +\n enrichment.guidelines.length +\n enrichment.a11yRules.length +\n enrichment.scenarioTags.length +\n enrichment.tags.length;\n\n enrichments.set(name, enrichment);\n process.stdout.write(`\\r ${pc.green('✓')} ${name} (${fieldCount} fields)\\n`);\n } catch (e) {\n const msg = e instanceof Error ? e.message : String(e);\n process.stdout.write(`\\r ${pc.yellow('!')} ${name}: ${msg}\\n`);\n }\n }\n\n return { enrichments, totalInputTokens, totalOutputTokens, model };\n}\n\nexport function buildEnrichedUsageBlock(enrichment: EnrichmentResult): string {\n const lines: string[] = [];\n lines.push(' usage: {');\n\n if (enrichment.when.length > 0) {\n lines.push(' when: [');\n for (const item of enrichment.when) {\n lines.push(` '${escapeQuotes(item)}',`);\n }\n lines.push(' ],');\n } else {\n lines.push(' when: [],');\n }\n\n if (enrichment.whenNot.length > 0) {\n lines.push(' whenNot: [');\n for (const item of enrichment.whenNot) {\n lines.push(` '${escapeQuotes(item)}',`);\n }\n lines.push(' ],');\n } else {\n lines.push(' whenNot: [],');\n }\n\n if (enrichment.guidelines.length > 0) {\n lines.push(' guidelines: [');\n for (const item of enrichment.guidelines) {\n lines.push(` '${escapeQuotes(item)}',`);\n }\n lines.push(' ],');\n }\n\n lines.push(' },');\n return lines.join('\\n');\n}\n\n// ---------------------------------------------------------------------------\n// Confidence Scoring\n// ---------------------------------------------------------------------------\n\nexport function calculateFieldConfidence(\n data: ComponentData,\n enrichment?: EnrichmentResult\n): FieldConfidence {\n let score = 0;\n const todoFields: string[] = [];\n\n const props = data.meta?.props ?? {};\n // Use only local props — inherited HTML/React props inflate confidence\n // without adding value to the generated fragment output\n const localEntries = Object.values(props).filter(p => p.source === 'local');\n const hasProps = localEntries.length > 0;\n\n // Props extracted: +30\n if (hasProps) {\n score += 30;\n }\n\n // JSDoc description found: +15\n if (data.meta?.description) {\n score += 15;\n } else {\n todoFields.push(\"meta.description\");\n }\n\n // Category inferred from path/name (not fallback): +10\n const localProps = Object.fromEntries(\n Object.entries(props).filter(([_, p]) => p.source === 'local')\n );\n const category = inferCategoryFromMeta(data.component.name, localProps);\n if (category !== \"Components\") {\n score += 10;\n } else {\n todoFields.push(\"meta.category\");\n }\n\n // Story variants found: +25\n if (data.storyVariants.length > 0) {\n score += 25;\n }\n\n // All prop types resolved (no \"custom\"): +10\n if (hasProps) {\n const allResolved = localEntries.every((p) => p.typeKind !== \"custom\");\n if (allResolved) {\n score += 10;\n }\n }\n\n // Compound component detected with sub-component props: +10\n if (data.meta?.composition && data.meta.composition.parts.length > 0) {\n score += 10;\n }\n\n // Has default values: +5\n if (hasProps) {\n const hasDefaults = localEntries.some((p) => p.default !== undefined);\n if (hasDefaults) {\n score += 5;\n }\n }\n\n // Enrichment bonuses — if LLM filled these, they're no longer TODOs\n if (enrichment && enrichment.when.length > 0) {\n score += 10;\n } else {\n todoFields.push(\"usage.when\");\n }\n\n if (enrichment && enrichment.whenNot.length > 0) {\n score += 5;\n } else {\n todoFields.push(\"usage.whenNot\");\n }\n\n if (enrichment && enrichment.guidelines.length > 0) {\n score += 5;\n }\n\n if (enrichment && enrichment.tags.length > 0) {\n score += 5;\n }\n\n return { score: Math.min(score, 100), todoFields };\n}\n\n// ---------------------------------------------------------------------------\n// Category / Status / Description inference (adapted from generate.ts + scan.ts)\n// ---------------------------------------------------------------------------\n\nconst CATEGORY_PATTERNS: Record<string, string[]> = {\n Actions: [\"button\", \"action\", \"cta\", \"fab\", \"floatingaction\"],\n Forms: [\n \"form\", \"input\", \"select\", \"checkbox\", \"radio\", \"textarea\", \"field\",\n \"textfield\", \"datepicker\", \"switch\", \"slider\", \"segmented\",\n ],\n Layout: [\n \"layout\", \"container\", \"grid\", \"flex\", \"stack\", \"box\", \"divider\", \"spacer\",\n \"sidebar\",\n ],\n Navigation: [\n \"nav\", \"menu\", \"breadcrumb\", \"tab\", \"link\", \"pagination\", \"stepper\",\n \"topbar\",\n ],\n Feedback: [\n \"alert\", \"toast\", \"notification\", \"message\", \"badge\", \"indicator\",\n \"progress\", \"spinner\", \"loading\", \"loader\", \"lozenge\", \"chip\",\n ],\n \"Data Display\": [\n \"table\", \"list\", \"card\", \"avatar\", \"stat\", \"timeline\", \"tree\", \"datalist\",\n \"datacard\",\n ],\n Overlays: [\n \"modal\", \"dialog\", \"drawer\", \"popover\", \"tooltip\", \"dropdown\",\n \"slidepanel\",\n ],\n Typography: [\"text\", \"heading\", \"title\", \"label\", \"paragraph\"],\n Media: [\"image\", \"video\", \"icon\", \"carousel\"],\n};\n\nfunction inferCategoryFromMeta(\n componentName: string,\n props: Record<string, PropMeta>\n): string {\n const lower = componentName.toLowerCase();\n\n for (const [category, patterns] of Object.entries(CATEGORY_PATTERNS)) {\n for (const pattern of patterns) {\n if (lower.includes(pattern)) {\n return category;\n }\n }\n }\n\n // Prop-based fallbacks\n const propNames = new Set(Object.keys(props));\n if (propNames.has(\"onClick\") || propNames.has(\"onPress\")) return \"Actions\";\n if (propNames.has(\"value\") || propNames.has(\"defaultValue\")) return \"Forms\";\n if (propNames.has(\"children\")) return \"Layout\";\n\n return \"Components\";\n}\n\nfunction inferStatus(\n filePath: string\n): \"draft\" | \"experimental\" | \"beta\" | \"stable\" | \"deprecated\" {\n const lowerPath = filePath.toLowerCase();\n if (lowerPath.includes(\"/experimental/\") || lowerPath.includes(\"/labs/\"))\n return \"experimental\";\n if (lowerPath.includes(\"/beta/\")) return \"beta\";\n if (lowerPath.includes(\"/deprecated/\") || lowerPath.includes(\"/legacy/\"))\n return \"deprecated\";\n if (lowerPath.includes(\"/draft/\") || lowerPath.includes(\"/wip/\"))\n return \"draft\";\n return \"stable\";\n}\n\nfunction inferDescriptionFromMeta(\n componentName: string,\n props: Record<string, PropMeta>\n): string {\n const words = componentName\n .replace(/([A-Z])/g, \" $1\")\n .trim()\n .toLowerCase();\n\n const propNames = new Set(Object.keys(props));\n const hasOnClick =\n propNames.has(\"onClick\") || propNames.has(\"onPress\");\n const hasValue =\n propNames.has(\"value\") || propNames.has(\"defaultValue\");\n const hasChildren = propNames.has(\"children\");\n\n if (hasOnClick && !hasValue)\n return `Interactive ${words} element for triggering actions`;\n if (hasValue) return `Form ${words} for user input`;\n if (hasChildren) return `Container ${words} for grouping content`;\n return `${words.charAt(0).toUpperCase() + words.slice(1)} component`;\n}\n\nfunction inferAccessibilityFromMeta(props: Record<string, PropMeta>): {\n role?: string;\n requirements?: string[];\n} {\n const propNames = new Set(Object.keys(props));\n const accessibility: { role?: string; requirements?: string[] } = {};\n\n const hasOnClick =\n propNames.has(\"onClick\") || propNames.has(\"onPress\");\n const hasAriaLabel =\n propNames.has(\"ariaLabel\") || propNames.has(\"aria-label\");\n const hasDisabled = propNames.has(\"disabled\");\n const hasHref = propNames.has(\"href\");\n\n if (hasOnClick && !hasHref) accessibility.role = \"button\";\n else if (hasHref) accessibility.role = \"link\";\n\n const requirements: string[] = [];\n if (hasOnClick && !hasAriaLabel)\n requirements.push(\"Should have visible text or aria-label\");\n if (hasDisabled)\n requirements.push(\n \"Disabled state should be conveyed to assistive technology\"\n );\n if (requirements.length > 0) accessibility.requirements = requirements;\n\n return accessibility;\n}\n\n// ---------------------------------------------------------------------------\n// Contract Block Generation\n// ---------------------------------------------------------------------------\n\ninterface ContractBlock {\n propsSummary: string[];\n compoundChildren?: Record<string, {\n required?: boolean;\n accepts?: string[];\n description?: string;\n }>;\n canonicalUsage?: string[];\n a11yRules?: string[];\n scenarioTags?: string[];\n}\n\nfunction buildContractBlock(\n componentName: string,\n props: Record<string, PropMeta>,\n composition: CompositionMeta | null,\n accessibility: { role?: string; requirements?: string[] }\n): ContractBlock | null {\n const localEntries = Object.entries(props).filter(([_, p]) => p.source === 'local');\n if (localEntries.length === 0 && !composition && !accessibility.requirements?.length) {\n return null;\n }\n\n const contract: ContractBlock = { propsSummary: [] };\n\n // propsSummary: \"propName: value1 | value2 | value3\" for enums, \"propName: type\" otherwise\n for (const [name, prop] of localEntries) {\n let summary = name + ': ';\n if (prop.typeKind === 'enum' && prop.values && prop.values.length > 0) {\n summary += prop.values.join(' | ');\n } else {\n summary += prop.typeKind;\n }\n if (prop.required) summary += ' (required)';\n if (prop.default !== undefined) summary += ` (default: ${prop.default})`;\n contract.propsSummary.push(summary);\n }\n\n // compoundChildren: built from CompositionMeta.parts[]\n if (composition && composition.parts.length > 0) {\n const children: Record<string, { required?: boolean; accepts?: string[]; description?: string }> = {};\n for (const part of composition.parts) {\n const childEntry: { required?: boolean; accepts?: string[]; description?: string } = {};\n if (composition.required.includes(part.name)) {\n childEntry.required = true;\n }\n // Infer accepts from ReactNode-type props on the sub-component\n const nodeProps = Object.entries(part.props)\n .filter(([_, p]) => p.typeKind === 'node' || p.typeKind === 'element')\n .map(([n]) => n);\n if (nodeProps.length > 0) {\n childEntry.accepts = nodeProps;\n }\n children[part.name] = childEntry;\n }\n contract.compoundChildren = children;\n }\n\n // canonicalUsage: generate 1 JSX snippet using actual component + sub-component names\n if (composition && composition.parts.length > 0) {\n const innerLines = composition.parts\n .map(part => ` <${componentName}.${part.name}>...</${componentName}.${part.name}>`)\n .join('\\n');\n contract.canonicalUsage = [\n `<${componentName}>\\n${innerLines}\\n</${componentName}>`,\n ];\n }\n\n // a11yRules: from accessibility inference\n const rules: string[] = [];\n if (accessibility.role) {\n rules.push(`Role: ${accessibility.role}`);\n }\n if (accessibility.requirements) {\n rules.push(...accessibility.requirements);\n }\n if (rules.length > 0) {\n contract.a11yRules = rules;\n }\n\n return contract;\n}\n\nfunction emitContractBlock(contract: ContractBlock): string {\n const lines: string[] = [];\n lines.push('\\n contract: {');\n\n // propsSummary\n if (contract.propsSummary.length > 0) {\n lines.push(' propsSummary: [');\n for (const s of contract.propsSummary) {\n lines.push(` '${escapeQuotes(s)}',`);\n }\n lines.push(' ],');\n }\n\n // compoundChildren\n if (contract.compoundChildren && Object.keys(contract.compoundChildren).length > 0) {\n lines.push(' compoundChildren: {');\n for (const [name, meta] of Object.entries(contract.compoundChildren)) {\n const fields: string[] = [];\n if (meta.required) fields.push(`required: true`);\n if (meta.accepts && meta.accepts.length > 0) {\n fields.push(`accepts: [${meta.accepts.map(a => `'${a}'`).join(', ')}]`);\n }\n if (meta.description) fields.push(`description: '${escapeQuotes(meta.description)}'`);\n lines.push(` ${name}: { ${fields.join(', ')} },`);\n }\n lines.push(' },');\n }\n\n // canonicalUsage\n if (contract.canonicalUsage && contract.canonicalUsage.length > 0) {\n lines.push(' canonicalUsage: [');\n for (const usage of contract.canonicalUsage) {\n lines.push(` \\`${usage}\\`,`);\n }\n lines.push(' ],');\n }\n\n // a11yRules\n if (contract.a11yRules && contract.a11yRules.length > 0) {\n lines.push(' a11yRules: [');\n for (const rule of contract.a11yRules) {\n lines.push(` '${escapeQuotes(rule)}',`);\n }\n lines.push(' ],');\n }\n\n // scenarioTags\n if (contract.scenarioTags && contract.scenarioTags.length > 0) {\n lines.push(' scenarioTags: [');\n for (const tag of contract.scenarioTags) {\n lines.push(` '${escapeQuotes(tag)}',`);\n }\n lines.push(' ],');\n }\n\n lines.push(' },');\n return lines.join('\\n');\n}\n\n// ---------------------------------------------------------------------------\n// Import path computation\n// ---------------------------------------------------------------------------\n\nfunction computeImportPath(\n fragmentDir: string,\n sourcePath: string,\n componentBaseName: string\n): string {\n const sourceDir = dirname(sourcePath);\n\n // Colocated: fragment sits next to the source file\n if (fragmentDir === sourceDir) {\n // If source is index.tsx, import from '.'\n if (componentBaseName === \"index\") {\n return \".\";\n }\n return `./${componentBaseName}`;\n }\n\n // Different directory: compute relative path\n let rel = relative(fragmentDir, sourceDir);\n if (!rel.startsWith(\".\")) rel = `./${rel}`;\n if (componentBaseName !== \"index\") {\n rel = `${rel}/${componentBaseName}`;\n }\n return rel;\n}\n\n// ---------------------------------------------------------------------------\n// Fragment file generation with TODO markers\n// ---------------------------------------------------------------------------\n\nfunction escapeQuotes(str: string): string {\n return str.replace(/'/g, \"\\\\'\");\n}\n\nfunction generateFragmentWithTodos(\n componentName: string,\n importPath: string,\n data: ComponentData,\n confidence: FieldConfidence,\n enrichment?: EnrichmentResult\n): string {\n const props = data.meta?.props ?? {};\n // Use only local props for inference — inherited HTML/React props cause\n // wrong descriptions (e.g., Card getting \"Form card for user input\" from\n // inherited defaultValue) and wrong roles (e.g., Card getting role: 'button'\n // from inherited onClick)\n const localProps = Object.fromEntries(\n Object.entries(props).filter(([_, p]) => p.source === 'local')\n );\n const composition = data.meta?.composition ?? null;\n const description = data.meta?.description || inferDescriptionFromMeta(componentName, localProps);\n const descriptionTodo = data.meta?.description ? \"\" : \" // TODO: Review description\";\n const category = inferCategoryFromMeta(componentName, localProps);\n const categoryTodo = category === \"Components\" ? \" // TODO: Set correct category\" : \"\";\n const status = inferStatus(data.component.sourcePath);\n const accessibility = inferAccessibilityFromMeta(localProps);\n\n // Build props block from PropMeta\n const propsBlock = buildPropsBlockFromMeta(props);\n\n // Build contract block — enrichment can replace/augment a11yRules and add scenarioTags\n const contract = buildContractBlock(componentName, props, composition, accessibility);\n if (contract && enrichment) {\n if (enrichment.a11yRules.length > 0) {\n contract.a11yRules = enrichment.a11yRules;\n }\n if (enrichment.scenarioTags.length > 0) {\n contract.scenarioTags = enrichment.scenarioTags;\n }\n }\n const contractBlock = contract ? emitContractBlock(contract) : '';\n\n // Build usage block — enriched or TODO markers\n let usageBlock: string;\n if (enrichment && (enrichment.when.length > 0 || enrichment.whenNot.length > 0)) {\n usageBlock = buildEnrichedUsageBlock(enrichment);\n } else {\n usageBlock = ` usage: {\n when: [\n // TODO: Describe when to use ${componentName}\n ],\n whenNot: [\n // TODO: Describe when NOT to use ${componentName}\n ],\n },`;\n }\n\n // Build meta.tags from enrichment\n const tagsLine = enrichment && enrichment.tags.length > 0\n ? `\\n tags: [${enrichment.tags.map(t => `'${escapeQuotes(t)}'`).join(', ')}],`\n : '';\n\n // Build variants (compound-aware)\n const variantsBlock = buildVariantsBlock(\n componentName,\n data.storyVariants,\n composition\n );\n\n // Build AI metadata block with composition info\n const aiBlock = buildAIBlock(composition);\n\n // Build provenance block\n const provenanceBlock = buildProvenanceBlock(confidence, props);\n\n // Build compound sub-components comment\n const compoundComment = composition && composition.parts.length > 0\n ? `\\n// Compound sub-components detected: ${composition.parts.map(p => p.name).join(', ')}`\n : '';\n\n return `// Auto-generated by fragments init --scan | Confidence: ${confidence.score}/100\n// ${confidence.todoFields.length} TODO(s) — search for \"TODO:\" and fill in human knowledge${compoundComment}\nimport React from 'react';\nimport { defineFragment } from '@fragments-sdk/core';\nimport { ${componentName} } from '${importPath}';\n\nexport default defineFragment({\n component: ${componentName},\n\n meta: {\n name: '${escapeQuotes(componentName)}',\n description: '${escapeQuotes(description)}',${descriptionTodo}\n category: '${escapeQuotes(category)}',${categoryTodo}\n status: '${status}',${tagsLine}\n },\n\n${usageBlock}\n\n props: ${propsBlock},${contractBlock}${aiBlock}\n\n variants: [\n${variantsBlock}\n ],\n${provenanceBlock}});\n`;\n}\n\nfunction buildPropsBlockFromMeta(props: Record<string, PropMeta>): string {\n const entries = Object.entries(props).filter(([_, p]) => p.source === 'local');\n if (entries.length === 0) return \"{}\";\n\n const lines = entries.map(([name, prop]) => {\n const type = prop.typeKind;\n const parts: string[] = [` type: '${type}'`];\n\n if (prop.description) {\n parts.push(\n ` description: '${escapeQuotes(prop.description.replace(/\\n/g, \" \"))}'`\n );\n }\n\n parts.push(` required: ${prop.required}`);\n\n if (prop.default !== undefined) {\n parts.push(` default: ${JSON.stringify(prop.default)}`);\n }\n\n if (prop.values && prop.values.length > 0) {\n parts.push(` values: ${JSON.stringify(prop.values)}`);\n }\n\n const todoComment =\n type === \"custom\" ? \" // TODO: Review type\" : \"\";\n\n return ` ${name}: {\\n${parts.join(\",\\n\")},\\n },${todoComment}`;\n });\n\n return `{\\n${lines.join(\"\\n\")}\\n }`;\n}\n\nfunction buildVariantsBlock(\n componentName: string,\n storyVariants: StoryVariant[],\n composition?: CompositionMeta | null\n): string {\n const entries: string[] = [];\n\n // Always include a Default variant\n const hasDefault = storyVariants.some((v) => v.name === \"Default\");\n if (!hasDefault) {\n if (composition && composition.pattern === 'compound' && composition.parts.length > 0) {\n // Generate a compound-aware default variant\n entries.push(formatCompoundVariantEntry(componentName, composition));\n } else {\n entries.push(formatVariantEntry(componentName, \"Default\", `Default ${componentName}`, {}));\n }\n }\n\n for (const variant of storyVariants) {\n const description = variant.name.replace(/([A-Z])/g, \" $1\").trim();\n entries.push(\n formatVariantEntry(\n componentName,\n variant.name,\n `${description} ${componentName}`,\n variant.args\n )\n );\n }\n\n return entries.join(\"\\n\");\n}\n\nfunction formatVariantEntry(\n componentName: string,\n name: string,\n description: string,\n args: Record<string, unknown>\n): string {\n const jsxCode = buildJsxString(componentName, args);\n return ` {\n name: '${escapeQuotes(name)}',\n description: '${escapeQuotes(description)}',\n code: \\`${jsxCode}\\`,\n render: () => ${jsxCode},\n },`;\n}\n\n/**\n * Generate a compound-aware default variant showing sub-component usage.\n */\nfunction formatCompoundVariantEntry(\n componentName: string,\n composition: CompositionMeta\n): string {\n const parts = composition.parts;\n const innerJsx = parts\n .map((part) => ` <${componentName}.${part.name}>...</${componentName}.${part.name}>`)\n .join('\\n');\n\n const jsxCode = `<${componentName}>\\n${innerJsx}\\n </${componentName}>`;\n\n return ` {\n name: 'Default',\n description: 'Default ${componentName} with sub-components',\n code: \\`${jsxCode}\\`,\n render: () => (\\n ${jsxCode}\\n ),\n },`;\n}\n\n/**\n * Build the ai block with composition metadata.\n */\nfunction buildAIBlock(composition: CompositionMeta | null): string {\n if (!composition || composition.parts.length === 0) return \"\";\n\n const subComponents = composition.parts.map((p) => `'${p.name}'`).join(', ');\n const pattern = `\\n <Component>\\n${composition.parts.map((p) => ` <Component.${p.name}>...</Component.${p.name}>`).join('\\n')}\\n </Component>`;\n\n return `\n\n ai: {\n compositionPattern: '${composition.pattern}',\n subComponents: [${subComponents}],\n commonPatterns: [\\`${pattern}\\n \\`],\n },`;\n}\n\n/**\n * Build _generated provenance block.\n */\nfunction buildProvenanceBlock(\n confidence: FieldConfidence,\n props: Record<string, PropMeta>\n): string {\n const autoFields = Object.keys(props).filter((name) => props[name].source === 'local');\n return `\n _generated: {\n source: 'ai',\n confidence: ${(confidence.score / 100).toFixed(2)},\n timestamp: '${new Date().toISOString()}',\n },\n`;\n}\n\nfunction buildJsxString(\n componentName: string,\n args: Record<string, unknown>\n): string {\n const { children, ...restArgs } = args;\n const propParts: string[] = [];\n\n for (const [key, value] of Object.entries(restArgs)) {\n if (typeof value === \"string\") {\n propParts.push(`${key}=\"${escapeQuotes(value)}\"`);\n } else if (typeof value === \"boolean\") {\n propParts.push(value ? key : `${key}={false}`);\n } else if (typeof value === \"number\") {\n propParts.push(`${key}={${value}}`);\n }\n }\n\n const propsStr = propParts.length > 0 ? \" \" + propParts.join(\" \") : \"\";\n\n if (typeof children === \"string\") {\n return `<${componentName}${propsStr}>${children}</${componentName}>`;\n }\n\n return `<${componentName}${propsStr} />`;\n}\n"],"mappings":"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","names":[]}
@@ -97,12 +97,12 @@ import {
97
97
  shutdownSharedPool,
98
98
  sleep,
99
99
  summarizePatternsForPrompt
100
- } from "./chunk-D5PYOXEI.js";
100
+ } from "./chunk-LVWFOLUZ.js";
101
101
  import "./chunk-D2CDBRNU.js";
102
102
  import {
103
103
  BRAND,
104
104
  DEFAULTS
105
- } from "./chunk-OQO55NKV.js";
105
+ } from "./chunk-WFS63PCW.js";
106
106
  import "./chunk-Z7EY4VHE.js";
107
107
  export {
108
108
  BRAND,
@@ -206,4 +206,4 @@ export {
206
206
  sleep,
207
207
  summarizePatternsForPrompt
208
208
  };
209
- //# sourceMappingURL=service-TQYWY65E.js.map
209
+ //# sourceMappingURL=service-VMGNJZ42.js.map
@@ -2,7 +2,7 @@ import { createRequire as __banner_createRequire } from 'module'; const require
2
2
  import "./chunk-D2CDBRNU.js";
3
3
  import {
4
4
  BRAND
5
- } from "./chunk-OQO55NKV.js";
5
+ } from "./chunk-WFS63PCW.js";
6
6
  import "./chunk-Z7EY4VHE.js";
7
7
 
8
8
  // src/commands/snapshot.ts
@@ -136,4 +136,4 @@ ${BRAND.name} Visual Snapshots
136
136
  export {
137
137
  snapshot
138
138
  };
139
- //# sourceMappingURL=snapshot-SV2JOFZH.js.map
139
+ //# sourceMappingURL=snapshot-XOISO2IS.js.map
@@ -2,12 +2,12 @@ import { createRequire as __banner_createRequire } from 'module'; const require
2
2
  import {
3
3
  generateStaticViewer,
4
4
  generateViewerFromJson
5
- } from "./chunk-PW7QTQA6.js";
5
+ } from "./chunk-4OC7FTJB.js";
6
6
  import "./chunk-D2CDBRNU.js";
7
- import "./chunk-OQO55NKV.js";
7
+ import "./chunk-WFS63PCW.js";
8
8
  import "./chunk-Z7EY4VHE.js";
9
9
  export {
10
10
  generateStaticViewer,
11
11
  generateViewerFromJson
12
12
  };
13
- //# sourceMappingURL=static-viewer-NUBFPKWH.js.map
13
+ //# sourceMappingURL=static-viewer-5GXH2MGE.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
@@ -1,11 +1,11 @@
1
1
  import { createRequire as __banner_createRequire } from 'module'; const require = __banner_createRequire(import.meta.url);
2
2
  import {
3
3
  parseFragmentFile
4
- } from "./chunk-HRFUSSZI.js";
4
+ } from "./chunk-AM4MRTMN.js";
5
5
  import {
6
6
  discoverFragmentFiles
7
- } from "./chunk-WXSR2II7.js";
8
- import "./chunk-OQO55NKV.js";
7
+ } from "./chunk-OQKMEFOS.js";
8
+ import "./chunk-WFS63PCW.js";
9
9
  import "./chunk-Z7EY4VHE.js";
10
10
 
11
11
  // src/test/index.ts
@@ -1071,4 +1071,4 @@ export {
1071
1071
  listTests,
1072
1072
  runTestCommand
1073
1073
  };
1074
- //# sourceMappingURL=test-Z5LVO724.js.map
1074
+ //# sourceMappingURL=test-SI4NSHQX.js.map
@@ -1,15 +1,15 @@
1
1
  import { createRequire as __banner_createRequire } from 'module'; const require = __banner_createRequire(import.meta.url);
2
2
  import {
3
3
  loadConfig
4
- } from "./chunk-HRFUSSZI.js";
5
- import "./chunk-WXSR2II7.js";
4
+ } from "./chunk-AM4MRTMN.js";
5
+ import "./chunk-OQKMEFOS.js";
6
6
  import {
7
7
  parseTokenFiles
8
- } from "./chunk-D5PYOXEI.js";
8
+ } from "./chunk-LVWFOLUZ.js";
9
9
  import "./chunk-D2CDBRNU.js";
10
10
  import {
11
11
  BRAND
12
- } from "./chunk-OQO55NKV.js";
12
+ } from "./chunk-WFS63PCW.js";
13
13
  import "./chunk-Z7EY4VHE.js";
14
14
 
15
15
  // src/commands/tokens.ts
@@ -172,4 +172,4 @@ export {
172
172
  tokens_default as default,
173
173
  tokens
174
174
  };
175
- //# sourceMappingURL=tokens-CE46OTMD.js.map
175
+ //# sourceMappingURL=tokens-T6SIVUT5.js.map
@@ -5,16 +5,16 @@ import {
5
5
  generatePreviewModule,
6
6
  loadConfig,
7
7
  parseFragmentFile
8
- } from "./chunk-HRFUSSZI.js";
8
+ } from "./chunk-AM4MRTMN.js";
9
9
  import {
10
10
  discoverFragmentFiles,
11
11
  discoverInstalledFragments
12
- } from "./chunk-WXSR2II7.js";
12
+ } from "./chunk-OQKMEFOS.js";
13
13
  import "./chunk-D2CDBRNU.js";
14
14
  import {
15
15
  BRAND,
16
16
  generateContext
17
- } from "./chunk-OQO55NKV.js";
17
+ } from "./chunk-WFS63PCW.js";
18
18
  import "./chunk-Z7EY4VHE.js";
19
19
 
20
20
  // src/viewer/server.ts
@@ -373,7 +373,7 @@ var sharedRenderPool = null;
373
373
  var browserPoolModule = null;
374
374
  async function getSharedRenderPool() {
375
375
  if (!browserPoolModule) {
376
- browserPoolModule = await import("./service-TQYWY65E.js");
376
+ browserPoolModule = await import("./service-VMGNJZ42.js");
377
377
  }
378
378
  if (!sharedRenderPool) {
379
379
  sharedRenderPool = new browserPoolModule.BrowserPool({
@@ -616,7 +616,7 @@ function fragmentsPlugin(options) {
616
616
  const address = _server.httpServer?.address();
617
617
  const port = typeof address === "object" && address ? address.port : 6006;
618
618
  const renderViewport = viewport || { width: 800, height: 600 };
619
- const { FigmaClient, bufferToBase64Url } = await import("./service-TQYWY65E.js");
619
+ const { FigmaClient, bufferToBase64Url } = await import("./service-VMGNJZ42.js");
620
620
  const figmaClient = new FigmaClient({
621
621
  accessToken: figmaToken
622
622
  });
@@ -707,7 +707,7 @@ function fragmentsPlugin(options) {
707
707
  );
708
708
  return;
709
709
  }
710
- const { FigmaClient } = await import("./service-TQYWY65E.js");
710
+ const { FigmaClient } = await import("./service-VMGNJZ42.js");
711
711
  const figmaClient = new FigmaClient({ accessToken: figmaToken });
712
712
  const { fileKey, nodeId } = figmaClient.parseUrl(figmaUrl);
713
713
  const figmaDesignProps = await figmaClient.getNodeProperties(
@@ -748,7 +748,7 @@ function fragmentsPlugin(options) {
748
748
  }));
749
749
  return;
750
750
  }
751
- const { getSharedTokenRegistry } = await import("./service-TQYWY65E.js");
751
+ const { getSharedTokenRegistry } = await import("./service-VMGNJZ42.js");
752
752
  const registry = getSharedTokenRegistry();
753
753
  if (!registry.isInitialized()) {
754
754
  await registry.initialize(config.tokens, projectRoot);
@@ -808,7 +808,7 @@ function fragmentsPlugin(options) {
808
808
  }));
809
809
  return;
810
810
  }
811
- const { getSharedTokenRegistry } = await import("./service-TQYWY65E.js");
811
+ const { getSharedTokenRegistry } = await import("./service-VMGNJZ42.js");
812
812
  const registry = getSharedTokenRegistry();
813
813
  if (!registry.isInitialized()) {
814
814
  await registry.initialize(config.tokens, projectRoot);
@@ -870,7 +870,7 @@ function fragmentsPlugin(options) {
870
870
  res.end(JSON.stringify({ error: "Could not resolve fragment file path" }));
871
871
  return;
872
872
  }
873
- const { getSharedTokenRegistry } = await import("./service-TQYWY65E.js");
873
+ const { getSharedTokenRegistry } = await import("./service-VMGNJZ42.js");
874
874
  const registry = getSharedTokenRegistry();
875
875
  if (!registry.isInitialized()) {
876
876
  await registry.initialize(config.tokens, projectRoot);
@@ -1048,7 +1048,7 @@ function fragmentsPlugin(options) {
1048
1048
  }
1049
1049
  if (!config.tokens || !config.tokens.include || config.tokens.include.length === 0) {
1050
1050
  try {
1051
- const { discoverTokenFiles } = await import("./discovery-NEOY4MPN.js");
1051
+ const { discoverTokenFiles } = await import("./discovery-ZJQSXF56.js");
1052
1052
  const discovered = await discoverTokenFiles(projectRoot);
1053
1053
  if (discovered.length > 0) {
1054
1054
  config.tokens = {
@@ -1085,7 +1085,7 @@ function fragmentsPlugin(options) {
1085
1085
  const {
1086
1086
  getSharedTokenRegistry,
1087
1087
  generateTokenPatches
1088
- } = await import("./service-TQYWY65E.js");
1088
+ } = await import("./service-VMGNJZ42.js");
1089
1089
  const registry = getSharedTokenRegistry();
1090
1090
  if (!registry.isInitialized()) {
1091
1091
  await registry.initialize(config.tokens, projectRoot);
@@ -2194,7 +2194,7 @@ async function loadFullFragmentForCompare(_server, _fragmentFiles, componentName
2194
2194
  }
2195
2195
  }
2196
2196
  async function compareImages(image1Base64, image2Base64, threshold) {
2197
- const { DiffEngine, base64UrlToBuffer, bufferToBase64Url } = await import("./service-TQYWY65E.js");
2197
+ const { DiffEngine, base64UrlToBuffer, bufferToBase64Url } = await import("./service-VMGNJZ42.js");
2198
2198
  const { PNG } = await import("pngjs");
2199
2199
  const buffer1 = base64UrlToBuffer(image1Base64);
2200
2200
  const buffer2 = base64UrlToBuffer(image2Base64);
@@ -2727,4 +2727,4 @@ export {
2727
2727
  createDevServer,
2728
2728
  fragmentsPlugin
2729
2729
  };
2730
- //# sourceMappingURL=viewer-DLLJIMCK.js.map
2730
+ //# sourceMappingURL=viewer-7ZEAFBVN.js.map
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@fragments-sdk/cli",
3
- "version": "0.11.1",
3
+ "version": "0.12.1",
4
4
  "license": "FSL-1.1-MIT",
5
5
  "description": "CLI, MCP server, and dev tools for Fragments design system",
6
6
  "author": "Conan McNicholl",
@@ -79,9 +79,9 @@
79
79
  "vite": "^6.0.0",
80
80
  "vite-plugin-svgr": "^4.5.0",
81
81
  "zod": "^3.24.1",
82
- "@fragments-sdk/core": "0.2.0",
83
- "@fragments-sdk/context": "0.4.3",
84
- "@fragments-sdk/viewer": "0.2.1",
82
+ "@fragments-sdk/core": "0.3.0",
83
+ "@fragments-sdk/context": "0.4.4",
84
+ "@fragments-sdk/viewer": "0.2.3",
85
85
  "@fragments-sdk/webmcp": "1.0.1"
86
86
  },
87
87
  "devDependencies": {
@@ -0,0 +1,156 @@
1
+ /**
2
+ * Shared AI client infrastructure for LLM-powered CLI commands.
3
+ *
4
+ * Used by:
5
+ * - `fragments enhance` (usage analysis + documentation generation)
6
+ * - `fragments init --scan --enrich` (knowledge field enrichment)
7
+ *
8
+ * Supports Anthropic (Claude) and OpenAI APIs via dynamic import.
9
+ * No new dependencies — uses existing @anthropic-ai/sdk and openai.
10
+ */
11
+
12
+ export type AIProvider = 'anthropic' | 'openai' | 'none';
13
+
14
+ /**
15
+ * Default lightweight models for enrichment (cheap, fast).
16
+ * Enhance uses its own heavier models locally.
17
+ */
18
+ export const ENRICHMENT_MODELS: Record<AIProvider, string> = {
19
+ anthropic: 'claude-haiku-4-5-20251001',
20
+ openai: 'gpt-4o-mini',
21
+ none: '',
22
+ };
23
+
24
+ /**
25
+ * Detect which AI provider to use based on available API keys and options.
26
+ */
27
+ export function detectProvider(opts?: {
28
+ provider?: AIProvider;
29
+ apiKey?: string;
30
+ }): AIProvider {
31
+ if (opts?.provider) return opts.provider;
32
+ if (opts?.apiKey) {
33
+ if (opts.apiKey.startsWith('sk-ant-')) return 'anthropic';
34
+ if (opts.apiKey.startsWith('sk-')) return 'openai';
35
+ }
36
+ if (process.env.ANTHROPIC_API_KEY) return 'anthropic';
37
+ if (process.env.OPENAI_API_KEY) return 'openai';
38
+ return 'none';
39
+ }
40
+
41
+ /**
42
+ * Resolve the API key for a given provider.
43
+ */
44
+ export function getApiKey(provider: AIProvider, explicitKey?: string): string | undefined {
45
+ if (explicitKey) return explicitKey;
46
+ if (provider === 'anthropic') return process.env.ANTHROPIC_API_KEY;
47
+ if (provider === 'openai') return process.env.OPENAI_API_KEY;
48
+ return undefined;
49
+ }
50
+
51
+ /**
52
+ * Create an AI client for the given provider via dynamic import.
53
+ */
54
+ export async function createAIClient(provider: AIProvider, apiKey: string): Promise<unknown> {
55
+ if (provider === 'anthropic') {
56
+ const Anthropic = (await import('@anthropic-ai/sdk')).default;
57
+ return new Anthropic({ apiKey });
58
+ }
59
+ if (provider === 'openai') {
60
+ const OpenAI = (await import('openai')).default;
61
+ return new OpenAI({ apiKey });
62
+ }
63
+ throw new Error(`Unknown provider: ${provider}`);
64
+ }
65
+
66
+ export interface CompletionResult {
67
+ text: string;
68
+ inputTokens: number;
69
+ outputTokens: number;
70
+ }
71
+
72
+ /**
73
+ * Generate a completion using the appropriate provider API.
74
+ */
75
+ export async function generateCompletion(
76
+ client: unknown,
77
+ provider: AIProvider,
78
+ model: string,
79
+ system: string,
80
+ user: string,
81
+ maxTokens: number = 1024
82
+ ): Promise<CompletionResult> {
83
+ if (provider === 'anthropic') {
84
+ const anthropic = client as import('@anthropic-ai/sdk').default;
85
+ const response = await anthropic.messages.create({
86
+ model,
87
+ max_tokens: maxTokens,
88
+ system,
89
+ messages: [{ role: 'user', content: user }],
90
+ });
91
+
92
+ const content = response.content[0];
93
+ if (content.type !== 'text') {
94
+ throw new Error('Unexpected response type');
95
+ }
96
+
97
+ return {
98
+ text: content.text,
99
+ inputTokens: response.usage?.input_tokens || 0,
100
+ outputTokens: response.usage?.output_tokens || 0,
101
+ };
102
+ }
103
+
104
+ if (provider === 'openai') {
105
+ const openai = client as import('openai').default;
106
+ const response = await openai.chat.completions.create({
107
+ model,
108
+ max_tokens: maxTokens,
109
+ messages: [
110
+ { role: 'system', content: system },
111
+ { role: 'user', content: user },
112
+ ],
113
+ });
114
+
115
+ const content = response.choices[0]?.message?.content;
116
+ if (!content) {
117
+ throw new Error('No response from OpenAI');
118
+ }
119
+
120
+ return {
121
+ text: content,
122
+ inputTokens: response.usage?.prompt_tokens || 0,
123
+ outputTokens: response.usage?.completion_tokens || 0,
124
+ };
125
+ }
126
+
127
+ throw new Error(`Unknown provider: ${provider}`);
128
+ }
129
+
130
+ /**
131
+ * Parse a JSON response from an LLM, handling ```json fences and raw JSON.
132
+ */
133
+ export function parseJSONResponse<T = unknown>(text: string): T {
134
+ const jsonMatch = text.match(/```json\n?([\s\S]*?)\n?```/) || text.match(/\{[\s\S]*\}/);
135
+ const jsonStr = jsonMatch ? (jsonMatch[1] || jsonMatch[0]) : text;
136
+ return JSON.parse(jsonStr);
137
+ }
138
+
139
+ /**
140
+ * Calculate estimated cost based on model and token usage.
141
+ */
142
+ export function calculateCost(model: string, inputTokens: number, outputTokens: number): number {
143
+ // Approximate costs per 1M tokens (input/output)
144
+ const pricing: Record<string, { input: number; output: number }> = {
145
+ // Anthropic
146
+ 'claude-haiku-4-5-20251001': { input: 0.80, output: 4.00 },
147
+ 'claude-sonnet-4-20250514': { input: 3.00, output: 15.00 },
148
+ // OpenAI
149
+ 'gpt-4o-mini': { input: 0.15, output: 0.60 },
150
+ 'gpt-4o': { input: 2.50, output: 10.00 },
151
+ };
152
+
153
+ const modelPricing = pricing[model] || { input: 3.00, output: 15.00 };
154
+ return (inputTokens / 1_000_000) * modelPricing.input +
155
+ (outputTokens / 1_000_000) * modelPricing.output;
156
+ }