@fragments-sdk/cli 0.7.3 → 0.7.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +1 -4
- package/README.md +2 -0
- package/dist/bin.js +39 -16
- package/dist/bin.js.map +1 -1
- package/dist/{chunk-D34Q6A7S.js → chunk-AWYCDRPG.js} +8 -2
- package/dist/chunk-AWYCDRPG.js.map +1 -0
- package/dist/{chunk-R2YH7NLN.js → chunk-CR3XHBGM.js} +3 -3
- package/dist/{chunk-QPY4DUFB.js → chunk-EFQ7SIBX.js} +583 -108
- package/dist/chunk-EFQ7SIBX.js.map +1 -0
- package/dist/{chunk-UXLGIGSX.js → chunk-GIC3I2KZ.js} +2 -2
- package/dist/{chunk-R6IZZSE7.js → chunk-JZNATKQA.js} +9 -3
- package/dist/chunk-JZNATKQA.js.map +1 -0
- package/dist/{chunk-P33AKQJW.js → chunk-SFWZ4K7C.js} +8 -2
- package/dist/{chunk-P33AKQJW.js.map → chunk-SFWZ4K7C.js.map} +1 -1
- package/dist/{core-3NMNCLFW.js → core-T7BDYEGO.js} +3 -3
- package/dist/{discovery-AKGA6CJD.js → discovery-Z4RDDFVR.js} +2 -2
- package/dist/{generate-JAUEHKK7.js → generate-C2DKFCFJ.js} +5 -5
- package/dist/index.d.ts +28 -2
- package/dist/index.js +9 -7
- package/dist/index.js.map +1 -1
- package/dist/{init-DZQOT54X.js → init-O3FCHEPN.js} +26 -8
- package/dist/init-O3FCHEPN.js.map +1 -0
- package/dist/mcp-bin.js +3 -3
- package/dist/{scan-OJRCVKK2.js → scan-IYTZDUKG.js} +6 -6
- package/dist/{service-CFFBHW4X.js → service-VA6XKADO.js} +3 -3
- package/dist/{static-viewer-VA2JXSCX.js → static-viewer-5N42MBDR.js} +3 -3
- package/dist/{test-O7DZNKDC.js → test-OMMDWL2W.js} +4 -4
- package/dist/{tokens-N7THFD6J.js → tokens-6VJAHFIG.js} +5 -5
- package/dist/{viewer-QTR7QJMM.js → viewer-IVP5XC7U.js} +37 -17
- package/dist/viewer-IVP5XC7U.js.map +1 -0
- package/package.json +8 -2
- package/src/bin.ts +4 -0
- package/src/commands/add.ts +6 -0
- package/src/commands/init.ts +24 -4
- package/src/commands/validate.ts +24 -2
- package/src/core/config.ts +6 -0
- package/src/core/discovery.ts +7 -1
- package/src/core/index.ts +1 -0
- package/src/core/schema.ts +6 -0
- package/src/core/types.ts +21 -0
- package/src/index.ts +2 -1
- package/src/migrate/detect.ts +4 -0
- package/src/service/snippet-validation.test.ts +209 -0
- package/src/service/snippet-validation.ts +635 -0
- package/src/validators.ts +53 -5
- package/src/viewer/__tests__/viewer-integration.test.ts +8 -0
- package/src/viewer/components/App.tsx +63 -2
- package/src/viewer/components/CodePanel.naming.test.tsx +60 -0
- package/src/viewer/components/CodePanel.tsx +76 -468
- package/src/viewer/components/Layout.tsx +2 -2
- package/src/viewer/components/LeftSidebar.tsx +35 -77
- package/src/viewer/preview-frame.html +1 -1
- package/src/viewer/styles/globals.css +2 -1
- package/src/viewer/utils/a11y-fixes.ts +24 -9
- package/src/viewer/vite-plugin.ts +27 -2
- package/dist/chunk-D34Q6A7S.js.map +0 -1
- package/dist/chunk-QPY4DUFB.js.map +0 -1
- package/dist/chunk-R6IZZSE7.js.map +0 -1
- package/dist/init-DZQOT54X.js.map +0 -1
- package/dist/viewer-QTR7QJMM.js.map +0 -1
- /package/dist/{chunk-R2YH7NLN.js.map → chunk-CR3XHBGM.js.map} +0 -0
- /package/dist/{chunk-UXLGIGSX.js.map → chunk-GIC3I2KZ.js.map} +0 -0
- /package/dist/{core-3NMNCLFW.js.map → core-T7BDYEGO.js.map} +0 -0
- /package/dist/{discovery-AKGA6CJD.js.map → discovery-Z4RDDFVR.js.map} +0 -0
- /package/dist/{generate-JAUEHKK7.js.map → generate-C2DKFCFJ.js.map} +0 -0
- /package/dist/{scan-OJRCVKK2.js.map → scan-IYTZDUKG.js.map} +0 -0
- /package/dist/{service-CFFBHW4X.js.map → service-VA6XKADO.js.map} +0 -0
- /package/dist/{static-viewer-VA2JXSCX.js.map → static-viewer-5N42MBDR.js.map} +0 -0
- /package/dist/{test-O7DZNKDC.js.map → test-OMMDWL2W.js.map} +0 -0
- /package/dist/{tokens-N7THFD6J.js.map → tokens-6VJAHFIG.js.map} +0 -0
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/validators.ts","../src/build.ts","../src/core/token-resolver.ts","../src/core/auto-props.ts","../src/core/graph-extractor.ts","../src/screenshot.ts","../src/diff.ts","../src/analyze.ts"],"sourcesContent":["import { fragmentDefinitionSchema, BRAND, type FragmentsConfig } from './core/index.js';\nimport {\n discoverFragmentFiles,\n discoverComponentFiles,\n extractComponentName,\n loadFragmentFile,\n type DiscoveredFile,\n} from './core/node.js';\n\nexport interface ValidationResult {\n valid: boolean;\n errors: ValidationError[];\n warnings: ValidationWarning[];\n}\n\nexport interface ValidationError {\n file: string;\n message: string;\n details?: string;\n}\n\nexport interface ValidationWarning {\n file: string;\n message: string;\n}\n\n/**\n * Validate fragment file schema\n */\nexport async function validateSchema(\n config: FragmentsConfig,\n configDir: string\n): Promise<ValidationResult> {\n const files = await discoverFragmentFiles(config, configDir);\n const errors: ValidationError[] = [];\n const warnings: ValidationWarning[] = [];\n\n for (const file of files) {\n try {\n const fragment = await loadFragmentFile(file.absolutePath);\n\n if (!fragment) {\n errors.push({\n file: file.relativePath,\n message: 'No default export found',\n details: `Fragment files must have a default export from defineFragment()`,\n });\n continue;\n }\n\n const result = fragmentDefinitionSchema.safeParse(fragment);\n\n if (!result.success) {\n const details = result.error.errors\n .map((e) => `${e.path.join('.')}: ${e.message}`)\n .join('; ');\n\n errors.push({\n file: file.relativePath,\n message: 'Invalid fragment schema',\n details,\n });\n }\n } catch (error) {\n errors.push({\n file: file.relativePath,\n message: 'Failed to load fragment file',\n details: error instanceof Error ? error.message : String(error),\n });\n }\n }\n\n return {\n valid: errors.length === 0,\n errors,\n warnings,\n };\n}\n\n/**\n * Validate coverage - ensure all components have fragments\n */\nexport async function validateCoverage(\n config: FragmentsConfig,\n configDir: string\n): Promise<ValidationResult> {\n const fragmentFiles = await discoverFragmentFiles(config, configDir);\n const componentFiles = await discoverComponentFiles(config, configDir);\n const errors: ValidationError[] = [];\n const warnings: ValidationWarning[] = [];\n\n // Build set of documented component names\n const documentedComponents = new Set<string>();\n\n for (const file of fragmentFiles) {\n try {\n const fragment = await loadFragmentFile(file.absolutePath);\n\n if (fragment?.meta?.name) {\n documentedComponents.add(fragment.meta.name);\n }\n } catch {\n // Skip files that fail to load - schema validation will catch these\n }\n }\n\n // Check each component file\n for (const file of componentFiles) {\n const componentName = extractComponentName(file.relativePath);\n\n // Check if there's a corresponding fragment file\n const fragmentPath = file.relativePath.replace(\n /\\.(tsx?|jsx?)$/,\n BRAND.fileExtension\n );\n const hasFragmentFile = fragmentFiles.some(\n (s) => s.relativePath === fragmentPath\n );\n\n if (!hasFragmentFile && !documentedComponents.has(componentName)) {\n warnings.push({\n file: file.relativePath,\n message: `Component \"${componentName}\" has no fragment documentation`,\n });\n }\n }\n\n return {\n valid: errors.length === 0,\n errors,\n warnings,\n };\n}\n\n/**\n * Run all validations\n */\nexport async function validateAll(\n config: FragmentsConfig,\n configDir: string\n): Promise<ValidationResult> {\n const [schemaResult, coverageResult] = await Promise.all([\n validateSchema(config, configDir),\n validateCoverage(config, configDir),\n ]);\n\n return {\n valid: schemaResult.valid && coverageResult.valid,\n errors: [...schemaResult.errors, ...coverageResult.errors],\n warnings: [...schemaResult.warnings, ...coverageResult.warnings],\n };\n}\n","import { readFile, writeFile, mkdir } from \"node:fs/promises\";\nimport { resolve, join } from \"node:path\";\nimport { existsSync } from \"node:fs\";\nimport type {\n FragmentsConfig,\n CompiledFragmentsFile,\n CompiledFragment,\n CompiledBlock,\n CompiledTokenData,\n} from \"./core/index.js\";\nimport { BRAND, compileBlock, parseTokenFile } from \"./core/index.js\";\nimport { resolveTokensWithSass } from \"./core/token-resolver.js\";\nimport type { BlockDefinition } from \"./core/index.js\";\nimport {\n discoverFragmentFiles,\n discoverBlockFiles,\n discoverTokenFiles,\n parseFragmentFile,\n loadFragmentFile,\n generateRegistry,\n generateContextMd,\n} from \"./core/node.js\";\nimport {\n extractCustomPropsFromComponentFile,\n resolveComponentSourcePath,\n type AutoDetectedPropDefinition,\n} from \"./core/auto-props.js\";\nimport { buildComponentGraph } from \"./core/graph-extractor.js\";\nimport { serializeGraph } from \"@fragments-sdk/context/graph\";\n\ntype CompiledProp = CompiledFragment[\"props\"][string];\n\nfunction normalizeParsedProps(\n parsedProps: Record<string, Partial<CompiledProp>>\n): Record<string, CompiledProp> {\n return Object.fromEntries(\n Object.entries(parsedProps).map(([name, prop]) => [\n name,\n {\n type: prop.type ?? \"custom\",\n description: prop.description ?? \"\",\n default: prop.default,\n required: prop.required,\n values: prop.values,\n constraints: prop.constraints,\n },\n ])\n );\n}\n\nfunction mergeDocumentedAndAutoProps(\n documentedProps: Record<string, CompiledProp>,\n autoProps: Record<string, AutoDetectedPropDefinition>\n): Record<string, CompiledProp> {\n return Object.fromEntries(\n Object.keys(autoProps).map((name) => {\n const documented = documentedProps[name];\n const auto = autoProps[name];\n\n return [\n name,\n {\n type: auto.type,\n description: documented?.description ?? auto.description ?? \"\",\n default: auto.default !== undefined ? auto.default : documented?.default,\n required: auto.required,\n values: auto.values ?? documented?.values,\n constraints: documented?.constraints,\n },\n ];\n })\n );\n}\n\nexport interface BuildResult {\n success: boolean;\n outputPath: string;\n fragmentCount: number;\n errors: Array<{ file: string; error: string }>;\n warnings: Array<{ file: string; warning: string }>;\n}\n\n/**\n * Build compiled fragments.json file for AI consumption.\n *\n * Uses AST parsing to extract metadata WITHOUT executing fragment files.\n * This means the build works without any project dependencies installed.\n */\nexport async function buildFragments(\n config: FragmentsConfig,\n configDir: string\n): Promise<BuildResult> {\n const files = await discoverFragmentFiles(config, configDir);\n const errors: Array<{ file: string; error: string }> = [];\n const warnings: Array<{ file: string; warning: string }> = [];\n const fragments: CompiledFragmentsFile[\"fragments\"] = {};\n\n for (const file of files) {\n try {\n // Read file content as text\n const content = await readFile(file.absolutePath, \"utf-8\");\n\n // Parse using AST (no execution)\n const parsed = parseFragmentFile(content, file.relativePath);\n\n // Collect warnings\n for (const warning of parsed.warnings) {\n warnings.push({ file: file.relativePath, warning });\n }\n\n // Check for required fields\n if (!parsed.meta.name) {\n errors.push({\n file: file.relativePath,\n error: \"Missing meta.name in fragment definition\",\n });\n continue;\n }\n\n const documentedProps = normalizeParsedProps(parsed.props);\n let mergedProps = documentedProps;\n\n const componentExportName = parsed.componentName ?? parsed.meta.name;\n const componentSourcePath = resolveComponentSourcePath(\n file.absolutePath,\n parsed.componentImport\n );\n\n if (componentExportName && componentSourcePath) {\n const autoPropsResult = extractCustomPropsFromComponentFile(\n componentSourcePath,\n componentExportName\n );\n\n for (const warning of autoPropsResult.warnings) {\n warnings.push({ file: file.relativePath, warning });\n }\n\n const hasAutoProps = Object.keys(autoPropsResult.props).length > 0;\n if (autoPropsResult.resolved && hasAutoProps) {\n const removedDocumentedProps = Object.keys(documentedProps).filter(\n (propName) => !(propName in autoPropsResult.props)\n );\n\n if (removedDocumentedProps.length > 0) {\n warnings.push({\n file: file.relativePath,\n warning: `Removed ${removedDocumentedProps.length} documented props not present in source API: ${removedDocumentedProps.join(\", \")}`,\n });\n }\n\n mergedProps = mergeDocumentedAndAutoProps(\n documentedProps,\n autoPropsResult.props\n );\n } else if (autoPropsResult.resolved && !hasAutoProps && Object.keys(documentedProps).length > 0) {\n warnings.push({\n file: file.relativePath,\n warning: \"Auto-props extraction returned no custom props; falling back to documented props\",\n });\n }\n } else if (!componentExportName) {\n warnings.push({\n file: file.relativePath,\n warning: \"Unable to resolve component export name for auto-props extraction\",\n });\n } else if (!componentSourcePath) {\n warnings.push({\n file: file.relativePath,\n warning: `Unable to resolve component source path from import: ${parsed.componentImport ?? \"unknown\"}`,\n });\n }\n\n // Build compiled fragment from parsed metadata\n const compiled: CompiledFragment = {\n filePath: file.relativePath,\n meta: {\n name: parsed.meta.name,\n description: parsed.meta.description ?? \"\",\n category: parsed.meta.category ?? \"Uncategorized\",\n status: parsed.meta.status,\n tags: parsed.meta.tags,\n since: parsed.meta.since,\n figma: parsed.meta.figma,\n },\n usage: {\n when: parsed.usage.when ?? [],\n whenNot: parsed.usage.whenNot ?? [],\n guidelines: parsed.usage.guidelines,\n accessibility: parsed.usage.accessibility,\n },\n props: mergedProps,\n relations: parsed.relations.map((rel) => ({\n component: rel.component,\n relationship: rel.relationship as\n | \"alternative\"\n | \"sibling\"\n | \"parent\"\n | \"child\"\n | \"composition\",\n note: rel.note,\n })),\n variants: parsed.variants.map((v) => ({\n name: v.name,\n description: v.description,\n ...(v.code && { code: v.code }),\n ...(v.figma && { figma: v.figma }),\n ...(v.args && { args: v.args }),\n })),\n // Include AI metadata if present\n ...(parsed.ai && { ai: parsed.ai }),\n // Include contract metadata if present\n ...(parsed.contract && { contract: parsed.contract }),\n };\n\n fragments[parsed.meta.name] = compiled;\n } catch (error) {\n errors.push({\n file: file.relativePath,\n error: error instanceof Error ? error.message : String(error),\n });\n }\n }\n\n // Discover and compile block files\n const blocks: Record<string, CompiledBlock> = {};\n try {\n const blockFiles = await discoverBlockFiles(configDir, config.exclude);\n for (const file of blockFiles) {\n try {\n // loadFragmentFile uses esbuild to bundle+evaluate, returns default export\n // CJS/ESM interop may double-wrap the default export\n let raw = await loadFragmentFile(file.absolutePath) as unknown as Record<string, unknown> | null;\n // Unwrap double-default from CJS interop\n if (raw && 'default' in raw && typeof raw.default === 'object') {\n raw = raw.default as Record<string, unknown>;\n }\n const def = raw;\n if (def && typeof def === 'object' && 'name' in def && 'code' in def && 'components' in def) {\n const compiled = compileBlock(def as unknown as BlockDefinition, file.relativePath);\n blocks[compiled.name] = compiled;\n }\n } catch (error) {\n warnings.push({\n file: file.relativePath,\n warning: `Failed to load block: ${error instanceof Error ? error.message : String(error)}`,\n });\n }\n }\n } catch {\n // Block discovery failure is non-fatal\n }\n\n // Discover and extract design tokens from SCSS/CSS files\n let tokens: CompiledTokenData | undefined;\n try {\n const tokenPatterns = config.tokens?.include;\n const tokenFiles = await discoverTokenFiles(configDir, tokenPatterns, config.exclude);\n if (tokenFiles.length > 0) {\n // Merge tokens from all discovered files\n const mergedCategories: Record<string, Array<{ name: string; value?: string; description?: string }>> = {};\n let prefix = '--';\n let total = 0;\n\n // Read all file contents first for cross-file SCSS variable resolution\n const fileContents: Array<{ content: string; path: string }> = [];\n for (const file of tokenFiles) {\n const content = await readFile(file.absolutePath, 'utf-8');\n fileContents.push({ content, path: file.relativePath });\n }\n\n // Concatenate all contents so parseTokenFile can resolve SCSS vars across files\n const allContent = fileContents.map((f) => f.content).join('\\n');\n\n for (const { content, path } of fileContents) {\n // Parse with the combined content to enable cross-file SCSS var resolution\n const parsed = parseTokenFile(allContent, path);\n // But only use tokens from THIS file's content to avoid duplicates\n const fileParsed = parseTokenFile(content, path);\n prefix = fileParsed.prefix;\n total += fileParsed.total;\n for (const [cat, catTokens] of Object.entries(fileParsed.categories)) {\n if (!mergedCategories[cat]) {\n mergedCategories[cat] = [];\n }\n for (const t of catTokens) {\n // Deduplicate by name\n if (!mergedCategories[cat].some((e) => e.name === t.name)) {\n // Use resolved value from the combined parse if available\n const combinedToken = Object.values(parsed.categories)\n .flat()\n .find((ct) => ct.name === t.name);\n const resolvedValue = combinedToken?.resolvedValue ?? t.resolvedValue;\n\n mergedCategories[cat].push({\n name: t.name,\n ...(resolvedValue\n ? { value: resolvedValue }\n : t.value ? { value: t.value } : {}),\n description: t.description,\n });\n }\n }\n }\n }\n\n // Sass compilation fallback: resolve tokens the regex parser couldn't handle\n if (total > 0) {\n const allTokens = Object.values(mergedCategories).flat();\n const unresolved = allTokens.filter(\n t => t.value && (t.value.includes('#{') || t.value.includes('$'))\n );\n\n if (unresolved.length > 0 && tokenFiles.length > 0) {\n // Determine the tokens directory from the first discovered file\n const tokensDir = resolve(configDir, tokenFiles[0].relativePath, '..');\n const sassResolved = await resolveTokensWithSass(\n unresolved as Array<{ name: string; value: string }>,\n tokensDir,\n );\n\n // Merge sass-resolved values back into the categories\n if (sassResolved.size > 0) {\n for (const catTokens of Object.values(mergedCategories)) {\n for (const token of catTokens) {\n const resolved = sassResolved.get(token.name);\n if (resolved && token.value && (token.value.includes('#{') || token.value.includes('$'))) {\n token.value = resolved;\n }\n }\n }\n }\n }\n\n tokens = { prefix, total, categories: mergedCategories };\n }\n }\n } catch {\n // Token extraction failure is non-fatal\n }\n\n // Read package name for import statements\n let packageName: string | undefined;\n const pkgJsonPath = resolve(configDir, \"package.json\");\n if (existsSync(pkgJsonPath)) {\n try {\n const pkg = JSON.parse(await readFile(pkgJsonPath, \"utf-8\"));\n if (pkg.name) packageName = pkg.name;\n } catch {\n // Non-fatal\n }\n }\n\n // Build component graph for AI structural queries\n // Derive component directory from configDir (typically src/components/)\n const componentDir = resolve(configDir, \"src\", \"components\");\n let graphData: ReturnType<typeof serializeGraph> | undefined;\n try {\n const graphResult = await buildComponentGraph(fragments, blocks, componentDir);\n\n // Auto-enrich fragments with detected metadata\n for (const [name, fragment] of Object.entries(fragments)) {\n const detected = graphResult.autoDetected.get(name);\n if (!detected) continue;\n\n if (!fragment.ai) fragment.ai = {};\n if (!fragment.ai.subComponents && detected.subComponents) {\n fragment.ai.subComponents = detected.subComponents;\n }\n if (!fragment.ai.compositionPattern && detected.compositionPattern) {\n fragment.ai.compositionPattern = detected.compositionPattern;\n }\n if (!fragment.ai.commonPatterns && detected.commonPatterns) {\n fragment.ai.commonPatterns = detected.commonPatterns;\n }\n if (!fragment.ai.requiredChildren && detected.requiredChildren) {\n fragment.ai.requiredChildren = detected.requiredChildren;\n }\n }\n\n // Report drift warnings\n for (const w of graphResult.warnings) {\n warnings.push({ file: \"graph\", warning: w });\n }\n\n graphData = serializeGraph(graphResult.graph);\n } catch (error) {\n warnings.push({\n file: \"graph\",\n warning: `Graph extraction failed: ${error instanceof Error ? error.message : String(error)}`,\n });\n }\n\n const output: CompiledFragmentsFile = {\n version: \"1.0.0\",\n generatedAt: new Date().toISOString(),\n ...(packageName && { packageName }),\n fragments,\n ...(Object.keys(blocks).length > 0 && { blocks }),\n ...(tokens && { tokens }),\n ...(graphData && { graph: graphData }),\n };\n\n const outputPath = resolve(configDir, config.outFile ?? BRAND.outFile);\n await writeFile(outputPath, JSON.stringify(output));\n\n return {\n success: errors.length === 0,\n outputPath,\n fragmentCount: Object.keys(fragments).length,\n errors,\n warnings,\n };\n}\n\n/**\n * Result of building the .fragments directory\n */\nexport interface FragmentsBuildResult {\n success: boolean;\n indexPath: string;\n registryPath: string;\n contextPath: string;\n componentCount: number;\n errors: Array<{ file: string; error: string }>;\n warnings: Array<{ file: string; warning: string }>;\n}\n\n/**\n * Build the .fragments/ directory with index.json, registry.json, and context.md\n *\n * This generates:\n * - .fragments/index.json - Minimal name → path mapping (tiny, for quick lookups)\n * - .fragments/registry.json - Component index with paths and enrichment references\n * - .fragments/context.md - AI-ready consolidated context file\n */\nexport async function buildFragmentsDir(\n config: FragmentsConfig,\n configDir: string\n): Promise<FragmentsBuildResult> {\n const fragmentsDir = join(configDir, BRAND.dataDir);\n const componentsDir = join(fragmentsDir, BRAND.componentsDir);\n\n // Create directories\n await mkdir(fragmentsDir, { recursive: true });\n await mkdir(componentsDir, { recursive: true });\n\n // Generate registry with config options\n const registryResult = await generateRegistry({\n projectRoot: configDir,\n componentPatterns: config.components || [\"src/**/*.tsx\", \"src/**/*.ts\"],\n storyPatterns: config.include || [\"src/**/*.stories.tsx\"],\n fragmentsDir,\n registryOptions: config.registry || {},\n });\n\n const errors = [...registryResult.errors];\n const warnings = [...registryResult.warnings];\n\n // Write index.json (minimal name → path)\n const indexPath = join(fragmentsDir, \"index.json\");\n await writeFile(indexPath, JSON.stringify(registryResult.index, null, 2));\n\n // Write registry.json (full metadata)\n const registryPath = join(fragmentsDir, BRAND.registryFile);\n await writeFile(registryPath, JSON.stringify(registryResult.registry, null, 2));\n\n // Generate context.md - focus on semantic knowledge, skip props (AI can read source)\n const contextResult = generateContextMd(registryResult.registry, {\n format: \"markdown\",\n compact: false,\n include: {\n props: false, // AI can read TypeScript directly\n relations: true,\n code: false,\n },\n });\n\n // Write context.md\n const contextPath = join(fragmentsDir, BRAND.contextFile);\n await writeFile(contextPath, contextResult.content);\n\n return {\n success: errors.length === 0,\n indexPath,\n registryPath,\n contextPath,\n componentCount: registryResult.registry.componentCount,\n errors,\n warnings,\n };\n}\n","/**\n * Token Resolver — resolves unresolved SCSS token values using actual sass compilation.\n *\n * The regex-based resolver in token-parser.ts cannot handle SCSS module namespaces,\n * functions (color.scale, math.div), or map lookups. This module provides a fallback\n * that compiles the actual SCSS sources to extract computed CSS custom property values.\n *\n * Used at build time only — no runtime sass dependency in the MCP server.\n */\n\nimport { resolve, dirname, basename } from 'node:path';\nimport { existsSync, readdirSync } from 'node:fs';\n\n/**\n * Round fractional RGB channel values produced by Sass color math.\n *\n * Sass functions like color.scale() and color.adjust() can produce\n * fractional values like rgb(217.8, 217.8, 217.8). This normalizes\n * them to clean integers: rgb(218, 218, 218).\n *\n * Alpha channels in rgba() are preserved as-is (they're legitimately fractional).\n */\nfunction roundRgbValues(value: string): string {\n return value\n .replace(\n /rgb\\(([^)]+)\\)/g,\n (_full, inner: string) => {\n const parts = inner.split(',').map(p => p.trim());\n const rounded = parts.map(p => {\n const num = parseFloat(p);\n return isNaN(num) ? p : String(Math.round(num));\n });\n return `rgb(${rounded.join(', ')})`;\n },\n )\n .replace(\n /rgba\\(([^)]+)\\)/g,\n (_full, inner: string) => {\n const parts = inner.split(',').map(p => p.trim());\n // Round RGB channels (first 3), preserve alpha as-is\n const rounded = parts.map((p, i) => {\n if (i >= 3) return p; // alpha channel — don't round\n const num = parseFloat(p);\n return isNaN(num) ? p : String(Math.round(num));\n });\n return `rgba(${rounded.join(', ')})`;\n },\n );\n}\n\n/**\n * Resolve unresolved SCSS token values by compiling the project's token SCSS.\n *\n * Generates a temporary SCSS string that imports the project's _variables.scss\n * and includes the fui-css-variables mixin, then compiles it with sass to extract\n * the actual computed values for all CSS custom properties.\n *\n * @param unresolvedTokens - Tokens with values containing #{} or $ that need resolution\n * @param tokensDir - Absolute path to the directory containing _variables.scss\n * @returns Map of token name → resolved CSS value (only for tokens that were unresolved)\n */\nexport async function resolveTokensWithSass(\n unresolvedTokens: Array<{ name: string; value: string }>,\n tokensDir: string,\n): Promise<Map<string, string>> {\n const resolvedMap = new Map<string, string>();\n\n // Filter to only tokens that actually need resolution\n const needsResolution = unresolvedTokens.filter(\n t => t.value.includes('#{') || t.value.includes('$'),\n );\n\n if (needsResolution.length === 0) {\n return resolvedMap;\n }\n\n try {\n // Dynamic import so sass is only loaded when needed (build-time only)\n const sass = await import('sass');\n\n // Find the _variables.scss file and determine the mixin name\n const variablesPath = findVariablesFile(tokensDir);\n if (!variablesPath) {\n return resolvedMap;\n }\n\n // Determine the module name for @use (filename without _ prefix and .scss suffix)\n const fileName = basename(variablesPath);\n const moduleName = fileName.replace(/^_/, '').replace(/\\.scss$/, '');\n\n // Generate a SCSS string that imports the variables and includes the mixin\n const scssSource = `\n@use '${moduleName}' as vars;\n:root { @include vars.fui-css-variables; }\n`;\n\n // Compile the SCSS with the tokens directory as a load path\n const compiled = sass.compileString(scssSource, {\n loadPaths: [tokensDir, dirname(tokensDir)],\n style: 'expanded',\n // Suppress sass deprecation warnings during build\n logger: { warn() {}, debug() {} } as never,\n });\n\n // Parse the compiled CSS for --fui-* custom property declarations\n const cssVarRegex = /(--[\\w-]+)\\s*:\\s*([^;]+)/g;\n let match: RegExpExecArray | null;\n const allResolved = new Map<string, string>();\n\n while ((match = cssVarRegex.exec(compiled.css)) !== null) {\n allResolved.set(match[1], roundRgbValues(match[2].trim()));\n }\n\n // Only return values for the tokens that were originally unresolved\n for (const token of needsResolution) {\n const value = allResolved.get(token.name);\n if (value !== undefined) {\n resolvedMap.set(token.name, value);\n }\n }\n } catch {\n // Sass compilation failure is non-fatal — fall back to regex-resolved values\n }\n\n return resolvedMap;\n}\n\n/**\n * Find the _variables.scss file in the tokens directory.\n */\nfunction findVariablesFile(tokensDir: string): string | null {\n const candidates = ['_variables.scss', 'variables.scss'];\n for (const name of candidates) {\n const path = resolve(tokensDir, name);\n if (existsSync(path)) {\n return path;\n }\n }\n\n // Fallback: look for any SCSS file containing 'css-variables' mixin\n try {\n const files = readdirSync(tokensDir).filter(f => f.endsWith('.scss'));\n for (const file of files) {\n const path = resolve(tokensDir, file);\n // We just need to find it exists; the @use import will handle the rest\n if (file.includes('variables') || file.includes('tokens')) {\n return path;\n }\n }\n } catch {\n // Directory read failure\n }\n\n return null;\n}\n","import { existsSync, statSync } from \"node:fs\";\nimport { dirname, extname, join, resolve } from \"node:path\";\nimport ts from \"typescript\";\nimport type { PropDefinition } from \"./types.js\";\n\nexport interface AutoDetectedPropDefinition {\n type: PropDefinition[\"type\"];\n description: string;\n required: boolean;\n default?: unknown;\n values?: readonly string[];\n}\n\nexport interface AutoPropsExtractionResult {\n props: Record<string, AutoDetectedPropDefinition>;\n warnings: string[];\n resolved: boolean;\n}\n\ninterface ResolvedComponentSignature {\n propsTypeNode: ts.TypeNode | null;\n componentNode: ts.FunctionLikeDeclarationBase | null;\n}\n\nfunction toPosixPath(filePath: string): string {\n return filePath.replace(/\\\\/g, \"/\");\n}\n\nfunction isFile(filePath: string): boolean {\n if (!existsSync(filePath)) return false;\n try {\n return statSync(filePath).isFile();\n } catch {\n return false;\n }\n}\n\nfunction resolveModulePath(basePath: string): string | null {\n const candidates: string[] = [];\n const extension = extname(basePath);\n\n if (extension) {\n candidates.push(basePath);\n } else {\n candidates.push(\n `${basePath}.tsx`,\n `${basePath}.ts`,\n `${basePath}.jsx`,\n `${basePath}.js`,\n join(basePath, \"index.tsx\"),\n join(basePath, \"index.ts\"),\n join(basePath, \"index.jsx\"),\n join(basePath, \"index.js\")\n );\n }\n\n for (const candidate of candidates) {\n if (isFile(candidate)) {\n return resolve(candidate);\n }\n }\n\n return null;\n}\n\n/**\n * Resolve a component source file from the fragment file path + component import path.\n * Supports relative imports like \".\", \"./Button\", \"../components/Button\".\n */\nexport function resolveComponentSourcePath(\n fragmentFileAbsolutePath: string,\n componentImportPath: string | null\n): string | null {\n if (!componentImportPath) return null;\n if (!componentImportPath.startsWith(\".\")) return null;\n\n const fragmentDir = dirname(fragmentFileAbsolutePath);\n const basePath = resolve(fragmentDir, componentImportPath);\n return resolveModulePath(basePath);\n}\n\nfunction collectTopLevelDeclarations(sourceFile: ts.SourceFile): {\n typeDeclarations: Map<string, ts.InterfaceDeclaration | ts.TypeAliasDeclaration>;\n functionDeclarations: Map<string, ts.FunctionDeclaration>;\n variableDeclarations: Map<string, ts.VariableDeclaration>;\n} {\n const typeDeclarations = new Map<string, ts.InterfaceDeclaration | ts.TypeAliasDeclaration>();\n const functionDeclarations = new Map<string, ts.FunctionDeclaration>();\n const variableDeclarations = new Map<string, ts.VariableDeclaration>();\n\n for (const node of sourceFile.statements) {\n if (ts.isInterfaceDeclaration(node)) {\n typeDeclarations.set(node.name.text, node);\n continue;\n }\n\n if (ts.isTypeAliasDeclaration(node)) {\n typeDeclarations.set(node.name.text, node);\n continue;\n }\n\n if (ts.isFunctionDeclaration(node) && node.name) {\n functionDeclarations.set(node.name.text, node);\n continue;\n }\n\n if (ts.isVariableStatement(node)) {\n for (const declaration of node.declarationList.declarations) {\n if (ts.isIdentifier(declaration.name)) {\n variableDeclarations.set(declaration.name.text, declaration);\n }\n }\n }\n }\n\n return { typeDeclarations, functionDeclarations, variableDeclarations };\n}\n\nfunction readDefaultValue(expression: ts.Expression): unknown {\n if (ts.isStringLiteral(expression) || ts.isNoSubstitutionTemplateLiteral(expression)) {\n return expression.text;\n }\n\n if (ts.isNumericLiteral(expression)) {\n return Number(expression.text);\n }\n\n if (expression.kind === ts.SyntaxKind.TrueKeyword) return true;\n if (expression.kind === ts.SyntaxKind.FalseKeyword) return false;\n if (expression.kind === ts.SyntaxKind.NullKeyword) return null;\n\n if (\n ts.isPrefixUnaryExpression(expression) &&\n expression.operator === ts.SyntaxKind.MinusToken &&\n ts.isNumericLiteral(expression.operand)\n ) {\n return -Number(expression.operand.text);\n }\n\n return undefined;\n}\n\nfunction extractDefaultValues(\n componentNode: ts.FunctionLikeDeclarationBase | null\n): Record<string, unknown> {\n const defaults: Record<string, unknown> = {};\n if (!componentNode?.parameters?.length) return defaults;\n\n const firstParam = componentNode.parameters[0];\n if (!ts.isObjectBindingPattern(firstParam.name)) return defaults;\n\n for (const element of firstParam.name.elements) {\n let propName: string | null = null;\n\n if (element.propertyName) {\n if (ts.isIdentifier(element.propertyName) || ts.isStringLiteral(element.propertyName)) {\n propName = element.propertyName.text;\n }\n } else if (ts.isIdentifier(element.name)) {\n propName = element.name.text;\n }\n\n if (!propName || !element.initializer) continue;\n\n const value = readDefaultValue(element.initializer);\n if (value !== undefined) {\n defaults[propName] = value;\n }\n }\n\n return defaults;\n}\n\nfunction isNullishType(type: ts.Type): boolean {\n return (\n (type.flags & ts.TypeFlags.Null) !== 0 ||\n (type.flags & ts.TypeFlags.Undefined) !== 0 ||\n (type.flags & ts.TypeFlags.Void) !== 0\n );\n}\n\nfunction isBooleanLikeType(type: ts.Type): boolean {\n return (\n (type.flags & ts.TypeFlags.BooleanLike) !== 0 ||\n type.flags === ts.TypeFlags.BooleanLiteral\n );\n}\n\nfunction inferPropType(\n type: ts.Type,\n checker: ts.TypeChecker\n): Pick<AutoDetectedPropDefinition, \"type\" | \"values\"> {\n const typeText = checker.typeToString(type, undefined, ts.TypeFormatFlags.NoTruncation);\n\n if (typeText.includes(\"ReactNode\")) {\n return { type: \"node\" };\n }\n if (typeText.includes(\"ReactElement\") || typeText.includes(\"JSX.Element\")) {\n return { type: \"element\" };\n }\n\n if (type.getCallSignatures().length > 0) {\n return { type: \"function\" };\n }\n\n if (checker.isArrayType(type) || checker.isTupleType(type)) {\n return { type: \"array\" };\n }\n\n if (type.isUnion()) {\n const nonNullableTypes = type.types.filter((unionType) => !isNullishType(unionType));\n\n if (nonNullableTypes.length === 1) {\n return inferPropType(nonNullableTypes[0], checker);\n }\n\n const stringLiteralValues = nonNullableTypes\n .filter((unionType) => (unionType.flags & ts.TypeFlags.StringLiteral) !== 0)\n .map((unionType) => (unionType as ts.StringLiteralType).value);\n\n if (stringLiteralValues.length > 0 && stringLiteralValues.length === nonNullableTypes.length) {\n return { type: \"enum\", values: stringLiteralValues };\n }\n\n if (nonNullableTypes.every((unionType) => isBooleanLikeType(unionType))) {\n return { type: \"boolean\" };\n }\n\n return { type: \"union\" };\n }\n\n if ((type.flags & ts.TypeFlags.StringLike) !== 0) {\n return { type: \"string\" };\n }\n if ((type.flags & ts.TypeFlags.NumberLike) !== 0) {\n return { type: \"number\" };\n }\n if ((type.flags & ts.TypeFlags.BooleanLike) !== 0) {\n return { type: \"boolean\" };\n }\n\n if ((type.flags & ts.TypeFlags.Object) !== 0) {\n return { type: \"object\" };\n }\n\n return { type: \"custom\" };\n}\n\nfunction resolveComponentSignature(\n exportName: string,\n declarations: ReturnType<typeof collectTopLevelDeclarations>,\n sourceFile: ts.SourceFile\n): ResolvedComponentSignature {\n const visitedNames = new Set<string>();\n\n const typeNodeFromFunction = (\n node: ts.FunctionLikeDeclarationBase\n ): ResolvedComponentSignature => ({\n propsTypeNode: node.parameters[0]?.type ?? null,\n componentNode: node,\n });\n\n const resolveFromExpression = (expression: ts.Expression): ResolvedComponentSignature => {\n if (ts.isParenthesizedExpression(expression)) {\n return resolveFromExpression(expression.expression);\n }\n if (ts.isAsExpression(expression) || ts.isTypeAssertionExpression(expression)) {\n return resolveFromExpression(expression.expression);\n }\n if (ts.isArrowFunction(expression) || ts.isFunctionExpression(expression)) {\n return typeNodeFromFunction(expression);\n }\n if (ts.isIdentifier(expression)) {\n return resolveFromIdentifier(expression.text);\n }\n\n if (ts.isCallExpression(expression)) {\n if (\n ts.isPropertyAccessExpression(expression.expression) &&\n expression.expression.name.text === \"forwardRef\"\n ) {\n const forwardRefPropsType = expression.typeArguments?.[1] ?? null;\n const innerArg = expression.arguments[0];\n const inner = innerArg && (ts.isArrowFunction(innerArg) || ts.isFunctionExpression(innerArg))\n ? typeNodeFromFunction(innerArg)\n : innerArg && ts.isIdentifier(innerArg)\n ? resolveFromIdentifier(innerArg.text)\n : { propsTypeNode: null, componentNode: null };\n\n return {\n propsTypeNode: forwardRefPropsType ?? inner.propsTypeNode,\n componentNode: inner.componentNode,\n };\n }\n\n if (\n ts.isPropertyAccessExpression(expression.expression) &&\n expression.expression.name.text === \"memo\" &&\n expression.arguments[0]\n ) {\n return resolveFromExpression(expression.arguments[0]);\n }\n\n if (\n ts.isPropertyAccessExpression(expression.expression) &&\n expression.expression.expression.getText(sourceFile) === \"Object\" &&\n expression.expression.name.text === \"assign\" &&\n expression.arguments[0]\n ) {\n return resolveFromExpression(expression.arguments[0]);\n }\n }\n\n return { propsTypeNode: null, componentNode: null };\n };\n\n const resolveFromVariable = (declaration: ts.VariableDeclaration): ResolvedComponentSignature => {\n if (\n declaration.type &&\n ts.isTypeReferenceNode(declaration.type) &&\n declaration.type.typeArguments?.length\n ) {\n const typeName = declaration.type.typeName.getText(sourceFile);\n if (typeName.includes(\"FC\") || typeName.includes(\"FunctionComponent\")) {\n const componentNode =\n declaration.initializer &&\n (ts.isArrowFunction(declaration.initializer) || ts.isFunctionExpression(declaration.initializer))\n ? declaration.initializer\n : null;\n\n return {\n propsTypeNode: declaration.type.typeArguments[0] ?? null,\n componentNode,\n };\n }\n }\n\n if (declaration.initializer) {\n return resolveFromExpression(declaration.initializer);\n }\n\n return { propsTypeNode: null, componentNode: null };\n };\n\n const resolveFromIdentifier = (name: string): ResolvedComponentSignature => {\n if (!name || visitedNames.has(name)) {\n return { propsTypeNode: null, componentNode: null };\n }\n visitedNames.add(name);\n\n const functionDeclaration = declarations.functionDeclarations.get(name);\n if (functionDeclaration) {\n return typeNodeFromFunction(functionDeclaration);\n }\n\n const variableDeclaration = declarations.variableDeclarations.get(name);\n if (variableDeclaration) {\n return resolveFromVariable(variableDeclaration);\n }\n\n return { propsTypeNode: null, componentNode: null };\n };\n\n return resolveFromIdentifier(exportName);\n}\n\n/**\n * Extract custom component props from a source file.\n * Custom props are identified as props whose declarations originate from the component's source file.\n */\nexport function extractCustomPropsFromComponentFile(\n componentFilePath: string,\n exportName: string\n): AutoPropsExtractionResult {\n const warnings: string[] = [];\n const resolvedPath = resolve(componentFilePath);\n\n if (!existsSync(resolvedPath)) {\n return {\n props: {},\n warnings: [`Component file not found: ${resolvedPath}`],\n resolved: false,\n };\n }\n\n const compilerOptions: ts.CompilerOptions = {\n target: ts.ScriptTarget.ESNext,\n module: ts.ModuleKind.ESNext,\n moduleResolution: ts.ModuleResolutionKind.Bundler,\n jsx: ts.JsxEmit.ReactJSX,\n allowSyntheticDefaultImports: true,\n esModuleInterop: true,\n skipLibCheck: true,\n strict: false,\n noEmit: true,\n };\n\n const program = ts.createProgram([resolvedPath], compilerOptions);\n const sourceFile = program.getSourceFile(resolvedPath);\n if (!sourceFile) {\n return {\n props: {},\n warnings: [`Unable to parse component source: ${resolvedPath}`],\n resolved: false,\n };\n }\n\n const checker = program.getTypeChecker();\n const declarations = collectTopLevelDeclarations(sourceFile);\n const signature = resolveComponentSignature(exportName, declarations, sourceFile);\n\n if (!signature.propsTypeNode) {\n return {\n props: {},\n warnings: [`Unable to resolve props type for export: ${exportName}`],\n resolved: false,\n };\n }\n\n const propsType = checker.getTypeFromTypeNode(signature.propsTypeNode);\n const defaultValues = extractDefaultValues(signature.componentNode);\n const sourceFilePath = toPosixPath(sourceFile.fileName);\n\n const extractedProps: Record<string, AutoDetectedPropDefinition> = {};\n for (const symbol of checker.getPropertiesOfType(propsType)) {\n const propName = symbol.getName();\n if (propName.startsWith(\"_\") || propName.startsWith(\"$\")) {\n continue;\n }\n\n const declarationsForSymbol = symbol.getDeclarations() ?? [];\n const localDeclarations = declarationsForSymbol.filter(\n (declaration) => toPosixPath(declaration.getSourceFile().fileName) === sourceFilePath\n );\n\n if (localDeclarations.length === 0) {\n continue;\n }\n\n const referenceNode = localDeclarations[0];\n const inferredType = inferPropType(checker.getTypeOfSymbolAtLocation(symbol, referenceNode), checker);\n const description = ts\n .displayPartsToString(symbol.getDocumentationComment(checker))\n .trim();\n\n extractedProps[propName] = {\n type: inferredType.type,\n description,\n required: (symbol.getFlags() & ts.SymbolFlags.Optional) === 0,\n ...(inferredType.values && { values: inferredType.values }),\n ...(defaultValues[propName] !== undefined && { default: defaultValues[propName] }),\n };\n }\n\n if (Object.keys(extractedProps).length === 0) {\n warnings.push(`Resolved props type for ${exportName}, but no local custom props were found`);\n }\n\n return {\n props: extractedProps,\n warnings,\n resolved: true,\n };\n}\n","/**\n * Graph extraction pipeline — builds the ComponentGraph from source code,\n * fragment metadata, and block definitions.\n *\n * Absorbs and enhances the logic from importAnalyzer.ts:\n * - Import detection (PascalCase component imports)\n * - Hook dependency detection (useX() calls)\n * - Object.assign sub-component detection\n * - JSX usage in variant code\n * - Block co-occurrence\n * - Fragment relation mapping\n * - Auto-detection of requiredChildren and commonPatterns\n */\n\nimport ts from 'typescript';\nimport { readFileSync, existsSync } from 'node:fs';\nimport { resolve, join, basename } from 'node:path';\nimport { readdirSync } from 'node:fs';\nimport type {\n ComponentGraph,\n ComponentNode,\n GraphEdge,\n GraphEdgeType,\n GraphHealth,\n} from '@fragments-sdk/context/graph';\nimport { EDGE_TYPE_WEIGHTS, computeHealthFromData } from '@fragments-sdk/context/graph';\nimport type { CompiledFragment, CompiledBlock } from './types.js';\n\n// ---------------------------------------------------------------------------\n// Public API\n// ---------------------------------------------------------------------------\n\nexport interface GraphBuildOptions {\n /** Skip source-code AST analysis (for testing with mock data) */\n skipSourceAnalysis?: boolean;\n}\n\nexport interface GraphBuildResult {\n graph: ComponentGraph;\n /** Auto-detected metadata keyed by component name */\n autoDetected: Map<string, AutoDetectedMetadata>;\n /** Warnings about drift between manual declarations and auto-detected values */\n warnings: string[];\n}\n\nexport interface AutoDetectedMetadata {\n subComponents?: string[];\n compositionPattern?: 'compound' | 'simple' | 'controlled';\n commonPatterns?: string[];\n requiredChildren?: string[];\n}\n\n/**\n * Build the full ComponentGraph from fragments, blocks, and source code.\n */\nexport async function buildComponentGraph(\n fragments: Record<string, CompiledFragment>,\n blocks: Record<string, CompiledBlock>,\n componentDir: string,\n options?: GraphBuildOptions,\n): Promise<GraphBuildResult> {\n const knownComponents = new Set(Object.keys(fragments));\n const allEdges: GraphEdge[] = [];\n const autoDetected = new Map<string, AutoDetectedMetadata>();\n const warnings: string[] = [];\n\n // 1. Extract edges from source code (imports, hooks, sub-components)\n if (!options?.skipSourceAnalysis) {\n const sourceEdges = extractImportAndHookEdges(componentDir, knownComponents);\n allEdges.push(...sourceEdges);\n\n const subComponentResults = extractSubComponents(componentDir, knownComponents);\n for (const [name, subs] of subComponentResults) {\n autoDetected.set(name, {\n ...autoDetected.get(name),\n subComponents: subs,\n compositionPattern: subs.length > 0 ? 'compound' : 'simple',\n });\n }\n }\n\n // 2. Extract JSX usage from variant code\n const jsxEdges = extractJsxUsageEdges(fragments, knownComponents);\n allEdges.push(...jsxEdges);\n\n // 3. Extract block co-occurrence edges\n const blockEdges = extractBlockEdges(blocks);\n allEdges.push(...blockEdges);\n\n // 4. Extract relation edges from fragment metadata\n const relationEdges = extractRelationEdges(fragments);\n allEdges.push(...relationEdges);\n\n // 5. Infer requiredChildren\n const requiredChildrenMap = inferRequiredChildren(fragments, autoDetected);\n for (const [name, children] of requiredChildrenMap) {\n const existing = autoDetected.get(name) ?? {};\n autoDetected.set(name, { ...existing, requiredChildren: children });\n }\n\n // 6. Generate common patterns\n const patternsMap = generateCommonPatterns(fragments, autoDetected);\n for (const [name, patterns] of patternsMap) {\n const existing = autoDetected.get(name) ?? {};\n autoDetected.set(name, { ...existing, commonPatterns: patterns });\n }\n\n // 7. Merge and deduplicate edges\n const mergedEdges = mergeAndDeduplicate(allEdges);\n\n // 8. Build nodes\n const nodes: ComponentNode[] = Object.entries(fragments).map(([name, fragment]) => {\n const detected = autoDetected.get(name);\n return {\n name,\n category: fragment.meta.category,\n status: fragment.meta.status ?? 'stable',\n compositionPattern: fragment.ai?.compositionPattern ?? detected?.compositionPattern,\n subComponents: fragment.ai?.subComponents ?? detected?.subComponents,\n };\n });\n\n // 9. Build block index for health computation\n const blockIndex = new Map<string, string[]>();\n for (const [blockName, block] of Object.entries(blocks)) {\n for (const comp of block.components) {\n const existing = blockIndex.get(comp);\n if (existing) existing.push(blockName);\n else blockIndex.set(comp, [blockName]);\n }\n }\n\n // 10. Compute health\n const health = computeHealthFromData(nodes, mergedEdges, blockIndex);\n\n // 11. Generate drift warnings\n for (const [name, fragment] of Object.entries(fragments)) {\n const detected = autoDetected.get(name);\n if (!detected) continue;\n\n // Sub-components drift\n if (fragment.ai?.subComponents && detected.subComponents) {\n const declared = new Set(fragment.ai.subComponents);\n const found = new Set(detected.subComponents);\n const missing = detected.subComponents.filter(s => !declared.has(s));\n const extra = fragment.ai.subComponents.filter(s => !found.has(s));\n\n if (missing.length > 0) {\n warnings.push(\n `${name}: declares ${declared.size} subComponents but code has ${found.size}. ` +\n `Missing from declaration: ${missing.join(', ')}`\n );\n }\n if (extra.length > 0) {\n warnings.push(\n `${name}: declares subComponents [${extra.join(', ')}] not found in Object.assign`\n );\n }\n }\n }\n\n return {\n graph: { nodes, edges: mergedEdges, health },\n autoDetected,\n warnings,\n };\n}\n\n// ---------------------------------------------------------------------------\n// Extraction functions\n// ---------------------------------------------------------------------------\n\n/**\n * Parse index.tsx files for import declarations and hook calls.\n * Returns import edges and hook-depends edges.\n */\nexport function extractImportAndHookEdges(\n componentDir: string,\n knownComponents: Set<string>,\n): GraphEdge[] {\n const edges: GraphEdge[] = [];\n\n for (const componentName of knownComponents) {\n const indexPath = findComponentIndex(componentDir, componentName);\n if (!indexPath) continue;\n\n let sourceText: string;\n try {\n sourceText = readFileSync(indexPath, 'utf-8');\n } catch {\n continue;\n }\n\n const sourceFile = ts.createSourceFile(\n indexPath,\n sourceText,\n ts.ScriptTarget.Latest,\n true,\n indexPath.endsWith('.tsx') ? ts.ScriptKind.TSX : ts.ScriptKind.TS,\n );\n\n // Walk AST for imports and hook calls\n const visitNode = (node: ts.Node) => {\n // Import declarations\n if (ts.isImportDeclaration(node)) {\n const moduleSpecifier = node.moduleSpecifier;\n if (ts.isStringLiteral(moduleSpecifier)) {\n const importPath = moduleSpecifier.text;\n // Only relative imports\n if (importPath.startsWith('.') || importPath.startsWith('/')) {\n const clause = node.importClause;\n if (clause) {\n // Default import\n if (clause.name && isPascalCase(clause.name.text) && knownComponents.has(clause.name.text)) {\n edges.push({\n source: componentName,\n target: clause.name.text,\n type: 'imports',\n weight: EDGE_TYPE_WEIGHTS['imports'],\n provenance: `source:${componentName}/index.tsx`,\n });\n }\n // Named imports\n if (clause.namedBindings && ts.isNamedImports(clause.namedBindings)) {\n for (const element of clause.namedBindings.elements) {\n const name = element.name.text;\n if (isPascalCase(name) && knownComponents.has(name) && name !== componentName) {\n edges.push({\n source: componentName,\n target: name,\n type: 'imports',\n weight: EDGE_TYPE_WEIGHTS['imports'],\n provenance: `source:${componentName}/index.tsx`,\n });\n }\n }\n }\n }\n }\n }\n }\n\n // Hook calls: useX() where X is a known component\n if (ts.isCallExpression(node) && ts.isIdentifier(node.expression)) {\n const callName = node.expression.text;\n const hookMatch = callName.match(/^use([A-Z][a-zA-Z]*)$/);\n if (hookMatch) {\n const hookTarget = hookMatch[1];\n if (knownComponents.has(hookTarget) && hookTarget !== componentName) {\n edges.push({\n source: componentName,\n target: hookTarget,\n type: 'hook-depends',\n weight: EDGE_TYPE_WEIGHTS['hook-depends'],\n provenance: `source:${componentName}/index.tsx`,\n });\n }\n }\n }\n\n ts.forEachChild(node, visitNode);\n };\n\n ts.forEachChild(sourceFile, visitNode);\n }\n\n return edges;\n}\n\n/**\n * Detect Object.assign(Root, { Sub1, Sub2 }) patterns in index.tsx files.\n * Returns a map of component name → sub-component names.\n */\nexport function extractSubComponents(\n componentDir: string,\n knownComponents: Set<string>,\n): Map<string, string[]> {\n const result = new Map<string, string[]>();\n\n for (const componentName of knownComponents) {\n const indexPath = findComponentIndex(componentDir, componentName);\n if (!indexPath) continue;\n\n let sourceText: string;\n try {\n sourceText = readFileSync(indexPath, 'utf-8');\n } catch {\n continue;\n }\n\n // Quick regex check first for performance\n if (!sourceText.includes('Object.assign')) continue;\n\n const sourceFile = ts.createSourceFile(\n indexPath,\n sourceText,\n ts.ScriptTarget.Latest,\n true,\n indexPath.endsWith('.tsx') ? ts.ScriptKind.TSX : ts.ScriptKind.TS,\n );\n\n const subComponents: string[] = [];\n\n const visitNode = (node: ts.Node) => {\n // Look for Object.assign(X, { A, B, C })\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 propsArg = node.arguments[1];\n if (ts.isObjectLiteralExpression(propsArg)) {\n for (const prop of propsArg.properties) {\n if (ts.isShorthandPropertyAssignment(prop)) {\n subComponents.push(prop.name.text);\n } else if (ts.isPropertyAssignment(prop) && ts.isIdentifier(prop.name)) {\n subComponents.push(prop.name.text);\n }\n }\n }\n }\n ts.forEachChild(node, visitNode);\n };\n\n ts.forEachChild(sourceFile, visitNode);\n\n if (subComponents.length > 0) {\n result.set(componentName, subComponents);\n }\n }\n\n return result;\n}\n\n/**\n * Scan variant code strings for JSX element names → renders edges.\n */\nexport function extractJsxUsageEdges(\n fragments: Record<string, CompiledFragment>,\n knownComponents: Set<string>,\n): GraphEdge[] {\n const edges: GraphEdge[] = [];\n const jsxTagRegex = /<([A-Z][a-zA-Z]*(?:\\.[A-Z][a-zA-Z]*)?)/g;\n\n for (const [name, fragment] of Object.entries(fragments)) {\n const usedComponents = new Set<string>();\n\n for (const variant of fragment.variants) {\n if (!variant.code) continue;\n\n let match: RegExpExecArray | null;\n jsxTagRegex.lastIndex = 0;\n while ((match = jsxTagRegex.exec(variant.code)) !== null) {\n let tagName = match[1];\n // Handle compound: Header.Nav → Header\n if (tagName.includes('.')) {\n tagName = tagName.split('.')[0];\n }\n if (knownComponents.has(tagName) && tagName !== name) {\n usedComponents.add(tagName);\n }\n }\n }\n\n for (const target of usedComponents) {\n edges.push({\n source: name,\n target,\n type: 'renders',\n weight: EDGE_TYPE_WEIGHTS['renders'],\n provenance: `variant:${name}`,\n });\n }\n }\n\n return edges;\n}\n\n/**\n * Create composes edges from block component lists (pairwise).\n */\nexport function extractBlockEdges(\n blocks: Record<string, CompiledBlock>,\n): GraphEdge[] {\n const edges: GraphEdge[] = [];\n\n for (const [blockName, block] of Object.entries(blocks)) {\n const components = block.components;\n // Create pairwise edges (avoid duplicates by only going i < j)\n for (let i = 0; i < components.length; i++) {\n for (let j = i + 1; j < components.length; j++) {\n edges.push({\n source: components[i],\n target: components[j],\n type: 'composes',\n weight: EDGE_TYPE_WEIGHTS['composes'],\n provenance: `block:${blockName}`,\n });\n }\n }\n }\n\n return edges;\n}\n\n/**\n * Map fragment ComponentRelation[] to typed graph edges.\n */\nexport function extractRelationEdges(\n fragments: Record<string, CompiledFragment>,\n): GraphEdge[] {\n const edges: GraphEdge[] = [];\n\n const relationToEdgeType: Record<string, GraphEdgeType> = {\n parent: 'parent-of',\n child: 'parent-of', // reversed: if A declares child B, edge is A parent-of B\n composition: 'composes',\n alternative: 'alternative-to',\n sibling: 'sibling-of',\n };\n\n for (const [name, fragment] of Object.entries(fragments)) {\n if (!fragment.relations) continue;\n\n for (const rel of fragment.relations) {\n const edgeType = relationToEdgeType[rel.relationship];\n if (!edgeType) continue;\n\n // For 'child' relation, source is the current component (parent)\n // For 'parent' relation, source is the related component\n let source: string;\n let target: string;\n\n if (rel.relationship === 'parent') {\n source = rel.component;\n target = name;\n } else {\n source = name;\n target = rel.component;\n }\n\n edges.push({\n source,\n target,\n type: edgeType,\n weight: EDGE_TYPE_WEIGHTS[edgeType],\n note: rel.note,\n provenance: 'relation',\n });\n }\n }\n\n return edges;\n}\n\n/**\n * Infer requiredChildren: sub-components that appear in ALL variant code strings.\n */\nexport function inferRequiredChildren(\n fragments: Record<string, CompiledFragment>,\n autoDetected: Map<string, AutoDetectedMetadata>,\n): Map<string, string[]> {\n const result = new Map<string, string[]>();\n\n for (const [name, fragment] of Object.entries(fragments)) {\n const detected = autoDetected.get(name);\n const subs = detected?.subComponents ?? fragment.ai?.subComponents;\n if (!subs || subs.length === 0) continue;\n\n const variantsWithCode = fragment.variants.filter(v => v.code);\n if (variantsWithCode.length === 0) continue;\n\n const required: string[] = [];\n for (const sub of subs) {\n // Check if this sub-component appears in ALL variants\n const inAll = variantsWithCode.every(v => {\n // Match <ComponentName.SubName or just <SubName\n const patterns = [\n new RegExp(`<${name}\\\\.${sub}[\\\\s/>]`),\n new RegExp(`<${sub}[\\\\s/>]`),\n ];\n return patterns.some(p => p.test(v.code!));\n });\n if (inAll) required.push(sub);\n }\n\n if (required.length > 0) {\n result.set(name, required);\n }\n }\n\n return result;\n}\n\n/**\n * Generate simplified JSX skeleton patterns from variant code.\n */\nexport function generateCommonPatterns(\n fragments: Record<string, CompiledFragment>,\n autoDetected: Map<string, AutoDetectedMetadata>,\n): Map<string, string[]> {\n const result = new Map<string, string[]>();\n\n for (const [name, fragment] of Object.entries(fragments)) {\n const detected = autoDetected.get(name);\n const subs = detected?.subComponents ?? fragment.ai?.subComponents;\n if (!subs || subs.length === 0) continue;\n\n // Build a simplified pattern from the first variant that has code\n const firstVariant = fragment.variants.find(v => v.code);\n if (!firstVariant?.code) continue;\n\n // Extract used sub-components from the code\n const usedSubs: string[] = [];\n for (const sub of subs) {\n const patterns = [\n new RegExp(`<${name}\\\\.${sub}`),\n new RegExp(`<${sub}[\\\\s/>]`),\n ];\n if (patterns.some(p => p.test(firstVariant.code!))) {\n usedSubs.push(sub);\n }\n }\n\n if (usedSubs.length > 0) {\n const pattern = `<${name}>\\n${usedSubs.map(s => ` <${name}.${s}>...</${name}.${s}>`).join('\\n')}\\n</${name}>`;\n result.set(name, [pattern]);\n }\n }\n\n return result;\n}\n\n/**\n * Merge edges: key by (source, target, type), keep highest weight.\n */\nexport function mergeAndDeduplicate(edges: GraphEdge[]): GraphEdge[] {\n const edgeMap = new Map<string, GraphEdge>();\n\n for (const edge of edges) {\n const key = `${edge.source}→${edge.target}:${edge.type}`;\n const existing = edgeMap.get(key);\n if (!existing || edge.weight > existing.weight) {\n edgeMap.set(key, edge);\n }\n }\n\n return [...edgeMap.values()];\n}\n\n// ---------------------------------------------------------------------------\n// Helpers\n// ---------------------------------------------------------------------------\n\nfunction isPascalCase(name: string): boolean {\n return /^[A-Z][a-zA-Z0-9]*$/.test(name);\n}\n\n/**\n * Find the index.tsx file for a component in the component directory.\n */\nfunction findComponentIndex(componentDir: string, componentName: string): string | null {\n // Try direct path: componentDir/ComponentName/index.tsx\n const candidates = [\n join(componentDir, componentName, 'index.tsx'),\n join(componentDir, componentName, 'index.ts'),\n join(componentDir, componentName, `${componentName}.tsx`),\n join(componentDir, componentName, `${componentName}.ts`),\n ];\n\n // Also search within subdirectories matching the component structure\n // e.g., src/components/ComponentName/index.tsx\n for (const candidate of candidates) {\n if (existsSync(candidate)) {\n return candidate;\n }\n }\n\n // Try scanning componentDir for directories matching the component name\n try {\n const entries = readdirSync(componentDir, { withFileTypes: true });\n for (const entry of entries) {\n if (entry.isDirectory() && entry.name === componentName) {\n const subCandidates = [\n join(componentDir, entry.name, 'index.tsx'),\n join(componentDir, entry.name, 'index.ts'),\n ];\n for (const sc of subCandidates) {\n if (existsSync(sc)) return sc;\n }\n }\n }\n } catch {\n // Directory might not exist\n }\n\n return null;\n}\n","import pc from 'picocolors';\nimport {\n BRAND,\n DEFAULTS,\n type FragmentsConfig,\n type FragmentDefinition,\n type Theme,\n} from './core/index.js';\nimport { discoverFragmentFiles, loadFragmentFile } from './core/node.js';\nimport {\n BrowserPool,\n CaptureEngine,\n StorageManager,\n formatMs,\n type CaptureOptions,\n} from './service/index.js';\n\n/**\n * Options for the screenshot command\n */\nexport interface ScreenshotCommandOptions {\n /** Specific component to capture */\n component?: string;\n\n /** Specific variant to capture */\n variant?: string;\n\n /** Theme to capture */\n theme?: Theme;\n\n /** Update existing baselines */\n update?: boolean;\n\n /** CI mode - no interactive prompts */\n ci?: boolean;\n\n /** Viewport width */\n width?: number;\n\n /** Viewport height */\n height?: number;\n}\n\n/**\n * Result of the screenshot command\n */\nexport interface ScreenshotResult {\n success: boolean;\n captured: number;\n skipped: number;\n errors: Array<{ component: string; variant: string; error: string }>;\n totalTimeMs: number;\n}\n\n/**\n * Execute the screenshot command\n */\nexport async function runScreenshotCommand(\n config: FragmentsConfig,\n configDir: string,\n options: ScreenshotCommandOptions = {}\n): Promise<ScreenshotResult> {\n const startTime = Date.now();\n const errors: ScreenshotResult['errors'] = [];\n\n // Initialize storage\n const storage = new StorageManager({\n projectRoot: configDir,\n viewport: options.width && options.height\n ? { width: options.width, height: options.height }\n : config.screenshots?.viewport,\n });\n await storage.initialize();\n\n // Discover fragments\n const fragmentFiles = await discoverFragmentFiles(config, configDir);\n\n if (fragmentFiles.length === 0) {\n console.log(pc.yellow('No fragment files found.'));\n return {\n success: true,\n captured: 0,\n skipped: 0,\n errors: [],\n totalTimeMs: Date.now() - startTime,\n };\n }\n\n // Load all fragments\n const fragments: Array<{ path: string; fragment: FragmentDefinition }> = [];\n\n for (const file of fragmentFiles) {\n try {\n const fragment = await loadFragmentFile(file.absolutePath);\n if (fragment) {\n fragments.push({ path: file.relativePath, fragment });\n }\n } catch (error) {\n errors.push({\n component: file.relativePath,\n variant: '',\n error: error instanceof Error ? error.message : String(error),\n });\n }\n }\n\n // Filter by component if specified\n const filteredFragments = options.component\n ? fragments.filter((s) => s.fragment.meta.name === options.component)\n : fragments;\n\n if (options.component && filteredFragments.length === 0) {\n console.log(pc.yellow(`Component \"${options.component}\" not found.`));\n return {\n success: false,\n captured: 0,\n skipped: 0,\n errors: [],\n totalTimeMs: Date.now() - startTime,\n };\n }\n\n // Build list of variants to capture\n const variantsToCapture: Array<{\n component: string;\n variant: string;\n render: () => unknown;\n }> = [];\n\n for (const { fragment } of filteredFragments) {\n const variants = options.variant\n ? fragment.variants.filter((v) => v.name === options.variant)\n : fragment.variants;\n\n for (const variant of variants) {\n variantsToCapture.push({\n component: fragment.meta.name,\n variant: variant.name,\n render: variant.render,\n });\n }\n }\n\n if (variantsToCapture.length === 0) {\n console.log(pc.yellow('No variants to capture.'));\n return {\n success: true,\n captured: 0,\n skipped: 0,\n errors: [],\n totalTimeMs: Date.now() - startTime,\n };\n }\n\n // Determine theme\n const theme: Theme = options.theme ?? DEFAULTS.theme;\n const viewport = {\n width: options.width ?? config.screenshots?.viewport?.width ?? DEFAULTS.viewport.width,\n height: options.height ?? config.screenshots?.viewport?.height ?? DEFAULTS.viewport.height,\n };\n\n console.log(pc.cyan(`\\n${BRAND.name} Screenshot\\n`));\n console.log(pc.dim(`Capturing variants (theme: ${theme}, viewport: ${viewport.width}x${viewport.height}):\\n`));\n\n // Initialize browser pool\n const pool = new BrowserPool({\n viewport,\n });\n\n // Create capture engine - use the viewer URL\n const viewerPort = DEFAULTS.port;\n const baseUrl = `http://localhost:${viewerPort}`;\n const captureEngine = new CaptureEngine(pool, baseUrl);\n\n let captured = 0;\n let skipped = 0;\n\n const captureOptions: CaptureOptions = {\n theme,\n viewport,\n delay: config.screenshots?.delay ?? DEFAULTS.captureDelayMs,\n };\n\n try {\n // Warm up the pool\n console.log(pc.dim('Starting browser...'));\n await pool.warmup();\n console.log(pc.dim('Browser ready.\\n'));\n\n // Capture each variant\n for (const { component, variant } of variantsToCapture) {\n const hasExisting = storage.hasBaseline(component, variant, theme);\n\n // Skip if exists and not updating\n if (hasExisting && !options.update) {\n console.log(` ${pc.dim('○')} ${component}/${variant} ${pc.dim('(skipped)')}`);\n skipped++;\n continue;\n }\n\n try {\n const screenshot = await captureEngine.captureVariant(\n component,\n variant,\n captureOptions\n );\n\n await storage.saveBaseline(screenshot);\n\n const totalTime = screenshot.metadata.renderTimeMs + screenshot.metadata.captureTimeMs;\n console.log(\n ` ${pc.green('✓')} ${component}/${variant} ${pc.dim(formatMs(totalTime))}`\n );\n captured++;\n } catch (error) {\n const errorMsg = error instanceof Error ? error.message : String(error);\n console.log(` ${pc.red('✗')} ${component}/${variant} ${pc.dim(errorMsg)}`);\n errors.push({ component, variant, error: errorMsg });\n }\n }\n } finally {\n // Shutdown browser pool\n await pool.shutdown();\n }\n\n const totalTimeMs = Date.now() - startTime;\n\n // Print summary\n console.log();\n if (errors.length === 0) {\n console.log(pc.green(`✓ Captured ${captured} screenshot(s) in ${formatMs(totalTimeMs)}`));\n } else {\n console.log(pc.yellow(`⚠ Captured ${captured} screenshot(s) with ${errors.length} error(s)`));\n }\n\n if (skipped > 0) {\n console.log(pc.dim(` ${skipped} skipped (use --update to recapture)`));\n }\n\n console.log(pc.dim(` Stored in ${storage.screenshotsDirPath}\\n`));\n\n return {\n success: errors.length === 0,\n captured,\n skipped,\n errors,\n totalTimeMs,\n };\n}\n","import pc from 'picocolors';\nimport {\n BRAND,\n DEFAULTS,\n type FragmentsConfig,\n type FragmentDefinition,\n type Theme,\n} from './core/index.js';\nimport { discoverFragmentFiles, loadFragmentFile } from './core/node.js';\nimport {\n BrowserPool,\n CaptureEngine,\n StorageManager,\n DiffEngine,\n formatMs,\n type CaptureOptions,\n type DiffResult,\n} from './service/index.js';\n\n/**\n * Options for the diff command\n */\nexport interface DiffCommandOptions {\n /** Specific component to diff */\n component?: string;\n\n /** Specific variant to diff */\n variant?: string;\n\n /** Theme to compare */\n theme?: Theme;\n\n /** CI mode - exit 1 on differences */\n ci?: boolean;\n\n /** Diff threshold percentage */\n threshold?: number;\n\n /** Open diff images */\n open?: boolean;\n}\n\n/**\n * Single diff result with metadata\n */\nexport interface VariantDiffResult {\n component: string;\n variant: string;\n theme: Theme;\n result: DiffResult;\n diffImagePath?: string;\n}\n\n/**\n * Result of the diff command\n */\nexport interface DiffCommandResult {\n success: boolean;\n total: number;\n passed: number;\n failed: number;\n missing: number;\n results: VariantDiffResult[];\n totalTimeMs: number;\n}\n\n/**\n * Execute the diff command\n */\nexport async function runDiffCommand(\n config: FragmentsConfig,\n configDir: string,\n options: DiffCommandOptions = {}\n): Promise<DiffCommandResult> {\n const startTime = Date.now();\n const results: VariantDiffResult[] = [];\n\n // Initialize storage\n const storage = new StorageManager({\n projectRoot: configDir,\n viewport: config.screenshots?.viewport,\n });\n await storage.initialize();\n\n // Initialize diff engine\n const threshold = options.threshold ?? config.screenshots?.threshold ?? DEFAULTS.diffThreshold;\n const diffEngine = new DiffEngine(threshold);\n\n // Discover fragments\n const fragmentFiles = await discoverFragmentFiles(config, configDir);\n\n if (fragmentFiles.length === 0) {\n console.log(pc.yellow('No fragment files found.'));\n return {\n success: true,\n total: 0,\n passed: 0,\n failed: 0,\n missing: 0,\n results: [],\n totalTimeMs: Date.now() - startTime,\n };\n }\n\n // Load all fragments\n const fragments: Array<{ path: string; fragment: FragmentDefinition }> = [];\n\n for (const file of fragmentFiles) {\n try {\n const fragment = await loadFragmentFile(file.absolutePath);\n if (fragment) {\n fragments.push({ path: file.relativePath, fragment });\n }\n } catch {\n // Skip failed loads\n }\n }\n\n // Filter by component if specified\n const filteredFragments = options.component\n ? fragments.filter((s) => s.fragment.meta.name === options.component)\n : fragments;\n\n if (options.component && filteredFragments.length === 0) {\n console.log(pc.yellow(`Component \"${options.component}\" not found.`));\n return {\n success: false,\n total: 0,\n passed: 0,\n failed: 0,\n missing: 0,\n results: [],\n totalTimeMs: Date.now() - startTime,\n };\n }\n\n // Build list of variants to diff\n const variantsToDiff: Array<{\n component: string;\n variant: string;\n }> = [];\n\n for (const { fragment } of filteredFragments) {\n const variants = options.variant\n ? fragment.variants.filter((v) => v.name === options.variant)\n : fragment.variants;\n\n for (const variant of variants) {\n variantsToDiff.push({\n component: fragment.meta.name,\n variant: variant.name,\n });\n }\n }\n\n if (variantsToDiff.length === 0) {\n console.log(pc.yellow('No variants to compare.'));\n return {\n success: true,\n total: 0,\n passed: 0,\n failed: 0,\n missing: 0,\n results: [],\n totalTimeMs: Date.now() - startTime,\n };\n }\n\n // Determine theme\n const theme: Theme = options.theme ?? DEFAULTS.theme;\n const viewport = config.screenshots?.viewport ?? DEFAULTS.viewport;\n\n console.log(pc.cyan(`\\n${BRAND.name} Diff\\n`));\n console.log(pc.dim(`Comparing against baselines (theme: ${theme}, threshold: ${threshold}%):\\n`));\n\n // Initialize browser pool\n const pool = new BrowserPool({\n viewport,\n });\n\n // Create capture engine\n const viewerPort = DEFAULTS.port;\n const baseUrl = `http://localhost:${viewerPort}`;\n const captureEngine = new CaptureEngine(pool, baseUrl);\n\n let passed = 0;\n let failed = 0;\n let missing = 0;\n\n const captureOptions: CaptureOptions = {\n theme,\n viewport,\n delay: config.screenshots?.delay ?? DEFAULTS.captureDelayMs,\n };\n\n try {\n // Warm up the pool\n await pool.warmup();\n\n // Compare each variant\n for (const { component, variant } of variantsToDiff) {\n // Load baseline\n const baseline = await storage.loadBaseline(component, variant, theme);\n\n if (!baseline) {\n console.log(\n ` ${pc.yellow('?')} ${component}/${variant} ${pc.dim('(no baseline)')}`\n );\n missing++;\n continue;\n }\n\n try {\n // Capture current\n const current = await captureEngine.captureVariant(\n component,\n variant,\n captureOptions\n );\n\n // Quick hash check\n if (diffEngine.areIdentical(current, baseline)) {\n console.log(` ${pc.green('✓')} ${component}/${variant} ${pc.dim('0.0%')}`);\n results.push({\n component,\n variant,\n theme,\n result: {\n matches: true,\n diffPercentage: 0,\n diffPixelCount: 0,\n totalPixels: current.viewport.width * current.viewport.height,\n changedRegions: [],\n diffTimeMs: 0,\n },\n });\n passed++;\n continue;\n }\n\n // Full diff\n const diffResult = diffEngine.compare(current, baseline, { threshold });\n\n if (diffResult.matches) {\n console.log(\n ` ${pc.green('✓')} ${component}/${variant} ${pc.dim(`${diffResult.diffPercentage}%`)}`\n );\n passed++;\n } else {\n // Save diff image\n let diffImagePath: string | undefined;\n if (diffResult.diffImage) {\n diffImagePath = await storage.saveDiff(\n component,\n variant,\n theme,\n diffResult.diffImage\n );\n }\n\n console.log(\n ` ${pc.red('✗')} ${component}/${variant} ${pc.yellow(`${diffResult.diffPercentage}%`)}` +\n (diffImagePath ? pc.dim(` → ${diffImagePath}`) : '')\n );\n failed++;\n\n results.push({\n component,\n variant,\n theme,\n result: diffResult,\n diffImagePath,\n });\n continue;\n }\n\n results.push({\n component,\n variant,\n theme,\n result: diffResult,\n });\n } catch (error) {\n const errorMsg = error instanceof Error ? error.message : String(error);\n console.log(` ${pc.red('!')} ${component}/${variant} ${pc.dim(errorMsg)}`);\n failed++;\n }\n }\n } finally {\n // Shutdown browser pool\n await pool.shutdown();\n }\n\n const totalTimeMs = Date.now() - startTime;\n const total = passed + failed + missing;\n\n // Print summary\n console.log();\n if (failed === 0 && missing === 0) {\n console.log(pc.green(`✓ All ${passed} variant(s) match baselines`));\n } else if (failed > 0) {\n console.log(pc.red(`✗ ${failed} variant(s) differ from baselines`));\n }\n\n if (missing > 0) {\n console.log(pc.yellow(` ${missing} variant(s) have no baseline (run \\`${BRAND.cliCommand} screenshot\\`)`));\n }\n\n console.log(pc.dim(` Completed in ${formatMs(totalTimeMs)}\\n`));\n\n // In CI mode, exit with error if any failed\n const success = failed === 0;\n\n return {\n success,\n total,\n passed,\n failed,\n missing,\n results,\n totalTimeMs,\n };\n}\n","/**\n * CLI command: fragments analyze\n *\n * Analyzes the design system and generates an HTML report with:\n * - Component inventory\n * - Documentation coverage\n * - Quality insights\n * - Actionable recommendations\n */\n\nimport { existsSync } from \"node:fs\";\nimport { readFile, writeFile, mkdir } from \"node:fs/promises\";\nimport { join, dirname } from \"node:path\";\nimport pc from \"picocolors\";\nimport type { FragmentsConfig, CompiledFragmentsFile } from \"./core/index.js\";\nimport { BRAND, DEFAULTS } from \"./core/index.js\";\nimport {\n analyzeDesignSystem,\n generateHtmlReport,\n getGrade,\n getScoreColor,\n type DesignSystemAnalytics,\n} from \"./service/index.js\";\n\nexport interface AnalyzeOptions {\n /** Output format */\n format?: \"html\" | \"json\" | \"console\";\n /** Output file path (default: fragments-report.html) */\n output?: string;\n /** Open report in browser after generation */\n open?: boolean;\n /** CI mode - exit with appropriate code */\n ci?: boolean;\n /** Minimum score to pass in CI mode */\n minScore?: number;\n}\n\nexport interface AnalyzeResult {\n success: boolean;\n analytics: DesignSystemAnalytics;\n outputPath?: string;\n}\n\n/**\n * Run the analyze command\n */\nexport async function runAnalyzeCommand(\n config: FragmentsConfig,\n configDir: string,\n options: AnalyzeOptions = {}\n): Promise<AnalyzeResult> {\n const format = options.format ?? \"html\";\n const minScore = options.minScore ?? 0;\n\n console.log(pc.cyan(`\\n${BRAND.name} Analyzer\\n`));\n\n // Load compiled fragments\n const fragmentsPath = join(configDir, config.outFile ?? \"fragments.json\");\n\n if (!existsSync(fragmentsPath)) {\n console.log(pc.red(`✗ No fragments.json found. Run \\`${BRAND.cliCommand} build\\` first.\\n`));\n return {\n success: false,\n analytics: createEmptyAnalytics(),\n };\n }\n\n console.log(pc.dim(\"Analyzing design system...\\n\"));\n\n const content = await readFile(fragmentsPath, \"utf-8\");\n const data: CompiledFragmentsFile = JSON.parse(content);\n\n // Run analysis\n const analytics = analyzeDesignSystem(data);\n\n // Print summary to console\n printConsoleSummary(analytics);\n\n // Generate output based on format\n let outputPath: string | undefined;\n\n if (format === \"html\" || format === \"json\") {\n outputPath = options.output ?? getDefaultOutputPath(format, configDir);\n\n // Ensure output directory exists\n await mkdir(dirname(outputPath), { recursive: true });\n\n if (format === \"html\") {\n const html = generateHtmlReport(analytics);\n await writeFile(outputPath, html);\n console.log(pc.green(`✓ Report generated: ${outputPath}\\n`));\n } else {\n await writeFile(outputPath, JSON.stringify(analytics, null, 2));\n console.log(pc.green(`✓ JSON report generated: ${outputPath}\\n`));\n }\n\n // Open in browser if requested\n if (options.open && format === \"html\") {\n await openInBrowser(outputPath);\n }\n }\n\n // CI mode - check score\n const passedCi = analytics.summary.overallScore >= minScore;\n\n if (options.ci) {\n if (passedCi) {\n console.log(\n pc.green(`✓ Score ${analytics.summary.overallScore} meets minimum threshold ${minScore}\\n`)\n );\n } else {\n console.log(\n pc.red(\n `✗ Score ${analytics.summary.overallScore} below minimum threshold ${minScore}\\n`\n )\n );\n }\n }\n\n return {\n success: !options.ci || passedCi,\n analytics,\n outputPath,\n };\n}\n\n/**\n * Print a summary to the console\n */\nfunction printConsoleSummary(analytics: DesignSystemAnalytics): void {\n const { summary, coverage, recommendations } = analytics;\n const grade = getGrade(summary.overallScore);\n\n // Score header\n console.log(\n pc.bold(\n `Overall Score: ${colorizeScore(summary.overallScore)} (${grade})\\n`\n )\n );\n\n // Summary stats\n console.log(pc.dim(\"Summary\"));\n console.log(` Components: ${pc.white(summary.totalComponents.toString())}`);\n console.log(` Variants: ${pc.white(summary.totalVariants.toString())}`);\n console.log(` Props: ${pc.white(summary.totalProps.toString())}`);\n console.log(` Categories: ${pc.white(summary.categories.join(\", \"))}`);\n console.log();\n\n // Coverage\n console.log(pc.dim(\"Coverage\"));\n console.log(` Description: ${formatCoverage(coverage.fields.description)}`);\n console.log(` Usage when: ${formatCoverage(coverage.fields.usageWhen)}`);\n console.log(` Usage whenNot:${formatCoverage(coverage.fields.usageWhenNot)}`);\n console.log(` Guidelines: ${formatCoverage(coverage.fields.guidelines)}`);\n console.log(` Relations: ${formatCoverage(coverage.fields.relations)}`);\n console.log();\n\n // Top recommendations\n if (recommendations.length > 0) {\n console.log(pc.dim(\"Top Recommendations\"));\n for (const rec of recommendations.slice(0, 3)) {\n const priority = rec.priority === \"high\"\n ? pc.red(`[${rec.priority}]`)\n : rec.priority === \"medium\"\n ? pc.yellow(`[${rec.priority}]`)\n : pc.dim(`[${rec.priority}]`);\n console.log(` ${priority} ${rec.title}`);\n }\n console.log();\n }\n}\n\n/**\n * Format coverage percentage with color\n */\nfunction formatCoverage(field: { percentage: number; covered: number; total: number }): string {\n const pct = colorizeScore(field.percentage);\n return `${pct} (${field.covered}/${field.total})`;\n}\n\n/**\n * Colorize a score\n */\nfunction colorizeScore(score: number): string {\n if (score >= 80) return pc.green(`${score}%`);\n if (score >= 60) return pc.yellow(`${score}%`);\n return pc.red(`${score}%`);\n}\n\n/**\n * Get default output path\n */\nfunction getDefaultOutputPath(format: \"html\" | \"json\", configDir: string): string {\n const filename = format === \"html\" ? \"fragments-report.html\" : \"fragments-report.json\";\n return join(configDir, filename);\n}\n\n/**\n * Open file in default browser\n */\nasync function openInBrowser(path: string): Promise<void> {\n const { platform } = await import(\"node:os\");\n const { exec } = await import(\"node:child_process\");\n\n const os = platform();\n const cmd = os === \"darwin\"\n ? `open \"${path}\"`\n : os === \"win32\"\n ? `start \"\" \"${path}\"`\n : `xdg-open \"${path}\"`;\n\n exec(cmd);\n}\n\n/**\n * Create empty analytics for error cases\n */\nfunction createEmptyAnalytics(): DesignSystemAnalytics {\n return {\n analyzedAt: new Date(),\n summary: {\n totalComponents: 0,\n totalVariants: 0,\n totalProps: 0,\n categories: [],\n overallScore: 0,\n },\n inventory: {\n byCategory: {},\n byStatus: {},\n byVariantCount: [],\n byPropCount: [],\n },\n coverage: {\n overall: 0,\n fields: {\n description: { covered: 0, total: 0, percentage: 0 },\n usageWhen: { covered: 0, total: 0, percentage: 0 },\n usageWhenNot: { covered: 0, total: 0, percentage: 0 },\n guidelines: { covered: 0, total: 0, percentage: 0 },\n accessibility: { covered: 0, total: 0, percentage: 0 },\n relations: { covered: 0, total: 0, percentage: 0 },\n propDescriptions: { covered: 0, total: 0, percentage: 0 },\n propConstraints: { covered: 0, total: 0, percentage: 0 },\n },\n incomplete: [],\n },\n quality: {\n missingWhenNot: [],\n isolated: [],\n deprecated: [],\n fewVariants: [],\n undocumentedProps: [],\n unconstrainedProps: [],\n },\n distribution: {\n variantsPerComponent: [],\n propsPerComponent: [],\n componentsPerCategory: [],\n statusDistribution: [],\n tagFrequency: [],\n },\n recommendations: [],\n };\n}\n"],"mappings":"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","names":["resolve","join","existsSync","existsSync","dirname","resolve","ts","existsSync","join","readdirSync","resolve","existsSync","join","pc","pc","existsSync","readFile","writeFile","mkdir","join","dirname","pc","pc","join","existsSync","readFile","mkdir","dirname","writeFile"]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/core/config.ts","../src/core/generators/typescript-extractor.ts","../src/core/generators/registry.ts","../src/core/generators/context.ts","../src/core/loader.ts","../src/core/parser.ts","../src/core/previewLoader.ts","../src/core/importAnalyzer.ts"],"sourcesContent":["import { existsSync } from 'node:fs';\nimport { resolve, dirname } from 'node:path';\nimport { createJiti } from 'jiti';\nimport { BRAND } from './constants.js';\nimport type { FragmentsConfig } from './types.js';\nimport { fragmentsConfigSchema } from './schema.js';\n\nconst DEFAULT_CONFIG: FragmentsConfig = {\n include: [\n `src/**/*${BRAND.fileExtension}`, // *.fragment.tsx files\n 'src/**/*.stories.tsx', // Storybook stories (auto-converted)\n ],\n exclude: ['**/node_modules/**'],\n components: ['src/**/index.tsx', 'src/**/*.tsx'],\n framework: 'react',\n};\n\n/**\n * Find the config file in the current directory or parent directories.\n * Checks for both the current config file name and the legacy name.\n */\nexport function findConfigFile(startDir: string = process.cwd()): string | null {\n let currentDir = startDir;\n\n while (currentDir !== dirname(currentDir)) {\n // Check for current config file name first\n const configPath = resolve(currentDir, BRAND.configFile);\n if (existsSync(configPath)) {\n return configPath;\n }\n // Also check for legacy config file name\n const legacyConfigPath = resolve(currentDir, BRAND.legacyConfigFile);\n if (existsSync(legacyConfigPath)) {\n return legacyConfigPath;\n }\n currentDir = dirname(currentDir);\n }\n\n return null;\n}\n\n/**\n * Load and validate the config file\n */\nexport async function loadConfig(configPath?: string): Promise<{\n config: FragmentsConfig;\n configDir: string;\n}> {\n const resolvedPath = configPath ?? findConfigFile();\n\n if (!resolvedPath) {\n return {\n config: DEFAULT_CONFIG,\n configDir: process.cwd(),\n };\n }\n\n try {\n // Use jiti to load TypeScript config files\n const jiti = createJiti(import.meta.url, {\n interopDefault: true,\n });\n const rawConfig = await jiti.import(resolvedPath);\n\n const result = fragmentsConfigSchema.safeParse(rawConfig);\n\n if (!result.success) {\n const errors = result.error.errors\n .map((e) => ` - ${e.path.join('.')}: ${e.message}`)\n .join('\\n');\n throw new Error(`Invalid config in ${resolvedPath}:\\n${errors}`);\n }\n\n return {\n config: { ...DEFAULT_CONFIG, ...result.data },\n configDir: dirname(resolvedPath),\n };\n } catch (error) {\n if (error instanceof Error && error.message.includes('Invalid config')) {\n throw error;\n }\n throw new Error(`Failed to load config from ${resolvedPath}: ${error}`);\n }\n}\n","/**\n * TypeScript Props Extractor\n * Extracts prop types, default values, and JSDoc comments from React component files\n */\n\nimport ts from \"typescript\";\nimport { readFileSync } from \"node:fs\";\nimport type { RegistryPropEntry } from \"../fragment-types.js\";\n\n/**\n * Result of extracting props from a component file\n */\nexport interface ExtractedProps {\n /** Component name */\n componentName: string;\n /** Props interface name (e.g., \"ButtonProps\") */\n propsInterfaceName?: string;\n /** Extracted props */\n props: Record<string, RegistryPropEntry>;\n /** Named exports from the file */\n exports: string[];\n /** Import statements (for dependency tracking) */\n imports: string[];\n}\n\n/**\n * Extract props from a TypeScript/TSX component file\n */\nexport function extractPropsFromFile(filePath: string): ExtractedProps | null {\n const sourceText = readFileSync(filePath, \"utf-8\");\n return extractPropsFromSource(sourceText, filePath);\n}\n\n/**\n * Extract props from TypeScript source code\n */\nexport function extractPropsFromSource(\n sourceText: string,\n fileName = \"component.tsx\"\n): ExtractedProps | null {\n const sourceFile = ts.createSourceFile(\n fileName,\n sourceText,\n ts.ScriptTarget.Latest,\n true,\n fileName.endsWith(\".tsx\") ? ts.ScriptKind.TSX : ts.ScriptKind.TS\n );\n\n const result: ExtractedProps = {\n componentName: \"\",\n props: {},\n exports: [],\n imports: [],\n };\n\n // Find all exports and props interfaces\n const propsInterfaces = new Map<string, ts.InterfaceDeclaration | ts.TypeAliasDeclaration>();\n const componentExports: string[] = [];\n const importedModules: string[] = [];\n\n ts.forEachChild(sourceFile, (node) => {\n // Track imports\n if (ts.isImportDeclaration(node)) {\n const moduleSpecifier = node.moduleSpecifier;\n if (ts.isStringLiteral(moduleSpecifier)) {\n importedModules.push(moduleSpecifier.text);\n }\n }\n\n // Find interface declarations (e.g., interface ButtonProps { ... })\n if (ts.isInterfaceDeclaration(node)) {\n const name = node.name.text;\n if (name.endsWith(\"Props\")) {\n propsInterfaces.set(name, node);\n }\n }\n\n // Find type alias declarations (e.g., type ButtonProps = { ... })\n if (ts.isTypeAliasDeclaration(node)) {\n const name = node.name.text;\n if (name.endsWith(\"Props\")) {\n propsInterfaces.set(name, node);\n }\n }\n\n // Find exported functions/components\n if (ts.isFunctionDeclaration(node) && node.name) {\n const hasExport = node.modifiers?.some(\n (m) => m.kind === ts.SyntaxKind.ExportKeyword\n );\n if (hasExport) {\n componentExports.push(node.name.text);\n }\n }\n\n // Find exported variable declarations (e.g., export const Button = ...)\n if (ts.isVariableStatement(node)) {\n const hasExport = node.modifiers?.some(\n (m) => m.kind === ts.SyntaxKind.ExportKeyword\n );\n if (hasExport) {\n for (const decl of node.declarationList.declarations) {\n if (ts.isIdentifier(decl.name)) {\n componentExports.push(decl.name.text);\n }\n }\n }\n }\n\n // Find export declarations (e.g., export { Button })\n if (ts.isExportDeclaration(node) && node.exportClause) {\n if (ts.isNamedExports(node.exportClause)) {\n for (const element of node.exportClause.elements) {\n componentExports.push(element.name.text);\n }\n }\n }\n });\n\n result.exports = componentExports;\n result.imports = importedModules;\n\n // Find the main component (first PascalCase export)\n const mainComponent = componentExports.find(\n (name) => /^[A-Z]/.test(name) && !name.endsWith(\"Props\")\n );\n\n if (!mainComponent) {\n return null;\n }\n\n result.componentName = mainComponent;\n\n // Find matching props interface\n const propsInterfaceName = `${mainComponent}Props`;\n const propsInterface = propsInterfaces.get(propsInterfaceName);\n\n if (propsInterface) {\n result.propsInterfaceName = propsInterfaceName;\n result.props = extractPropsFromInterface(propsInterface, sourceFile);\n }\n\n return result;\n}\n\n/**\n * Extract props from an interface or type alias declaration\n */\nfunction extractPropsFromInterface(\n node: ts.InterfaceDeclaration | ts.TypeAliasDeclaration,\n sourceFile: ts.SourceFile\n): Record<string, RegistryPropEntry> {\n const props: Record<string, RegistryPropEntry> = {};\n\n // Handle interface declaration\n if (ts.isInterfaceDeclaration(node)) {\n for (const member of node.members) {\n if (ts.isPropertySignature(member) && member.name) {\n const propName = member.name.getText(sourceFile);\n const prop = extractPropFromMember(member, sourceFile);\n if (prop) {\n props[propName] = prop;\n }\n }\n }\n }\n\n // Handle type alias declaration\n if (ts.isTypeAliasDeclaration(node)) {\n const typeNode = node.type;\n if (ts.isTypeLiteralNode(typeNode)) {\n for (const member of typeNode.members) {\n if (ts.isPropertySignature(member) && member.name) {\n const propName = member.name.getText(sourceFile);\n const prop = extractPropFromMember(member, sourceFile);\n if (prop) {\n props[propName] = prop;\n }\n }\n }\n }\n }\n\n return props;\n}\n\n/**\n * Extract prop info from a property signature\n */\nfunction extractPropFromMember(\n member: ts.PropertySignature,\n sourceFile: ts.SourceFile\n): RegistryPropEntry | null {\n const entry: RegistryPropEntry = {};\n\n // Check if required\n entry.required = !member.questionToken;\n\n // Get type\n if (member.type) {\n const typeInfo = parseTypeNode(member.type, sourceFile);\n entry.type = typeInfo.type;\n entry.typeKind = typeInfo.typeKind;\n if (typeInfo.options) {\n entry.options = typeInfo.options;\n }\n }\n\n // Get JSDoc comment\n const jsDocComment = getJSDocComment(member);\n if (jsDocComment) {\n entry.description = jsDocComment;\n }\n\n // Get default value from JSDoc @default tag\n const defaultValue = getJSDocDefault(member);\n if (defaultValue !== undefined) {\n entry.default = defaultValue;\n }\n\n return entry;\n}\n\n/**\n * Parse a TypeScript type node into a simplified representation\n */\nfunction parseTypeNode(\n typeNode: ts.TypeNode,\n sourceFile: ts.SourceFile\n): { type: string; typeKind: RegistryPropEntry[\"typeKind\"]; options?: string[] } {\n // String\n if (typeNode.kind === ts.SyntaxKind.StringKeyword) {\n return { type: \"string\", typeKind: \"string\" };\n }\n\n // Number\n if (typeNode.kind === ts.SyntaxKind.NumberKeyword) {\n return { type: \"number\", typeKind: \"number\" };\n }\n\n // Boolean\n if (typeNode.kind === ts.SyntaxKind.BooleanKeyword) {\n return { type: \"boolean\", typeKind: \"boolean\" };\n }\n\n // Union type (string literal union for enums)\n if (ts.isUnionTypeNode(typeNode)) {\n const options: string[] = [];\n let allLiterals = true;\n\n for (const subType of typeNode.types) {\n if (ts.isLiteralTypeNode(subType)) {\n if (ts.isStringLiteral(subType.literal)) {\n options.push(subType.literal.text);\n } else if (subType.literal.kind === ts.SyntaxKind.TrueKeyword) {\n options.push(\"true\");\n } else if (subType.literal.kind === ts.SyntaxKind.FalseKeyword) {\n options.push(\"false\");\n } else {\n allLiterals = false;\n }\n } else {\n allLiterals = false;\n }\n }\n\n if (allLiterals && options.length > 0) {\n return {\n type: options.map((o) => `\"${o}\"`).join(\" | \"),\n typeKind: \"enum\",\n options,\n };\n }\n\n return {\n type: typeNode.getText(sourceFile),\n typeKind: \"union\",\n };\n }\n\n // Function type\n if (ts.isFunctionTypeNode(typeNode)) {\n return {\n type: typeNode.getText(sourceFile),\n typeKind: \"function\",\n };\n }\n\n // Array type\n if (ts.isArrayTypeNode(typeNode)) {\n return {\n type: typeNode.getText(sourceFile),\n typeKind: \"array\",\n };\n }\n\n // Type reference (e.g., ReactNode, React.ReactElement)\n if (ts.isTypeReferenceNode(typeNode)) {\n const typeName = typeNode.typeName.getText(sourceFile);\n\n // React types\n if (typeName === \"ReactNode\" || typeName === \"React.ReactNode\") {\n return { type: \"ReactNode\", typeKind: \"node\" };\n }\n if (typeName === \"ReactElement\" || typeName === \"React.ReactElement\") {\n return { type: \"ReactElement\", typeKind: \"element\" };\n }\n if (typeName === \"JSX.Element\") {\n return { type: \"JSX.Element\", typeKind: \"element\" };\n }\n\n return {\n type: typeNode.getText(sourceFile),\n typeKind: \"object\",\n };\n }\n\n // Object type literal\n if (ts.isTypeLiteralNode(typeNode)) {\n return {\n type: typeNode.getText(sourceFile),\n typeKind: \"object\",\n };\n }\n\n // Default: unknown\n return {\n type: typeNode.getText(sourceFile),\n typeKind: \"unknown\",\n };\n}\n\n/**\n * Get JSDoc comment from a node\n */\nfunction getJSDocComment(node: ts.Node): string | undefined {\n const jsDocTags = ts.getJSDocTags(node);\n const jsDoc = (node as unknown as { jsDoc?: ts.JSDoc[] }).jsDoc;\n\n if (jsDoc && jsDoc.length > 0) {\n const comment = jsDoc[0].comment;\n if (typeof comment === \"string\") {\n return comment;\n }\n if (Array.isArray(comment)) {\n return comment.map((c) => (typeof c === \"string\" ? c : c.text)).join(\"\");\n }\n }\n\n return undefined;\n}\n\n/**\n * Get @default value from JSDoc\n */\nfunction getJSDocDefault(node: ts.Node): unknown | undefined {\n const jsDocTags = ts.getJSDocTags(node);\n\n for (const tag of jsDocTags) {\n if (tag.tagName.text === \"default\") {\n const comment = tag.comment;\n if (typeof comment === \"string\") {\n // Try to parse as JSON\n try {\n return JSON.parse(comment);\n } catch {\n return comment;\n }\n }\n }\n }\n\n return undefined;\n}\n","/**\n * Registry Generator\n * Generates .fragments/registry.json and .fragments/index.json\n *\n * Philosophy: Generate only what AI agents and humans can't easily get from source.\n * - Props: AI can read TypeScript directly (skip by default)\n * - Paths: Essential for file navigation (always include)\n * - Enrichment: Human knowledge that doesn't exist elsewhere (include references)\n */\n\nimport { readFileSync, existsSync } from \"node:fs\";\nimport { relative, dirname, basename, join } from \"node:path\";\nimport fg from \"fast-glob\";\nimport type {\n Fragment,\n FragmentRegistry,\n FragmentIndex,\n RegistryComponentEntry,\n} from \"../fragment-types.js\";\nimport { extractPropsFromFile } from \"./typescript-extractor.js\";\nimport { BRAND } from \"../constants.js\";\nimport type { RegistryOptions } from \"../types.js\";\n\n/**\n * Options for registry generation\n */\nexport interface RegistryGeneratorOptions {\n /** Project root directory */\n projectRoot: string;\n /** Glob patterns for component files */\n componentPatterns?: string[];\n /** Glob patterns for story files */\n storyPatterns?: string[];\n /** Path to .fragments directory */\n fragmentsDir?: string;\n /** Registry options from config */\n registryOptions?: RegistryOptions;\n}\n\n/**\n * Result of registry generation\n */\nexport interface RegistryGeneratorResult {\n /** Generated registry (full metadata) */\n registry: FragmentRegistry;\n /** Generated index (minimal name -> path) */\n index: FragmentIndex;\n /** Errors encountered during generation */\n errors: Array<{ file: string; error: string }>;\n /** Warnings */\n warnings: Array<{ file: string; warning: string }>;\n}\n\n/**\n * Generate a fragment registry and index by scanning the project\n */\nexport async function generateRegistry(\n options: RegistryGeneratorOptions\n): Promise<RegistryGeneratorResult> {\n const {\n projectRoot,\n componentPatterns = [\"src/**/*.tsx\", \"src/**/*.ts\"],\n storyPatterns = [\"src/**/*.stories.tsx\", \"src/**/*.stories.ts\"],\n fragmentsDir = join(projectRoot, BRAND.dataDir),\n registryOptions = {},\n } = options;\n\n const {\n requireStory = false,\n publicOnly = false,\n categoryDepth = 1,\n includeProps = false,\n embedFragments = false,\n } = registryOptions;\n\n const errors: Array<{ file: string; error: string }> = [];\n const warnings: Array<{ file: string; warning: string }> = [];\n\n // Find all story files first (for requireStory filtering)\n const storyFiles = await fg(storyPatterns, {\n cwd: projectRoot,\n ignore: [\"**/node_modules/**\"],\n absolute: true,\n });\n\n // Build a map of story files by component directory and base name\n const storyMap = new Map<string, string>();\n for (const storyPath of storyFiles) {\n const storyDir = dirname(storyPath);\n const storyBase = basename(storyPath).replace(/\\.stories\\.(tsx?|jsx?)$/, \"\");\n storyMap.set(`${storyDir}/${storyBase}`, storyPath);\n }\n\n // Find all component files\n const componentFiles = await fg(componentPatterns, {\n cwd: projectRoot,\n ignore: [\n \"**/node_modules/**\",\n \"**/*.stories.*\",\n \"**/*.test.*\",\n \"**/*.spec.*\",\n \"**/*.d.ts\",\n ],\n absolute: true,\n });\n\n // Find all fragment files\n const fragmentPattern = join(\n BRAND.dataDir,\n BRAND.componentsDir,\n `*${BRAND.fileExtension}`\n );\n const fragmentFiles = await fg(fragmentPattern, {\n cwd: projectRoot,\n absolute: true,\n });\n\n // Build fragment lookup map\n const fragmentMap = new Map<string, { path: string; fragment: Fragment }>();\n for (const fragmentPath of fragmentFiles) {\n const fragmentName = basename(fragmentPath).replace(BRAND.fileExtension, \"\");\n try {\n const content = readFileSync(fragmentPath, \"utf-8\");\n const fragment = JSON.parse(content) as Fragment;\n fragmentMap.set(fragmentName, {\n path: relative(projectRoot, fragmentPath),\n fragment,\n });\n } catch (e) {\n errors.push({\n file: fragmentPath,\n error: `Failed to parse fragment: ${e instanceof Error ? e.message : String(e)}`,\n });\n }\n }\n\n // Build component index\n const components: Record<string, RegistryComponentEntry> = {};\n const indexComponents: Record<string, string> = {};\n const categories: Record<string, string[]> = {};\n\n // Process component files\n for (const filePath of componentFiles) {\n try {\n const extracted = extractPropsFromFile(filePath);\n if (!extracted || !extracted.componentName) {\n continue;\n }\n\n const componentName = extracted.componentName;\n const relativePath = relative(projectRoot, filePath);\n\n // Check if component is exported (for publicOnly filter)\n if (publicOnly && !extracted.exports.includes(componentName)) {\n continue;\n }\n\n // Find matching story file\n const componentDir = dirname(filePath);\n const baseNameWithoutExt = basename(filePath).replace(/\\.(tsx?|jsx?)$/, \"\");\n const storyPath = storyMap.get(`${componentDir}/${baseNameWithoutExt}`);\n\n // Apply requireStory filter\n if (requireStory && !storyPath) {\n continue;\n }\n\n // Get fragment data if exists\n const fragmentData = fragmentMap.get(componentName);\n\n // Determine category from directory structure\n const category = fragmentData?.fragment?.meta?.status\n ? undefined // Don't use status as category\n : getCategoryFromPath(relativePath, categoryDepth);\n\n // Check if fragment has meaningful enrichment (not just a skeleton)\n const hasEnrichment = fragmentData\n ? hasRealEnrichment(fragmentData.fragment)\n : false;\n\n // Build minimal component entry\n const entry: RegistryComponentEntry = {\n path: relativePath,\n };\n\n // Add optional fields only if present\n if (storyPath) {\n entry.storyPath = relative(projectRoot, storyPath);\n }\n\n if (fragmentData) {\n entry.fragmentPath = fragmentData.path;\n if (hasEnrichment) {\n entry.hasEnrichment = true;\n }\n if (fragmentData.fragment.description) {\n entry.description = fragmentData.fragment.description;\n }\n if (fragmentData.fragment.meta?.status) {\n entry.status = fragmentData.fragment.meta.status;\n }\n // Only embed full fragment if explicitly requested\n if (embedFragments) {\n entry.fragment = fragmentData.fragment;\n }\n }\n\n if (category) {\n entry.category = category;\n // Add to categories index\n if (!categories[category]) {\n categories[category] = [];\n }\n categories[category].push(componentName);\n }\n\n // Only include props if explicitly requested\n if (includeProps && extracted.props) {\n entry.props = extracted.props;\n }\n\n // Only include exports if multiple exports\n if (extracted.exports.length > 1) {\n entry.exports = extracted.exports;\n }\n\n components[componentName] = entry;\n indexComponents[componentName] = relativePath;\n } catch (e) {\n errors.push({\n file: filePath,\n error: `Failed to extract component: ${e instanceof Error ? e.message : String(e)}`,\n });\n }\n }\n\n const componentCount = Object.keys(components).length;\n\n // Build registry\n const registry: FragmentRegistry = {\n $schema: \"https://fragments.dev/schema/registry-v1.json\",\n version: \"1.0\",\n generatedAt: new Date().toISOString(),\n componentCount,\n components,\n categories,\n };\n\n // Build minimal index\n const index: FragmentIndex = {\n version: \"1.0\",\n generatedAt: registry.generatedAt,\n components: indexComponents,\n categories,\n };\n\n return { registry, index, errors, warnings };\n}\n\n/**\n * Get category from file path based on directory structure\n *\n * Examples:\n * - src/components/buttons/Button.tsx -> \"buttons\"\n * - src/components/forms/inputs/TextInput.tsx -> \"forms\" (depth 1) or \"forms/inputs\" (depth 2)\n * - src/components/Button/Button.tsx -> \"Button\" (component folder)\n */\nfunction getCategoryFromPath(relativePath: string, depth: number = 1): string | undefined {\n const parts = relativePath.split(\"/\");\n const componentsIndex = parts.findIndex((p) => p === \"components\");\n\n if (componentsIndex === -1) {\n return undefined;\n }\n\n // Get the parts after \"components/\"\n const afterComponents = parts.slice(componentsIndex + 1);\n\n // If it's just components/Button.tsx, no category\n if (afterComponents.length <= 1) {\n return undefined;\n }\n\n // If it's components/Button/Button.tsx (component folder pattern), no category\n const folderName = afterComponents[0];\n const fileName = afterComponents[afterComponents.length - 1].replace(/\\.(tsx?|jsx?)$/, \"\");\n if (afterComponents.length === 2 && folderName === fileName) {\n return undefined;\n }\n\n // Get category parts based on depth\n const categoryParts = afterComponents.slice(0, Math.min(depth, afterComponents.length - 1));\n\n // Filter out component folders (where folder name matches file name)\n const lastCategoryPart = categoryParts[categoryParts.length - 1];\n if (lastCategoryPart === fileName) {\n categoryParts.pop();\n }\n\n if (categoryParts.length === 0) {\n return undefined;\n }\n\n return categoryParts.join(\"/\");\n}\n\n/**\n * Check if a fragment has meaningful enrichment beyond a skeleton\n */\nfunction hasRealEnrichment(fragment: Fragment): boolean {\n // Has description beyond just the name\n if (fragment.description && fragment.description.length > 20) {\n return true;\n }\n\n // Has usage guidelines\n if (fragment.usage?.when && fragment.usage.when.length > 0) {\n return true;\n }\n if (fragment.usage?.doNot && fragment.usage.doNot.length > 0) {\n return true;\n }\n if (fragment.usage?.patterns && fragment.usage.patterns.length > 0) {\n return true;\n }\n\n // Has accessibility info\n if (fragment.accessibility?.requirements && fragment.accessibility.requirements.length > 0) {\n return true;\n }\n\n // Has figma links\n if (fragment.figma?.nodeId || fragment.figma?.variants) {\n return true;\n }\n\n // Has related components\n if (fragment.related?.similar && fragment.related.similar.length > 0) {\n return true;\n }\n\n return false;\n}\n\n/**\n * Resolve component name to path using registry\n */\nexport function resolveComponentPath(\n componentName: string,\n registry: FragmentRegistry\n): string | undefined {\n const entry = registry.components[componentName];\n return entry?.path;\n}\n\n/**\n * Get all components by category\n */\nexport function getComponentsByCategory(\n category: string,\n registry: FragmentRegistry\n): string[] {\n return registry.categories?.[category] || [];\n}\n","/**\n * Context.md Generator\n * Generates a consolidated AI-ready context file from the fragment registry\n *\n * Philosophy: Focus on human-authored semantic knowledge that AI can't get from source.\n * - Usage guidelines: When to use, when not to use\n * - Relationships: Similar components, alternatives\n * - Accessibility: Requirements and keyboard patterns\n * - DO NOT: Duplicate prop types (AI can read TypeScript directly)\n */\n\nimport type {\n FragmentRegistry,\n RegistryComponentEntry,\n RegistryPropEntry,\n FragmentDoNotItem,\n FragmentContextOptions,\n} from \"../fragment-types.js\";\n\n/**\n * Result of context generation\n */\nexport interface ContextGeneratorResult {\n /** Generated markdown content */\n content: string;\n /** Estimated token count */\n tokenEstimate: number;\n /** Component count */\n componentCount: number;\n}\n\n/**\n * Generate context.md from a fragment registry\n */\nexport function generateContextMd(\n registry: FragmentRegistry,\n options: FragmentContextOptions = {}\n): ContextGeneratorResult {\n const {\n format = \"markdown\",\n compact = false,\n include = { props: true, relations: true, code: false },\n } = options;\n\n if (format === \"json\") {\n return generateJsonContext(registry, include, compact);\n }\n\n return generateMarkdownContext(registry, include, compact);\n}\n\n/**\n * Generate markdown context\n */\nfunction generateMarkdownContext(\n registry: FragmentRegistry,\n include: { props?: boolean; relations?: boolean; code?: boolean },\n compact: boolean\n): ContextGeneratorResult {\n const lines: string[] = [];\n const componentNames = Object.keys(registry.components).sort();\n const componentCount = componentNames.length;\n\n // Header\n lines.push(\"# Component Library Context\");\n lines.push(`Generated: ${new Date().toISOString().split(\"T\")[0]} | Components: ${componentCount}`);\n lines.push(\"\");\n\n // Quick reference table\n lines.push(\"## Quick Reference\");\n lines.push(\"\");\n lines.push(\"| Component | Path | Category | Status |\");\n lines.push(\"|-----------|------|----------|--------|\");\n\n for (const name of componentNames) {\n const entry = registry.components[name];\n const status = entry.status || \"stable\";\n const category = entry.category || \"-\";\n lines.push(`| ${name} | ${entry.path} | ${category} | ${status} |`);\n }\n\n lines.push(\"\");\n\n // In compact mode, stop here\n if (compact) {\n const content = lines.join(\"\\n\");\n return {\n content,\n tokenEstimate: estimateTokens(content),\n componentCount,\n };\n }\n\n // Full documentation for each component\n lines.push(\"---\");\n lines.push(\"\");\n\n for (const name of componentNames) {\n const entry = registry.components[name];\n const fragment = entry.fragment;\n\n lines.push(`## ${name}`);\n lines.push(`**Path:** \\`${entry.path}\\``);\n\n if (entry.category) {\n lines.push(`**Category:** ${entry.category} | **Status:** ${entry.status || \"stable\"}`);\n }\n lines.push(\"\");\n\n // Description\n if (entry.description || fragment?.description) {\n lines.push(\"### Description\");\n lines.push(entry.description || fragment?.description || \"\");\n lines.push(\"\");\n }\n\n // Usage guidelines\n if (fragment?.usage) {\n if (fragment.usage.when && fragment.usage.when.length > 0) {\n lines.push(\"### When to Use\");\n for (const when of fragment.usage.when) {\n lines.push(`- ${when}`);\n }\n lines.push(\"\");\n }\n\n if (fragment.usage.doNot && fragment.usage.doNot.length > 0) {\n lines.push(\"### Do Not\");\n for (const doNotItem of fragment.usage.doNot) {\n if (typeof doNotItem === \"string\") {\n lines.push(`- ${doNotItem}`);\n } else {\n const item = doNotItem as FragmentDoNotItem;\n if (item.instead) {\n // Resolve component path\n const alternativePath = resolveComponentPath(item.instead, registry);\n if (alternativePath) {\n lines.push(`- ${item.text} → use **${item.instead}** (\\`${alternativePath}\\`)`);\n } else {\n lines.push(`- ${item.text} → use **${item.instead}**`);\n }\n } else {\n lines.push(`- ${item.text}`);\n }\n }\n }\n lines.push(\"\");\n }\n\n if (include.code && fragment.usage.patterns && fragment.usage.patterns.length > 0) {\n lines.push(\"### Patterns\");\n for (const pattern of fragment.usage.patterns) {\n lines.push(`**${pattern.name}**`);\n if (pattern.description) {\n lines.push(pattern.description);\n }\n lines.push(\"```tsx\");\n lines.push(pattern.code);\n lines.push(\"```\");\n lines.push(\"\");\n }\n }\n }\n\n // Props\n if (include.props && entry.props && Object.keys(entry.props).length > 0) {\n lines.push(\"### Props\");\n lines.push(\"| Prop | Type | Default | Description |\");\n lines.push(\"|------|------|---------|-------------|\");\n\n for (const [propName, prop] of Object.entries(entry.props)) {\n const type = formatPropType(prop);\n const defaultVal = prop.default !== undefined ? `\\`${JSON.stringify(prop.default)}\\`` : \"-\";\n const desc = prop.description || \"-\";\n const required = prop.required ? \" (required)\" : \"\";\n lines.push(`| ${propName}${required} | ${type} | ${defaultVal} | ${desc} |`);\n }\n lines.push(\"\");\n }\n\n // Accessibility\n if (fragment?.accessibility) {\n lines.push(\"### Accessibility\");\n if (fragment.accessibility.role) {\n lines.push(`**Role:** ${fragment.accessibility.role}`);\n }\n if (fragment.accessibility.requirements && fragment.accessibility.requirements.length > 0) {\n for (const req of fragment.accessibility.requirements) {\n lines.push(`- ${req}`);\n }\n }\n if (fragment.accessibility.keyboard) {\n lines.push(\"\");\n lines.push(\"**Keyboard:**\");\n for (const [key, action] of Object.entries(fragment.accessibility.keyboard)) {\n lines.push(`- \\`${key}\\`: ${action}`);\n }\n }\n lines.push(\"\");\n }\n\n // Related components\n if (include.relations && fragment?.related) {\n lines.push(\"### Related\");\n if (fragment.related.similar && fragment.related.similar.length > 0) {\n const resolved = fragment.related.similar.map((comp) => {\n const path = resolveComponentPath(comp, registry);\n return path ? `${comp} (\\`${path}\\`)` : comp;\n });\n lines.push(`- **Similar:** ${resolved.join(\", \")}`);\n }\n if (fragment.related.composedWith && fragment.related.composedWith.length > 0) {\n const resolved = fragment.related.composedWith.map((comp) => {\n const path = resolveComponentPath(comp, registry);\n return path ? `${comp} (\\`${path}\\`)` : comp;\n });\n lines.push(`- **Composed with:** ${resolved.join(\", \")}`);\n }\n if (fragment.related.usedIn && fragment.related.usedIn.length > 0) {\n const resolved = fragment.related.usedIn.map((comp) => {\n const path = resolveComponentPath(comp, registry);\n return path ? `${comp} (\\`${path}\\`)` : comp;\n });\n lines.push(`- **Used in:** ${resolved.join(\", \")}`);\n }\n lines.push(\"\");\n }\n\n lines.push(\"---\");\n lines.push(\"\");\n }\n\n const content = lines.join(\"\\n\");\n return {\n content,\n tokenEstimate: estimateTokens(content),\n componentCount,\n };\n}\n\n/**\n * Generate JSON context\n */\nfunction generateJsonContext(\n registry: FragmentRegistry,\n include: { props?: boolean; relations?: boolean; code?: boolean },\n compact: boolean\n): ContextGeneratorResult {\n const componentNames = Object.keys(registry.components).sort();\n const componentCount = componentNames.length;\n\n interface JsonComponent {\n path: string;\n category?: string;\n status?: string;\n description?: string;\n whenToUse?: string[];\n doNot?: Array<{ text: string; instead?: string; insteadPath?: string }>;\n props?: Record<string, {\n type: string;\n required?: boolean;\n default?: unknown;\n description?: string;\n }>;\n related?: {\n similar?: Array<{ name: string; path?: string }>;\n composedWith?: Array<{ name: string; path?: string }>;\n usedIn?: Array<{ name: string; path?: string }>;\n };\n }\n\n const components: Record<string, JsonComponent> = {};\n\n for (const name of componentNames) {\n const entry = registry.components[name];\n const fragment = entry.fragment;\n\n const component: JsonComponent = {\n path: entry.path,\n };\n\n if (entry.category) component.category = entry.category;\n if (entry.status) component.status = entry.status;\n if (entry.description || fragment?.description) {\n component.description = entry.description || fragment?.description;\n }\n\n if (!compact && fragment?.usage) {\n if (fragment.usage.when && fragment.usage.when.length > 0) {\n component.whenToUse = fragment.usage.when;\n }\n\n if (fragment.usage.doNot && fragment.usage.doNot.length > 0) {\n component.doNot = fragment.usage.doNot.map((item) => {\n if (typeof item === \"string\") {\n return { text: item };\n }\n const doNotItem = item as FragmentDoNotItem;\n const result: { text: string; instead?: string; insteadPath?: string } = {\n text: doNotItem.text,\n };\n if (doNotItem.instead) {\n result.instead = doNotItem.instead;\n result.insteadPath = resolveComponentPath(doNotItem.instead, registry);\n }\n return result;\n });\n }\n }\n\n if (!compact && include.props && entry.props) {\n component.props = {};\n for (const [propName, prop] of Object.entries(entry.props)) {\n component.props[propName] = {\n type: formatPropType(prop),\n };\n if (prop.required) component.props[propName].required = true;\n if (prop.default !== undefined) component.props[propName].default = prop.default;\n if (prop.description) component.props[propName].description = prop.description;\n }\n }\n\n if (!compact && include.relations && fragment?.related) {\n component.related = {};\n if (fragment.related.similar) {\n component.related.similar = fragment.related.similar.map((name) => ({\n name,\n path: resolveComponentPath(name, registry),\n }));\n }\n if (fragment.related.composedWith) {\n component.related.composedWith = fragment.related.composedWith.map((name) => ({\n name,\n path: resolveComponentPath(name, registry),\n }));\n }\n if (fragment.related.usedIn) {\n component.related.usedIn = fragment.related.usedIn.map((name) => ({\n name,\n path: resolveComponentPath(name, registry),\n }));\n }\n }\n\n components[name] = component;\n }\n\n const output = {\n version: \"1.0\",\n generatedAt: registry.generatedAt,\n componentCount,\n categories: registry.categories,\n components,\n };\n\n const content = JSON.stringify(output, null, 2);\n return {\n content,\n tokenEstimate: estimateTokens(content),\n componentCount,\n };\n}\n\n/**\n * Resolve component name to path\n */\nfunction resolveComponentPath(\n componentName: string,\n registry: FragmentRegistry\n): string | undefined {\n return registry.components[componentName]?.path;\n}\n\n/**\n * Format prop type for display\n */\nfunction formatPropType(prop: RegistryPropEntry): string {\n if (prop.options && prop.options.length > 0) {\n return prop.options.map((o) => `\"${o}\"`).join(\" \\\\| \");\n }\n return prop.type || \"unknown\";\n}\n\n/**\n * Estimate token count (rough: ~4 chars per token)\n */\nfunction estimateTokens(text: string): number {\n return Math.ceil(text.length / 4);\n}\n","import { writeFile, unlink } from 'node:fs/promises';\nimport { dirname, basename, join } from 'node:path';\nimport { pathToFileURL } from 'node:url';\nimport { build, type Plugin } from 'esbuild';\nimport type { FragmentDefinition } from './types.js';\nimport { BRAND } from './constants.js';\n\n// Inline implementation of defineFragment that will be injected into bundled files\n// This allows fragment files to work without @fragments/core being installed\nconst DEFINE_FRAGMENT_SHIM = `\nexport function defineFragment(def) {\n return def;\n}\nexport function defineBlock(def) {\n return def;\n}\nexport function defineRecipe(def) {\n return def;\n}\n`;\n\n/**\n * Creates an esbuild plugin that replaces @fragments/core imports with an inline shim.\n * This allows fragment files to be loaded without @fragments/core being installed in the target project.\n */\nfunction createFragmentsCoreShimPlugin(): Plugin {\n return {\n name: BRAND.vitePluginNamespace,\n setup(build) {\n // Intercept @fragments/core imports\n build.onResolve({ filter: /^@fragments\\/core$/ }, (args) => {\n return {\n path: args.path,\n namespace: BRAND.vitePluginNamespace,\n };\n });\n\n // Return the shim content\n build.onLoad({ filter: /.*/, namespace: BRAND.vitePluginNamespace }, () => {\n return {\n contents: DEFINE_FRAGMENT_SHIM,\n loader: 'js',\n };\n });\n },\n };\n}\n\n/**\n * Load a fragment file, handling TSX/TS transformation automatically.\n * Uses esbuild to bundle and transform TypeScript/JSX before importing.\n */\nexport async function loadFragmentFile(\n absolutePath: string\n): Promise<FragmentDefinition | null> {\n const ext = absolutePath.split('.').pop()?.toLowerCase();\n const needsTransform = ext === 'tsx' || ext === 'ts' || ext === 'jsx';\n\n if (!needsTransform) {\n // Plain JS file, import directly\n const fileUrl = pathToFileURL(absolutePath).href;\n const module = await import(fileUrl);\n return module.default ?? null;\n }\n\n // Bundle the file with all its local dependencies\n const sourceDir = dirname(absolutePath);\n const baseName = basename(absolutePath, `.${ext}`);\n // Use .cjs extension to ensure CommonJS evaluation\n const tempFile = join(sourceDir, `.${baseName}.fragments-temp-${Date.now()}.cjs`);\n\n try {\n // Use esbuild to bundle the fragment file\n // We inject a shim for @fragments/core so it doesn't need to be installed\n // Using CommonJS format to avoid ESM/CJS interop issues with node_modules\n await build({\n entryPoints: [absolutePath],\n outfile: tempFile,\n bundle: true,\n format: 'cjs', // Use CommonJS to avoid ESM issues with lodash, etc.\n target: 'es2022',\n jsx: 'automatic',\n platform: 'node',\n plugins: [createFragmentsCoreShimPlugin()],\n // Externalize all node_modules - we only need fragment metadata, not component code\n packages: 'external',\n // Also explicitly list patterns for nested imports\n external: [\n // React and its subpaths\n 'react',\n 'react-dom',\n 'react/*',\n 'react-dom/*',\n ],\n // Don't emit sourcemaps for temp files\n sourcemap: false,\n // Suppress warnings\n logLevel: 'silent',\n // Use loader to ignore style imports\n loader: {\n '.scss': 'empty',\n '.css': 'empty',\n '.svg': 'empty',\n '.png': 'empty',\n '.jpg': 'empty',\n '.jpeg': 'empty',\n '.gif': 'empty',\n '.woff': 'empty',\n '.woff2': 'empty',\n '.ttf': 'empty',\n '.eot': 'empty',\n },\n });\n\n const fileUrl = pathToFileURL(tempFile).href;\n const module = await import(fileUrl);\n return module.default ?? null;\n } finally {\n // Clean up temp file\n try {\n await unlink(tempFile);\n } catch {\n // Ignore cleanup errors\n }\n }\n}\n\n/**\n * Load multiple fragment files in parallel with error handling.\n */\nexport async function loadFragmentFiles(\n absolutePaths: string[]\n): Promise<Map<string, FragmentDefinition | Error>> {\n const results = new Map<string, FragmentDefinition | Error>();\n\n await Promise.all(\n absolutePaths.map(async (path) => {\n try {\n const fragment = await loadFragmentFile(path);\n if (fragment) {\n results.set(path, fragment);\n } else {\n results.set(path, new Error('No default export found'));\n }\n } catch (error) {\n results.set(\n path,\n error instanceof Error ? error : new Error(String(error))\n );\n }\n })\n );\n\n return results;\n}\n","/**\n * AST-Based Fragment Parser\n *\n * Extracts fragment metadata from .fragment.tsx files WITHOUT executing them.\n * This allows `fragments build` to work without any project dependencies.\n *\n * Uses TypeScript's compiler API to parse and analyze the AST.\n */\n\nimport ts from \"typescript\";\nimport type { FragmentMeta, FragmentUsage, PropDefinition, AIMetadata, FragmentContract } from \"./types.js\";\n\n/**\n * Parsed fragment metadata (extracted statically from AST)\n */\nexport interface ParsedFragmentMetadata {\n /** Component import path (e.g., \"./Button\" or \"@/components/Button\") */\n componentImport: string | null;\n\n /** Component name as used in the file */\n componentName: string | null;\n\n /** Component metadata */\n meta: Partial<FragmentMeta>;\n\n /** Usage guidelines */\n usage: Partial<FragmentUsage>;\n\n /** Props documentation */\n props: Record<string, Partial<PropDefinition>>;\n\n /** Variant definitions */\n variants: Array<{\n name: string;\n description: string;\n code?: string;\n figma?: string;\n args?: Record<string, unknown>;\n }>;\n\n /** Related components */\n relations: Array<{\n component: string;\n relationship: string;\n note: string;\n }>;\n\n /** AI-specific metadata for playground context generation */\n ai?: AIMetadata;\n\n /** Agent-optimized contract metadata */\n contract?: FragmentContract;\n\n /** Parse warnings */\n warnings: string[];\n}\n\n/**\n * Parse a fragment file and extract metadata without executing it.\n */\nexport function parseFragmentFile(\n fileContent: string,\n filePath?: string\n): ParsedFragmentMetadata {\n const warnings: string[] = [];\n\n // Parse the file\n const sourceFile = ts.createSourceFile(\n filePath ?? \"fragment.tsx\",\n fileContent,\n ts.ScriptTarget.Latest,\n true,\n ts.ScriptKind.TSX\n );\n\n // Find imports\n const imports = extractImports(sourceFile);\n\n // Find the defineFragment call\n const defineFragmentCall = findDefineFragmentCall(sourceFile);\n\n if (!defineFragmentCall) {\n warnings.push(\"No defineFragment() call found\");\n return {\n componentImport: null,\n componentName: null,\n meta: {},\n usage: { when: [], whenNot: [] },\n props: {},\n variants: [],\n relations: [],\n warnings,\n };\n }\n\n // Extract the argument (object literal)\n const arg = defineFragmentCall.arguments[0];\n if (!arg || !ts.isObjectLiteralExpression(arg)) {\n warnings.push(\"defineFragment() argument is not an object literal\");\n return {\n componentImport: null,\n componentName: null,\n meta: {},\n usage: { when: [], whenNot: [] },\n props: {},\n variants: [],\n relations: [],\n warnings,\n };\n }\n\n // Extract component reference\n const componentProp = findProperty(arg, \"component\");\n let componentName: string | null = null;\n let componentImport: string | null = null;\n\n if (componentProp && ts.isIdentifier(componentProp)) {\n componentName = componentProp.text;\n // Find the import for this component\n componentImport = imports.get(componentName) ?? null;\n }\n\n // Extract meta\n const meta = extractMeta(arg, warnings);\n\n // Extract usage\n const usage = extractUsage(arg, warnings);\n\n // Extract props\n const props = extractProps(arg, warnings);\n\n // Extract variants\n const variants = extractVariants(arg, sourceFile, warnings);\n\n // Extract relations\n const relations = extractRelations(arg, warnings);\n\n // Extract AI metadata\n const ai = extractAIMetadata(arg, warnings);\n\n // Extract contract metadata\n const contract = extractContractMetadata(arg);\n\n return {\n componentImport,\n componentName,\n meta,\n usage,\n props,\n variants,\n relations,\n ai,\n contract,\n warnings,\n };\n}\n\n/**\n * Extract all imports from the source file.\n * Returns a map of imported name -> module path\n */\nfunction extractImports(sourceFile: ts.SourceFile): Map<string, string> {\n const imports = new Map<string, string>();\n\n ts.forEachChild(sourceFile, (node) => {\n if (ts.isImportDeclaration(node)) {\n const moduleSpecifier = node.moduleSpecifier;\n if (ts.isStringLiteral(moduleSpecifier)) {\n const modulePath = moduleSpecifier.text;\n\n // Handle named imports\n const importClause = node.importClause;\n if (importClause) {\n // Default import\n if (importClause.name) {\n imports.set(importClause.name.text, modulePath);\n }\n\n // Named imports\n const namedBindings = importClause.namedBindings;\n if (namedBindings && ts.isNamedImports(namedBindings)) {\n for (const element of namedBindings.elements) {\n imports.set(element.name.text, modulePath);\n }\n }\n }\n }\n }\n });\n\n return imports;\n}\n\n/**\n * Find the defineFragment() call in the source file.\n */\nfunction findDefineFragmentCall(\n sourceFile: ts.SourceFile\n): ts.CallExpression | null {\n let result: ts.CallExpression | null = null;\n\n function visit(node: ts.Node): void {\n if (ts.isCallExpression(node)) {\n const expression = node.expression;\n if (ts.isIdentifier(expression) && expression.text === \"defineFragment\") {\n result = node;\n return;\n }\n }\n ts.forEachChild(node, visit);\n }\n\n visit(sourceFile);\n return result;\n}\n\n/**\n * Find a property in an object literal by name.\n */\nfunction findProperty(\n obj: ts.ObjectLiteralExpression,\n name: string\n): ts.Expression | null {\n for (const prop of obj.properties) {\n if (ts.isPropertyAssignment(prop)) {\n const propName = prop.name;\n if (ts.isIdentifier(propName) && propName.text === name) {\n return prop.initializer;\n }\n }\n }\n return null;\n}\n\n/**\n * Extract meta object from defineFragment call.\n */\nfunction extractMeta(\n arg: ts.ObjectLiteralExpression,\n warnings: string[]\n): Partial<FragmentMeta> {\n const metaProp = findProperty(arg, \"meta\");\n if (!metaProp || !ts.isObjectLiteralExpression(metaProp)) {\n warnings.push(\"No meta object found\");\n return {};\n }\n\n const meta: Partial<FragmentMeta> = {};\n\n // Extract string properties\n const name = extractStringProperty(metaProp, \"name\");\n if (name) meta.name = name;\n\n const description = extractStringProperty(metaProp, \"description\");\n if (description) meta.description = description;\n\n const category = extractStringProperty(metaProp, \"category\");\n if (category) meta.category = category;\n\n const status = extractStringProperty(metaProp, \"status\") as FragmentMeta[\"status\"];\n if (status) meta.status = status;\n\n const since = extractStringProperty(metaProp, \"since\");\n if (since) meta.since = since;\n\n // Extract figma URL\n const figma = extractStringProperty(metaProp, \"figma\");\n if (figma) meta.figma = figma;\n\n // Extract tags array\n const tags = extractStringArray(metaProp, \"tags\");\n if (tags.length > 0) meta.tags = tags;\n\n return meta;\n}\n\n/**\n * Extract usage object from defineFragment call.\n */\nfunction extractUsage(\n arg: ts.ObjectLiteralExpression,\n warnings: string[]\n): Partial<FragmentUsage> {\n const usageProp = findProperty(arg, \"usage\");\n if (!usageProp || !ts.isObjectLiteralExpression(usageProp)) {\n return { when: [], whenNot: [] };\n }\n\n return {\n when: extractStringArray(usageProp, \"when\"),\n whenNot: extractStringArray(usageProp, \"whenNot\"),\n guidelines: extractStringArray(usageProp, \"guidelines\"),\n accessibility: extractStringArray(usageProp, \"accessibility\"),\n };\n}\n\n/**\n * Extract props object from defineFragment call.\n */\nfunction extractProps(\n arg: ts.ObjectLiteralExpression,\n warnings: string[]\n): Record<string, Partial<PropDefinition>> {\n const propsProp = findProperty(arg, \"props\");\n if (!propsProp || !ts.isObjectLiteralExpression(propsProp)) {\n return {};\n }\n\n const props: Record<string, Partial<PropDefinition>> = {};\n\n for (const prop of propsProp.properties) {\n if (ts.isPropertyAssignment(prop) && ts.isIdentifier(prop.name)) {\n const propName = prop.name.text;\n const propValue = prop.initializer;\n\n if (ts.isObjectLiteralExpression(propValue)) {\n props[propName] = extractPropDefinition(propValue);\n }\n }\n }\n\n return props;\n}\n\n/**\n * Extract a single prop definition.\n */\nfunction extractPropDefinition(\n obj: ts.ObjectLiteralExpression\n): Partial<PropDefinition> {\n const def: Partial<PropDefinition> = {};\n\n const type = extractStringProperty(obj, \"type\");\n if (type) def.type = type as PropDefinition[\"type\"];\n\n const description = extractStringProperty(obj, \"description\");\n if (description) def.description = description;\n\n const values = extractStringArray(obj, \"values\");\n if (values.length > 0) def.values = values;\n\n const required = extractBooleanProperty(obj, \"required\");\n if (required !== null) def.required = required;\n\n // Default value - try to extract as literal\n const defaultProp = findProperty(obj, \"default\");\n if (defaultProp) {\n def.default = extractLiteralValue(defaultProp);\n }\n\n const constraints = extractStringArray(obj, \"constraints\");\n if (constraints.length > 0) def.constraints = constraints;\n\n return def;\n}\n\n/**\n * Extract variants array from defineFragment call.\n */\nfunction extractVariants(\n arg: ts.ObjectLiteralExpression,\n sourceFile: ts.SourceFile,\n warnings: string[]\n): Array<{ name: string; description: string; code?: string; figma?: string; args?: Record<string, unknown> }> {\n const variantsProp = findProperty(arg, \"variants\");\n if (!variantsProp || !ts.isArrayLiteralExpression(variantsProp)) {\n return [];\n }\n\n const variants: Array<{ name: string; description: string; code?: string; figma?: string; args?: Record<string, unknown> }> = [];\n\n for (const element of variantsProp.elements) {\n if (ts.isObjectLiteralExpression(element)) {\n const name = extractStringProperty(element, \"name\");\n const description = extractStringProperty(element, \"description\");\n\n if (name) {\n const variant: { name: string; description: string; code?: string; figma?: string; args?: Record<string, unknown> } = {\n name,\n description: description ?? \"\",\n };\n\n // Try to extract code property if present\n const codeProp = findProperty(element, \"code\");\n if (codeProp && (ts.isStringLiteral(codeProp) || ts.isNoSubstitutionTemplateLiteral(codeProp))) {\n variant.code = codeProp.text;\n }\n\n // Try to extract render function and convert to code string\n const renderProp = findProperty(element, \"render\");\n if (renderProp && !variant.code) {\n variant.code = extractRenderCode(renderProp, sourceFile);\n }\n\n // Extract figma URL for variant\n const figma = extractStringProperty(element, \"figma\");\n if (figma) {\n variant.figma = figma;\n }\n\n // Extract args object for code generation\n const argsProp = findProperty(element, \"args\");\n if (argsProp && ts.isObjectLiteralExpression(argsProp)) {\n const argsValue = extractLiteralValue(argsProp);\n if (argsValue && typeof argsValue === 'object' && !Array.isArray(argsValue)) {\n variant.args = argsValue as Record<string, unknown>;\n }\n }\n\n variants.push(variant);\n }\n }\n }\n\n return variants;\n}\n\n/**\n * Remove common leading whitespace from all lines (dedent).\n * This handles template literals and JSX that have extra indentation from code formatting.\n *\n * Special handling: If the first line has no indentation (common after .trim()),\n * we calculate minimum indent from subsequent lines only.\n */\nfunction dedent(str: string): string {\n const lines = str.split('\\n');\n\n if (lines.length <= 1) {\n return str;\n }\n\n // Check if first line has no indentation\n const firstLineIndent = lines[0].match(/^(\\s*)/)?.[1].length ?? 0;\n const startIndex = firstLineIndent === 0 ? 1 : 0;\n\n // Find the minimum indentation (ignoring empty lines)\n let minIndent = Infinity;\n for (let i = startIndex; i < lines.length; i++) {\n const line = lines[i];\n if (line.trim() === '') continue;\n const match = line.match(/^(\\s*)/);\n if (match) {\n minIndent = Math.min(minIndent, match[1].length);\n }\n }\n\n // If no indentation found, return as-is\n if (minIndent === Infinity || minIndent === 0) {\n return str;\n }\n\n // Remove the common indentation from all lines (except first if it had no indent)\n return lines\n .map((line, index) => {\n if (index === 0 && firstLineIndent === 0) {\n return line; // Keep first line as-is\n }\n return line.slice(minIndent);\n })\n .join('\\n');\n}\n\n/**\n * Extract the code from a render function.\n */\nfunction extractRenderCode(\n renderProp: ts.Expression,\n sourceFile: ts.SourceFile\n): string | undefined {\n // Handle arrow function: () => <Component />\n if (ts.isArrowFunction(renderProp)) {\n const body = renderProp.body;\n // Get the source text of the body\n const start = body.getStart(sourceFile);\n const end = body.getEnd();\n let code = sourceFile.text.substring(start, end).trim();\n\n // Remove parentheses if present\n if (code.startsWith(\"(\") && code.endsWith(\")\")) {\n code = code.slice(1, -1).trim();\n }\n\n // Dedent the code to remove common leading whitespace\n code = dedent(code);\n\n return code;\n }\n\n return undefined;\n}\n\n/**\n * Extract relations array from defineFragment call.\n */\nfunction extractRelations(\n arg: ts.ObjectLiteralExpression,\n warnings: string[]\n): Array<{ component: string; relationship: string; note: string }> {\n const relationsProp = findProperty(arg, \"relations\");\n if (!relationsProp || !ts.isArrayLiteralExpression(relationsProp)) {\n return [];\n }\n\n const relations: Array<{ component: string; relationship: string; note: string }> = [];\n\n for (const element of relationsProp.elements) {\n if (ts.isObjectLiteralExpression(element)) {\n const component = extractStringProperty(element, \"component\");\n const relationship = extractStringProperty(element, \"relationship\");\n const note = extractStringProperty(element, \"note\");\n\n if (component && relationship) {\n relations.push({\n component,\n relationship,\n note: note ?? \"\",\n });\n }\n }\n }\n\n return relations;\n}\n\n/**\n * Extract AI metadata from defineFragment call.\n */\nfunction extractAIMetadata(\n arg: ts.ObjectLiteralExpression,\n warnings: string[]\n): AIMetadata | undefined {\n const aiProp = findProperty(arg, \"ai\");\n if (!aiProp || !ts.isObjectLiteralExpression(aiProp)) {\n return undefined;\n }\n\n const ai: AIMetadata = {};\n\n // Extract compositionPattern\n const compositionPattern = extractStringProperty(aiProp, \"compositionPattern\");\n if (compositionPattern && ['compound', 'simple', 'controlled'].includes(compositionPattern)) {\n ai.compositionPattern = compositionPattern as AIMetadata['compositionPattern'];\n }\n\n // Extract subComponents array\n const subComponents = extractStringArray(aiProp, \"subComponents\");\n if (subComponents.length > 0) {\n ai.subComponents = subComponents;\n }\n\n // Extract requiredChildren array\n const requiredChildren = extractStringArray(aiProp, \"requiredChildren\");\n if (requiredChildren.length > 0) {\n ai.requiredChildren = requiredChildren;\n }\n\n // Extract commonPatterns array\n const commonPatterns = extractStringArray(aiProp, \"commonPatterns\");\n if (commonPatterns.length > 0) {\n ai.commonPatterns = commonPatterns;\n }\n\n // Only return if we have any fields\n if (Object.keys(ai).length > 0) {\n return ai;\n }\n\n return undefined;\n}\n\n/**\n * Extract contract metadata from defineFragment call.\n */\nfunction extractContractMetadata(\n arg: ts.ObjectLiteralExpression,\n): FragmentContract | undefined {\n const contractProp = findProperty(arg, \"contract\");\n if (!contractProp || !ts.isObjectLiteralExpression(contractProp)) {\n return undefined;\n }\n\n const contract: FragmentContract = {};\n\n // Extract propsSummary array\n const propsSummary = extractStringArray(contractProp, \"propsSummary\");\n if (propsSummary.length > 0) {\n contract.propsSummary = propsSummary;\n }\n\n // Extract a11yRules array\n const a11yRules = extractStringArray(contractProp, \"a11yRules\");\n if (a11yRules.length > 0) {\n contract.a11yRules = a11yRules;\n }\n\n // Only return if we have any fields\n if (Object.keys(contract).length > 0) {\n return contract;\n }\n\n return undefined;\n}\n\n/**\n * Extract a string property from an object literal.\n */\nfunction extractStringProperty(\n obj: ts.ObjectLiteralExpression,\n name: string\n): string | null {\n const prop = findProperty(obj, name);\n if (prop && ts.isStringLiteral(prop)) {\n return prop.text;\n }\n // Handle template literals without expressions\n if (prop && ts.isNoSubstitutionTemplateLiteral(prop)) {\n return prop.text;\n }\n return null;\n}\n\n/**\n * Extract a boolean property from an object literal.\n */\nfunction extractBooleanProperty(\n obj: ts.ObjectLiteralExpression,\n name: string\n): boolean | null {\n const prop = findProperty(obj, name);\n if (prop) {\n if (prop.kind === ts.SyntaxKind.TrueKeyword) return true;\n if (prop.kind === ts.SyntaxKind.FalseKeyword) return false;\n }\n return null;\n}\n\n/**\n * Extract a string array from an object literal.\n */\nfunction extractStringArray(\n obj: ts.ObjectLiteralExpression,\n name: string\n): string[] {\n const prop = findProperty(obj, name);\n if (!prop || !ts.isArrayLiteralExpression(prop)) {\n return [];\n }\n\n const result: string[] = [];\n for (const element of prop.elements) {\n if (ts.isStringLiteral(element)) {\n result.push(element.text);\n } else if (ts.isNoSubstitutionTemplateLiteral(element)) {\n result.push(element.text);\n }\n }\n return result;\n}\n\n/**\n * Extract a literal value from an expression.\n */\nfunction extractLiteralValue(expr: ts.Expression): unknown {\n if (ts.isStringLiteral(expr)) {\n return expr.text;\n }\n if (ts.isNumericLiteral(expr)) {\n return Number(expr.text);\n }\n if (expr.kind === ts.SyntaxKind.TrueKeyword) {\n return true;\n }\n if (expr.kind === ts.SyntaxKind.FalseKeyword) {\n return false;\n }\n if (expr.kind === ts.SyntaxKind.NullKeyword) {\n return null;\n }\n if (ts.isArrayLiteralExpression(expr)) {\n return expr.elements.map(extractLiteralValue);\n }\n if (ts.isObjectLiteralExpression(expr)) {\n const obj: Record<string, unknown> = {};\n for (const prop of expr.properties) {\n if (ts.isPropertyAssignment(prop) && ts.isIdentifier(prop.name)) {\n obj[prop.name.text] = extractLiteralValue(prop.initializer);\n }\n }\n return obj;\n }\n return undefined;\n}\n","/**\n * Loader for Storybook .storybook/preview.{tsx,ts,jsx,js} configuration.\n *\n * This module detects and loads Storybook preview configuration to extract:\n * - decorators: Global decorators applied to all stories\n * - parameters: Global parameters for all stories\n * - globalTypes: Global type definitions (e.g., for toolbar controls)\n * - args: Default args for all stories\n * - argTypes: Default argTypes for all stories\n * - loaders: Global loaders for all stories\n */\n\nimport { existsSync } from \"node:fs\";\nimport { join, resolve } from \"node:path\";\nimport type { PreviewConfig } from \"./storyAdapter.js\";\n\n/**\n * Possible file extensions and names for preview config\n */\nconst PREVIEW_FILES = [\n \"preview.tsx\",\n \"preview.ts\",\n \"preview.jsx\",\n \"preview.js\",\n];\n\n/**\n * Find the preview config file in a .storybook directory\n *\n * @param storybookDir - Path to the .storybook directory\n * @returns The full path to the preview file, or null if not found\n */\nexport function findPreviewConfigPath(storybookDir: string): string | null {\n for (const fileName of PREVIEW_FILES) {\n const filePath = join(storybookDir, fileName);\n if (existsSync(filePath)) {\n return filePath;\n }\n }\n return null;\n}\n\n/**\n * Find the .storybook directory starting from a project root.\n * Checks common locations.\n *\n * @param projectRoot - The project root directory\n * @returns The path to the .storybook directory, or null if not found\n */\nexport function findStorybookDir(projectRoot: string): string | null {\n const possiblePaths = [\n join(projectRoot, \".storybook\"),\n join(projectRoot, \"storybook\"),\n ];\n\n for (const dir of possiblePaths) {\n if (existsSync(dir)) {\n return dir;\n }\n }\n return null;\n}\n\n/**\n * Load and parse the Storybook preview configuration.\n *\n * This function dynamically imports the preview file and extracts\n * the relevant configuration exports.\n *\n * @param previewPath - Full path to the preview config file\n * @returns The parsed preview configuration\n */\nexport async function loadPreviewConfig(\n previewPath: string\n): Promise<PreviewConfig> {\n try {\n // Dynamic import with file:// URL for ESM compatibility\n const fileUrl = new URL(`file://${resolve(previewPath)}`);\n const module = await import(fileUrl.href);\n\n // Extract configuration from the module\n const config: PreviewConfig = {\n decorators: module.decorators ?? module.default?.decorators ?? [],\n parameters: module.parameters ?? module.default?.parameters ?? {},\n globalTypes: module.globalTypes ?? module.default?.globalTypes ?? {},\n args: module.args ?? module.default?.args ?? {},\n argTypes: module.argTypes ?? module.default?.argTypes ?? {},\n loaders: module.loaders ?? module.default?.loaders ?? [],\n };\n\n return config;\n } catch (error) {\n console.warn(\n `[Fragments] Failed to load preview config from ${previewPath}:`,\n error instanceof Error ? error.message : error\n );\n return {};\n }\n}\n\n/**\n * Auto-detect and load preview configuration from a project.\n *\n * @param projectRoot - The project root directory\n * @returns The preview configuration, or empty config if not found\n */\nexport async function autoLoadPreviewConfig(\n projectRoot: string\n): Promise<PreviewConfig> {\n const storybookDir = findStorybookDir(projectRoot);\n if (!storybookDir) {\n return {};\n }\n\n const previewPath = findPreviewConfigPath(storybookDir);\n if (!previewPath) {\n return {};\n }\n\n return loadPreviewConfig(previewPath);\n}\n\n/**\n * Generate a virtual module that imports and re-exports preview config.\n * This is used in the Vite plugin to make preview config available at runtime.\n *\n * @param previewPath - Full path to the preview config file, or null\n * @returns JavaScript code for the virtual module\n */\nexport function generatePreviewModule(previewPath: string | null): string {\n if (!previewPath) {\n // No preview config found - export empty defaults\n return `\nexport const decorators = [];\nexport const parameters = {};\nexport const globalTypes = {};\nexport const args = {};\nexport const argTypes = {};\nexport const loaders = [];\n\nexport default {\n decorators,\n parameters,\n globalTypes,\n args,\n argTypes,\n loaders,\n};\n`;\n }\n\n // Import from the actual preview file and re-export\n return `\nimport * as preview from \"${previewPath}\";\n\nexport const decorators = preview.decorators ?? preview.default?.decorators ?? [];\nexport const parameters = preview.parameters ?? preview.default?.parameters ?? {};\nexport const globalTypes = preview.globalTypes ?? preview.default?.globalTypes ?? {};\nexport const args = preview.args ?? preview.default?.args ?? {};\nexport const argTypes = preview.argTypes ?? preview.default?.argTypes ?? {};\nexport const loaders = preview.loaders ?? preview.default?.loaders ?? [];\n\nexport default {\n decorators,\n parameters,\n globalTypes,\n args,\n argTypes,\n loaders,\n};\n`;\n}\n","/**\n * AST-based import analysis for detecting component relationships\n *\n * Analyzes TypeScript/JavaScript files to find which components import others,\n * providing accurate \"used by\" relationships for the component graph.\n *\n * @deprecated Import analysis is now handled by graph-extractor.ts which provides\n * richer edge-typed relationships via the ComponentGraph. These functions are\n * maintained for backward compatibility. For new code, use\n * `buildComponentGraph()` from `./graph-extractor.js` instead.\n */\n\nimport ts from 'typescript';\nimport { readFileSync } from 'node:fs';\nimport { resolve, dirname, basename } from 'node:path';\n\nexport interface ComponentImport {\n /** Name of the imported component */\n name: string;\n /** Import path (relative or package) */\n path: string;\n /** Whether this is a default import */\n isDefault: boolean;\n /** Whether this is a namespace import */\n isNamespace: boolean;\n}\n\nexport interface ImportAnalysisResult {\n /** Source file path */\n filePath: string;\n /** Component name extracted from file */\n componentName: string;\n /** List of component imports found */\n imports: ComponentImport[];\n}\n\n/**\n * Extract component name from a file path\n */\nfunction extractComponentNameFromPath(filePath: string): string {\n const fileName = basename(filePath);\n\n // Handle index.tsx files - use parent directory name\n if (fileName === 'index.tsx' || fileName === 'index.ts' || fileName === 'index.jsx' || fileName === 'index.js') {\n const parentDir = basename(dirname(filePath));\n return parentDir;\n }\n\n // Remove extension and common suffixes\n return fileName\n .replace(/\\.(tsx?|jsx?)$/, '')\n .replace(/\\.stories$/, '')\n .replace(/\\.fragment$/, '')\n .replace(/\\.fragment$/, '');\n}\n\n/**\n * Check if an import path looks like a component import (not a utility/library)\n */\nfunction isComponentImportPath(importPath: string): boolean {\n // Skip node_modules packages\n if (!importPath.startsWith('.') && !importPath.startsWith('/')) {\n return false;\n }\n\n // Skip common utility/hook patterns\n const skipPatterns = [\n /\\/hooks\\//,\n /\\/utils\\//,\n /\\/helpers\\//,\n /\\/lib\\//,\n /\\/types\\//,\n /\\/constants\\//,\n /\\/styles\\//,\n /\\/context\\//,\n /\\.css$/,\n /\\.scss$/,\n /\\.less$/,\n /\\.json$/,\n ];\n\n for (const pattern of skipPatterns) {\n if (pattern.test(importPath)) {\n return false;\n }\n }\n\n return true;\n}\n\n/**\n * Check if an imported name looks like a React component (PascalCase)\n */\nfunction isPascalCase(name: string): boolean {\n return /^[A-Z][a-zA-Z0-9]*$/.test(name);\n}\n\n/**\n * Analyze a single file for component imports\n */\nexport function analyzeFileImports(filePath: string): ImportAnalysisResult {\n const componentName = extractComponentNameFromPath(filePath);\n const imports: ComponentImport[] = [];\n\n let sourceText: string;\n try {\n sourceText = readFileSync(filePath, 'utf-8');\n } catch {\n return { filePath, componentName, imports };\n }\n\n const sourceFile = ts.createSourceFile(\n filePath,\n sourceText,\n ts.ScriptTarget.Latest,\n true,\n filePath.endsWith('.tsx') || filePath.endsWith('.jsx')\n ? ts.ScriptKind.TSX\n : ts.ScriptKind.TS\n );\n\n ts.forEachChild(sourceFile, (node) => {\n if (ts.isImportDeclaration(node)) {\n const moduleSpecifier = node.moduleSpecifier;\n if (!ts.isStringLiteral(moduleSpecifier)) return;\n\n const importPath = moduleSpecifier.text;\n\n // Skip non-component imports\n if (!isComponentImportPath(importPath)) return;\n\n const importClause = node.importClause;\n if (!importClause) return;\n\n // Default import: import Button from './Button'\n if (importClause.name) {\n const name = importClause.name.text;\n if (isPascalCase(name)) {\n imports.push({\n name,\n path: importPath,\n isDefault: true,\n isNamespace: false,\n });\n }\n }\n\n // Named imports: import { Button, Icon } from './components'\n const namedBindings = importClause.namedBindings;\n if (namedBindings) {\n if (ts.isNamedImports(namedBindings)) {\n for (const element of namedBindings.elements) {\n const name = element.name.text;\n if (isPascalCase(name)) {\n imports.push({\n name,\n path: importPath,\n isDefault: false,\n isNamespace: false,\n });\n }\n }\n } else if (ts.isNamespaceImport(namedBindings)) {\n // Namespace import: import * as Components from './components'\n // We can't reliably know what components are used without more analysis\n const name = namedBindings.name.text;\n imports.push({\n name,\n path: importPath,\n isDefault: false,\n isNamespace: true,\n });\n }\n }\n }\n });\n\n return { filePath, componentName, imports };\n}\n\n/**\n * Build a map of which components are imported by which files\n */\nexport function buildImportGraph(\n filePaths: string[]\n): Map<string, string[]> {\n // Map from component name -> list of component names that import it\n const importedBy = new Map<string, string[]>();\n\n for (const filePath of filePaths) {\n const result = analyzeFileImports(filePath);\n\n for (const imp of result.imports) {\n // Skip namespace imports for now\n if (imp.isNamespace) continue;\n\n const importedComponent = imp.name;\n const importingComponent = result.componentName;\n\n // Don't add self-references\n if (importedComponent === importingComponent) continue;\n\n const existing = importedBy.get(importedComponent) || [];\n if (!existing.includes(importingComponent)) {\n existing.push(importingComponent);\n importedBy.set(importedComponent, existing);\n }\n }\n }\n\n return importedBy;\n}\n\n/**\n * Get components that import a specific component\n */\nexport function getImportedBy(\n componentName: string,\n importGraph: Map<string, string[]>\n): string[] {\n return importGraph.get(componentName) || [];\n}\n"],"mappings":"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","names":["readFileSync","dirname","dirname","readFileSync","content","name","dirname","basename","join","build","dirname","basename","join","ts","existsSync","join","resolve","ts","readFileSync","dirname","basename"]}
|