@fragments-sdk/cli 0.4.2 → 0.4.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/mcp-bin.js CHANGED
@@ -16,7 +16,7 @@ import {
16
16
  ListToolsRequestSchema
17
17
  } from "@modelcontextprotocol/sdk/types.js";
18
18
  import { readFile } from "fs/promises";
19
- import { existsSync, readFileSync } from "fs";
19
+ import { existsSync, readFileSync, readdirSync } from "fs";
20
20
  import { join, dirname, resolve } from "path";
21
21
  import { createRequire } from "module";
22
22
 
@@ -84,6 +84,98 @@ function filterPlaceholders(items) {
84
84
  (item) => !PLACEHOLDER_PATTERNS.some((pattern) => pattern.test(item.trim()))
85
85
  );
86
86
  }
87
+ function resolveWorkspaceGlob(baseDir, pattern) {
88
+ const parts = pattern.split("/");
89
+ let dirs = [baseDir];
90
+ for (const part of parts) {
91
+ if (part === "**") continue;
92
+ const next = [];
93
+ for (const d of dirs) {
94
+ if (part === "*") {
95
+ try {
96
+ for (const entry of readdirSync(d, { withFileTypes: true })) {
97
+ if (entry.isDirectory() && !entry.name.startsWith(".") && entry.name !== "node_modules") {
98
+ next.push(join(d, entry.name));
99
+ }
100
+ }
101
+ } catch {
102
+ }
103
+ } else {
104
+ const candidate = join(d, part);
105
+ if (existsSync(candidate)) next.push(candidate);
106
+ }
107
+ }
108
+ dirs = next;
109
+ }
110
+ return dirs;
111
+ }
112
+ function getWorkspaceDirs(rootDir) {
113
+ const dirs = [];
114
+ const rootPkgPath = join(rootDir, "package.json");
115
+ if (existsSync(rootPkgPath)) {
116
+ try {
117
+ const rootPkg = JSON.parse(readFileSync(rootPkgPath, "utf-8"));
118
+ const workspaces = Array.isArray(rootPkg.workspaces) ? rootPkg.workspaces : rootPkg.workspaces?.packages;
119
+ if (Array.isArray(workspaces)) {
120
+ for (const pattern of workspaces) {
121
+ dirs.push(...resolveWorkspaceGlob(rootDir, pattern));
122
+ }
123
+ return dirs;
124
+ }
125
+ } catch {
126
+ }
127
+ }
128
+ const pnpmWsPath = join(rootDir, "pnpm-workspace.yaml");
129
+ if (existsSync(pnpmWsPath)) {
130
+ try {
131
+ const content = readFileSync(pnpmWsPath, "utf-8");
132
+ const lines = content.split("\n");
133
+ let inPackages = false;
134
+ for (const line of lines) {
135
+ if (/^packages\s*:/.test(line)) {
136
+ inPackages = true;
137
+ continue;
138
+ }
139
+ if (inPackages) {
140
+ const match = line.match(/^\s+-\s+['"]?([^'"#\n]+)['"]?/);
141
+ if (match) {
142
+ dirs.push(...resolveWorkspaceGlob(rootDir, match[1].trim()));
143
+ } else if (/^\S/.test(line) && line.trim()) {
144
+ break;
145
+ }
146
+ }
147
+ }
148
+ } catch {
149
+ }
150
+ }
151
+ return dirs;
152
+ }
153
+ function findFragmentsInDeps(dir, found) {
154
+ const pkgJsonPath = join(dir, "package.json");
155
+ if (!existsSync(pkgJsonPath)) return;
156
+ try {
157
+ const pkgJson = JSON.parse(readFileSync(pkgJsonPath, "utf-8"));
158
+ const allDeps = {
159
+ ...pkgJson.dependencies,
160
+ ...pkgJson.devDependencies
161
+ };
162
+ const localRequire = createRequire(join(dir, "noop.js"));
163
+ for (const depName of Object.keys(allDeps)) {
164
+ try {
165
+ const depPkgPath = localRequire.resolve(`${depName}/package.json`);
166
+ const depPkg = JSON.parse(readFileSync(depPkgPath, "utf-8"));
167
+ if (depPkg.fragments) {
168
+ const fragmentsPath = join(dirname(depPkgPath), depPkg.fragments);
169
+ if (existsSync(fragmentsPath) && !found.includes(fragmentsPath)) {
170
+ found.push(fragmentsPath);
171
+ }
172
+ }
173
+ } catch {
174
+ }
175
+ }
176
+ } catch {
177
+ }
178
+ }
87
179
  function findFragmentsJson(startDir) {
88
180
  const found = [];
89
181
  const resolvedStart = resolve(startDir);
@@ -98,29 +190,11 @@ function findFragmentsJson(startDir) {
98
190
  if (parent === dir) break;
99
191
  dir = parent;
100
192
  }
101
- const pkgJsonPath = join(resolvedStart, "package.json");
102
- if (existsSync(pkgJsonPath)) {
103
- try {
104
- const pkgJson = JSON.parse(readFileSync(pkgJsonPath, "utf-8"));
105
- const allDeps = {
106
- ...pkgJson.dependencies,
107
- ...pkgJson.devDependencies
108
- };
109
- const localRequire = createRequire(join(resolvedStart, "noop.js"));
110
- for (const depName of Object.keys(allDeps)) {
111
- try {
112
- const depPkgPath = localRequire.resolve(`${depName}/package.json`);
113
- const depPkg = JSON.parse(readFileSync(depPkgPath, "utf-8"));
114
- if (depPkg.fragments) {
115
- const fragmentsPath = join(dirname(depPkgPath), depPkg.fragments);
116
- if (existsSync(fragmentsPath) && !found.includes(fragmentsPath)) {
117
- found.push(fragmentsPath);
118
- }
119
- }
120
- } catch {
121
- }
122
- }
123
- } catch {
193
+ findFragmentsInDeps(resolvedStart, found);
194
+ if (found.length === 0 || existsSync(join(resolvedStart, "pnpm-workspace.yaml"))) {
195
+ const workspaceDirs = getWorkspaceDirs(resolvedStart);
196
+ for (const wsDir of workspaceDirs) {
197
+ findFragmentsInDeps(wsDir, found);
124
198
  }
125
199
  }
126
200
  return found;
@@ -309,7 +383,8 @@ function createMcpServer(config) {
309
383
  }
310
384
  );
311
385
  let segmentsData = null;
312
- let packageName = null;
386
+ const segmentPackageMap = /* @__PURE__ */ new Map();
387
+ let defaultPackageName = null;
313
388
  let browserPool = null;
314
389
  let storageManager = null;
315
390
  let diffEngine = null;
@@ -326,8 +401,18 @@ function createMcpServer(config) {
326
401
  }
327
402
  const content = await readFile(paths[0], "utf-8");
328
403
  segmentsData = JSON.parse(content);
404
+ if (segmentsData.packageName) {
405
+ for (const name of Object.keys(segmentsData.segments)) {
406
+ segmentPackageMap.set(name, segmentsData.packageName);
407
+ }
408
+ }
329
409
  for (let i = 1; i < paths.length; i++) {
330
410
  const extra = JSON.parse(await readFile(paths[i], "utf-8"));
411
+ if (extra.packageName) {
412
+ for (const name of Object.keys(extra.segments)) {
413
+ segmentPackageMap.set(name, extra.packageName);
414
+ }
415
+ }
331
416
  Object.assign(segmentsData.segments, extra.segments);
332
417
  if (extra.recipes) {
333
418
  segmentsData.recipes = { ...segmentsData.recipes, ...extra.recipes };
@@ -335,14 +420,18 @@ function createMcpServer(config) {
335
420
  }
336
421
  return segmentsData;
337
422
  }
338
- async function getPackageName() {
339
- if (packageName) {
340
- return packageName;
423
+ async function getPackageName(segmentName) {
424
+ if (segmentName) {
425
+ const segPkg = segmentPackageMap.get(segmentName);
426
+ if (segPkg) return segPkg;
427
+ }
428
+ if (defaultPackageName) {
429
+ return defaultPackageName;
341
430
  }
342
431
  const data = await loadSegments();
343
432
  if (data.packageName) {
344
- packageName = data.packageName;
345
- return packageName;
433
+ defaultPackageName = data.packageName;
434
+ return defaultPackageName;
346
435
  }
347
436
  const packageJsonPath = join(config.projectRoot, "package.json");
348
437
  if (existsSync(packageJsonPath)) {
@@ -350,14 +439,14 @@ function createMcpServer(config) {
350
439
  const content = await readFile(packageJsonPath, "utf-8");
351
440
  const pkg = JSON.parse(content);
352
441
  if (pkg.name) {
353
- packageName = pkg.name;
354
- return packageName;
442
+ defaultPackageName = pkg.name;
443
+ return defaultPackageName;
355
444
  }
356
445
  } catch {
357
446
  }
358
447
  }
359
- packageName = "your-component-library";
360
- return packageName;
448
+ defaultPackageName = "your-component-library";
449
+ return defaultPackageName;
361
450
  }
362
451
  async function getBrowserPool() {
363
452
  if (!browserPool) {
@@ -663,14 +752,29 @@ function createMcpServer(config) {
663
752
  if (!segment) {
664
753
  throw new Error(`Component "${componentName}" not found. Use fragments_discover to see available components.`);
665
754
  }
666
- const pkgName = await getPackageName();
755
+ const pkgName = await getPackageName(segment.meta.name);
667
756
  let variants = segment.variants;
668
757
  if (variantName) {
669
- const filtered = variants.filter(
670
- (v) => v.name.toLowerCase() === variantName.toLowerCase()
758
+ const query = variantName.toLowerCase();
759
+ let filtered = variants.filter(
760
+ (v) => v.name.toLowerCase() === query
671
761
  );
762
+ if (filtered.length === 0) {
763
+ filtered = variants.filter(
764
+ (v) => v.name.toLowerCase().startsWith(query)
765
+ );
766
+ }
767
+ if (filtered.length === 0) {
768
+ filtered = variants.filter(
769
+ (v) => v.name.toLowerCase().includes(query)
770
+ );
771
+ }
672
772
  if (filtered.length > 0) {
673
773
  variants = filtered;
774
+ } else {
775
+ throw new Error(
776
+ `Variant "${variantName}" not found for ${componentName}. Available: ${segment.variants.map((v) => v.name).join(", ")}`
777
+ );
674
778
  }
675
779
  }
676
780
  if (maxExamples && maxExamples > 0) {
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/mcp/server.ts","../src/mcp/utils.ts","../src/mcp-bin.ts"],"sourcesContent":["import { Server } from '@modelcontextprotocol/sdk/server/index.js';\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\nimport {\n CallToolRequestSchema,\n ListToolsRequestSchema,\n type Tool,\n} from '@modelcontextprotocol/sdk/types.js';\nimport {\n BRAND,\n DEFAULTS,\n generateContext,\n type CompiledSegmentsFile,\n type VerifyResult,\n type Theme,\n} from '../core/index.js';\n// ../service is lazy-imported to avoid requiring playwright at startup.\n// Visual tools (render, fix) load it on first use.\ntype ServiceModule = typeof import('../service/index.js');\nlet _service: ServiceModule | null = null;\nasync function getService(): Promise<ServiceModule> {\n if (!_service) {\n try {\n _service = await import('../service/index.js');\n } catch {\n throw new Error(\n 'Visual tools require playwright. Install it with: npm install playwright'\n );\n }\n }\n return _service;\n}\nimport { readFile } from 'node:fs/promises';\nimport { existsSync, readFileSync } from 'node:fs';\nimport { join, dirname, resolve } from 'node:path';\nimport { createRequire } from 'node:module';\nimport { projectFields } from './utils.js';\n\n/**\n * MCP Tool names - derived from BRAND constants\n */\nconst TOOL_NAMES = {\n discover: `${BRAND.nameLower}_discover`,\n inspect: `${BRAND.nameLower}_inspect`,\n recipe: `${BRAND.nameLower}_recipe`,\n render: `${BRAND.nameLower}_render`,\n fix: `${BRAND.nameLower}_fix`,\n} as const;\n\n/**\n * Placeholder patterns to filter out from usage text.\n * These are auto-generated and provide no value to AI agents.\n */\nconst PLACEHOLDER_PATTERNS = [\n /^\\w+ component is needed$/i,\n /^Alternative component is more appropriate$/i,\n /^Use \\w+ when you need/i,\n];\n\n/**\n * Filter out placeholder text from usage arrays\n */\nfunction filterPlaceholders(items: string[] | undefined): string[] {\n if (!items) return [];\n return items.filter(item =>\n !PLACEHOLDER_PATTERNS.some(pattern => pattern.test(item.trim()))\n );\n}\n\n/**\n * Find fragments.json files:\n * 1. Walk up from startDir (for library authors with a local build)\n * 2. Read package.json deps and resolve packages with a \"fragments\" field\n *\n * Uses Node.js module resolution (createRequire) to handle all package\n * managers: pnpm symlinks, yarn PnP, monorepo hoisting, etc.\n */\nexport function findFragmentsJson(startDir: string): string[] {\n const found: string[] = [];\n const resolvedStart = resolve(startDir);\n\n // 1. Walk upward from startDir (library author flow)\n let dir = resolvedStart;\n while (true) {\n const candidate = join(dir, BRAND.outFile);\n if (existsSync(candidate)) {\n found.push(candidate);\n break;\n }\n const parent = dirname(dir);\n if (parent === dir) break;\n dir = parent;\n }\n\n // 2. Read package.json and resolve deps with \"fragments\" field\n const pkgJsonPath = join(resolvedStart, 'package.json');\n if (existsSync(pkgJsonPath)) {\n try {\n const pkgJson = JSON.parse(readFileSync(pkgJsonPath, 'utf-8'));\n const allDeps = {\n ...pkgJson.dependencies,\n ...pkgJson.devDependencies,\n };\n const localRequire = createRequire(join(resolvedStart, 'noop.js'));\n for (const depName of Object.keys(allDeps)) {\n try {\n const depPkgPath = localRequire.resolve(`${depName}/package.json`);\n const depPkg = JSON.parse(readFileSync(depPkgPath, 'utf-8'));\n if (depPkg.fragments) {\n const fragmentsPath = join(dirname(depPkgPath), depPkg.fragments);\n if (existsSync(fragmentsPath) && !found.includes(fragmentsPath)) {\n found.push(fragmentsPath);\n }\n }\n } catch {\n // Package not resolvable or unreadable, skip\n }\n }\n } catch {\n // No package.json or unreadable\n }\n }\n\n return found;\n}\n\n/**\n * MCP Server configuration\n */\nexport interface McpServerConfig {\n /** Project root directory */\n projectRoot: string;\n\n /** Viewer base URL */\n viewerUrl?: string;\n\n /** Default theme for verification */\n theme?: Theme;\n\n /** Diff threshold percentage */\n threshold?: number;\n}\n\n/**\n * Tool definitions for the MCP server — 5 consolidated tools\n */\nconst TOOLS: Tool[] = [\n {\n name: TOOL_NAMES.discover,\n description: `Discover components in the design system. Use with no params to list all components. Use 'useCase' for AI-powered suggestions. Use 'component' to find alternatives. Use 'compact' for a token-efficient overview.`,\n inputSchema: {\n type: 'object' as const,\n properties: {\n useCase: {\n type: 'string',\n description: 'Description of what you want to build — returns ranked suggestions (e.g., \"form for user email input\", \"button to submit data\")',\n },\n component: {\n type: 'string',\n description: 'Component name to find alternatives for (e.g., \"Button\")',\n },\n category: {\n type: 'string',\n description: 'Filter by category (e.g., \"actions\", \"forms\", \"layout\")',\n },\n search: {\n type: 'string',\n description: 'Search term to filter by name, description, or tags',\n },\n status: {\n type: 'string',\n enum: ['stable', 'beta', 'deprecated', 'experimental'],\n description: 'Filter by component status',\n },\n format: {\n type: 'string',\n enum: ['markdown', 'json'],\n description: 'Output format for context mode (default: markdown)',\n },\n compact: {\n type: 'boolean',\n description: 'If true, returns minimal output (just component names and categories)',\n },\n includeCode: {\n type: 'boolean',\n description: 'If true, includes code examples for each variant',\n },\n includeRelations: {\n type: 'boolean',\n description: 'If true, includes component relationships',\n },\n },\n },\n },\n {\n name: TOOL_NAMES.inspect,\n description: `Get detailed information about a specific component: props, usage guidelines, code examples, accessibility — all in one call. Use 'fields' to request only specific data for token efficiency.`,\n inputSchema: {\n type: 'object' as const,\n properties: {\n component: {\n type: 'string',\n description: 'Component name (e.g., \"Button\", \"Input\")',\n },\n fields: {\n type: 'array',\n items: { type: 'string' },\n description: 'Specific fields to return (e.g., [\"meta\", \"usage.when\", \"contract.propsSummary\", \"props\", \"examples\", \"guidelines\"]). If omitted, returns everything. Supports dot notation.',\n },\n variant: {\n type: 'string',\n description: 'Filter examples to a specific variant name (e.g., \"Default\", \"Primary\")',\n },\n maxExamples: {\n type: 'number',\n description: 'Maximum number of code examples to return (default: all)',\n },\n maxLines: {\n type: 'number',\n description: 'Maximum lines per code example (truncates longer examples)',\n },\n },\n required: ['component'],\n },\n },\n {\n name: TOOL_NAMES.recipe,\n description: `Search and retrieve composition recipes — named patterns showing how design system components wire together for common use cases (e.g., \"Login Form\", \"Settings Page\"). Returns the recipe with its code pattern.`,\n inputSchema: {\n type: 'object' as const,\n properties: {\n name: {\n type: 'string',\n description: 'Exact recipe name to retrieve (e.g., \"Login Form\")',\n },\n search: {\n type: 'string',\n description: 'Free-text search across recipe names, descriptions, tags, and components',\n },\n component: {\n type: 'string',\n description: 'Filter recipes that use a specific component (e.g., \"Button\")',\n },\n },\n },\n },\n {\n name: TOOL_NAMES.render,\n description: `Render a component and return a screenshot. Optionally compare against a stored baseline ('baseline: true') or against a Figma design ('figmaUrl'). Use this to verify your implementation looks correct.`,\n inputSchema: {\n type: 'object' as const,\n properties: {\n component: {\n type: 'string',\n description: 'Component name (e.g., \"Button\", \"Card\", \"Input\")',\n },\n variant: {\n type: 'string',\n description: 'Variant name for baseline/compare modes',\n },\n props: {\n type: 'object',\n description: 'Props to pass to the component (e.g., { \"variant\": \"primary\", \"children\": \"Click me\" })',\n },\n viewport: {\n type: 'object',\n properties: {\n width: { type: 'number', description: 'Viewport width (default: 800)' },\n height: { type: 'number', description: 'Viewport height (default: 600)' },\n },\n description: 'Optional viewport size for the render',\n },\n baseline: {\n type: 'boolean',\n description: 'If true, compares the render against the stored baseline screenshot (requires variant)',\n },\n figmaUrl: {\n type: 'string',\n description: 'Figma frame URL — if provided, compares the render against the Figma design',\n },\n theme: {\n type: 'string',\n enum: ['light', 'dark'],\n description: 'Theme for baseline verification (default: light)',\n },\n threshold: {\n type: 'number',\n description: 'Diff threshold percentage (default: 5 for baseline, 1 for Figma)',\n },\n },\n required: ['component'],\n },\n },\n {\n name: TOOL_NAMES.fix,\n description: `Generate patches to fix token compliance issues in a component. Returns unified diff patches that replace hardcoded CSS values with design token references. Use this after fragments_render identifies issues to automatically fix them.`,\n inputSchema: {\n type: 'object' as const,\n properties: {\n component: {\n type: 'string',\n description: 'Component name to generate fixes for (e.g., \"Button\", \"Card\")',\n },\n variant: {\n type: 'string',\n description: 'Specific variant to fix (optional, fixes all variants if omitted)',\n },\n fixType: {\n type: 'string',\n enum: ['token', 'all'],\n description: 'Type of fixes to generate: \"token\" for hardcoded→token replacements, \"all\" for all available fixes (default: \"all\")',\n },\n },\n required: ['component'],\n },\n },\n];\n\n/**\n * Create and configure the MCP server\n */\nexport function createMcpServer(config: McpServerConfig): Server {\n const server = new Server(\n {\n name: `${BRAND.nameLower}-mcp`,\n version: '0.0.1',\n },\n {\n capabilities: {\n tools: {},\n },\n }\n );\n\n // Lazy-loaded resources\n let segmentsData: CompiledSegmentsFile | null = null;\n let packageName: string | null = null;\n \n let browserPool: any = null;\n let storageManager: any = null;\n let diffEngine: any = null;\n let isPoolWarming = false;\n\n async function loadSegments(): Promise<CompiledSegmentsFile> {\n if (segmentsData) {\n return segmentsData;\n }\n\n const paths = findFragmentsJson(config.projectRoot);\n\n if (paths.length === 0) {\n throw new Error(\n `No ${BRAND.outFile} found. Searched ${config.projectRoot} and package.json dependencies. ` +\n `Either run \\`${BRAND.cliCommand} build\\` or install a package with a \"fragments\" field in its package.json.`\n );\n }\n\n // Load and merge all found fragments files\n const content = await readFile(paths[0], 'utf-8');\n segmentsData = JSON.parse(content) as CompiledSegmentsFile;\n\n for (let i = 1; i < paths.length; i++) {\n const extra = JSON.parse(await readFile(paths[i], 'utf-8')) as CompiledSegmentsFile;\n Object.assign(segmentsData.segments, extra.segments);\n if (extra.recipes) {\n segmentsData.recipes = { ...segmentsData.recipes, ...extra.recipes };\n }\n }\n\n return segmentsData;\n }\n\n /**\n * Get the package name for import statements.\n * Prefers packageName from fragments.json (set at build time),\n * falls back to the project's package.json name.\n */\n async function getPackageName(): Promise<string> {\n if (packageName) {\n return packageName;\n }\n\n // Prefer packageName from compiled fragments.json\n const data = await loadSegments();\n if (data.packageName) {\n packageName = data.packageName;\n return packageName;\n }\n\n // Fallback to project package.json\n const packageJsonPath = join(config.projectRoot, 'package.json');\n if (existsSync(packageJsonPath)) {\n try {\n const content = await readFile(packageJsonPath, 'utf-8');\n const pkg = JSON.parse(content) as { name?: string };\n if (pkg.name) {\n packageName = pkg.name;\n return packageName;\n }\n } catch {\n // Fall through to default\n }\n }\n\n // Default fallback\n packageName = 'your-component-library';\n return packageName;\n }\n\n /**\n * Get or create browser pool with extended idle timeout for MCP\n */\n async function getBrowserPool() {\n if (!browserPool) {\n const { BrowserPool } = await getService();\n browserPool = new BrowserPool({\n viewport: DEFAULTS.viewport,\n // 30 minute idle timeout for MCP - server runs continuously\n idleTimeoutMs: 30 * 60 * 1000,\n poolSize: 2, // Keep 2 contexts warm for faster captures\n });\n }\n return browserPool;\n }\n\n /**\n * Pre-warm browser pool in background (non-blocking)\n */\n function warmBrowserPool(): void {\n if (isPoolWarming || browserPool?.isReady) {\n return;\n }\n isPoolWarming = true;\n\n // Warm in background - don't await\n getBrowserPool().then((pool) => {\n pool.warmup().then(() => {\n isPoolWarming = false;\n }).catch(() => {\n isPoolWarming = false;\n });\n }).catch(() => {\n isPoolWarming = false;\n });\n }\n\n /**\n * Get or create storage manager\n */\n async function getStorageManager() {\n if (!storageManager) {\n const { StorageManager } = await getService();\n storageManager = new StorageManager({\n projectRoot: config.projectRoot,\n });\n await storageManager.initialize();\n }\n return storageManager;\n }\n\n /**\n * Get or create diff engine\n */\n async function getDiffEngine() {\n if (!diffEngine) {\n const { DiffEngine } = await getService();\n diffEngine = new DiffEngine(config.threshold ?? DEFAULTS.diffThreshold);\n }\n return diffEngine;\n }\n\n // Register tool listing\n server.setRequestHandler(ListToolsRequestSchema, async () => {\n return { tools: TOOLS };\n });\n\n // Register tool execution\n server.setRequestHandler(CallToolRequestSchema, async (request) => {\n const { name, arguments: args } = request.params;\n\n try {\n switch (name) {\n // ================================================================\n // DISCOVER — list, suggest, context, alternatives\n // ================================================================\n case TOOL_NAMES.discover: {\n const data = await loadSegments();\n const useCase = (args?.useCase as string) ?? undefined;\n const componentForAlts = (args?.component as string) ?? undefined;\n const category = (args?.category as string) ?? undefined;\n const search = (args?.search as string)?.toLowerCase() ?? undefined;\n const status = (args?.status as string) ?? undefined;\n const format = (args?.format as 'markdown' | 'json') ?? 'markdown';\n const compact = (args?.compact as boolean) ?? false;\n const includeCode = (args?.includeCode as boolean) ?? false;\n const includeRelations = (args?.includeRelations as boolean) ?? false;\n\n // --- Context mode: compact or format specified with no specific query ---\n if (compact || (args?.format && !useCase && !componentForAlts && !category && !search && !status)) {\n const segments = Object.values(data.segments);\n const recipes = Object.values(data.recipes ?? {});\n\n const { content: ctxContent, tokenEstimate } = generateContext(segments, {\n format,\n compact,\n include: {\n code: includeCode,\n relations: includeRelations,\n },\n }, recipes);\n\n return {\n content: [{\n type: 'text' as const,\n text: ctxContent,\n }],\n _meta: { tokenEstimate },\n };\n }\n\n // --- Suggest mode: useCase provided ---\n if (useCase) {\n const useCaseLower = useCase.toLowerCase();\n const context = ((args as Record<string, unknown>)?.context as string)?.toLowerCase() ?? '';\n const searchTerms = `${useCaseLower} ${context}`.split(/\\s+/).filter(Boolean);\n\n const synonymMap: Record<string, string[]> = {\n 'form': ['input', 'field', 'submit', 'validation'],\n 'input': ['form', 'field', 'text', 'entry'],\n 'button': ['action', 'click', 'submit', 'trigger'],\n 'action': ['button', 'click', 'trigger'],\n 'alert': ['notification', 'message', 'warning', 'error', 'feedback'],\n 'notification': ['alert', 'message', 'toast'],\n 'card': ['container', 'panel', 'box', 'content'],\n 'toggle': ['switch', 'checkbox', 'boolean', 'on/off'],\n 'switch': ['toggle', 'checkbox', 'boolean'],\n 'badge': ['tag', 'label', 'status', 'indicator'],\n 'status': ['badge', 'indicator', 'state'],\n 'login': ['auth', 'signin', 'authentication', 'form'],\n 'auth': ['login', 'signin', 'authentication'],\n };\n\n const expandedTerms = new Set(searchTerms);\n searchTerms.forEach(term => {\n const synonyms = synonymMap[term];\n if (synonyms) {\n synonyms.forEach(syn => expandedTerms.add(syn));\n }\n });\n\n const scored = Object.values(data.segments).map((s) => {\n let score = 0;\n const reasons: string[] = [];\n\n const nameLower = s.meta.name.toLowerCase();\n if (searchTerms.some((term) => nameLower.includes(term))) {\n score += 15;\n reasons.push(`Name matches search`);\n } else if (Array.from(expandedTerms).some((term) => nameLower.includes(term))) {\n score += 8;\n reasons.push(`Name matches related term`);\n }\n\n const desc = s.meta.description?.toLowerCase() ?? '';\n const descMatches = searchTerms.filter((term) => desc.includes(term));\n if (descMatches.length > 0) {\n score += descMatches.length * 6;\n reasons.push(`Description matches: ${descMatches.join(', ')}`);\n }\n\n const tags = s.meta.tags?.map((t) => t.toLowerCase()) ?? [];\n const tagMatches = searchTerms.filter((term) =>\n tags.some((tag) => tag.includes(term))\n );\n if (tagMatches.length > 0) {\n score += tagMatches.length * 4;\n reasons.push(`Tags match: ${tagMatches.join(', ')}`);\n }\n\n const whenUsed = s.usage?.when?.join(' ').toLowerCase() ?? '';\n const whenMatches = searchTerms.filter((term) => whenUsed.includes(term));\n if (whenMatches.length > 0) {\n score += whenMatches.length * 10;\n reasons.push(`Use cases match: \"${whenMatches.join(', ')}\"`);\n }\n\n const expandedWhenMatches = Array.from(expandedTerms).filter(\n (term) => !searchTerms.includes(term) && whenUsed.includes(term)\n );\n if (expandedWhenMatches.length > 0) {\n score += expandedWhenMatches.length * 5;\n reasons.push(`Related use cases: \"${expandedWhenMatches.join(', ')}\"`);\n }\n\n const cat = s.meta.category?.toLowerCase() ?? '';\n if (searchTerms.some((term) => cat.includes(term))) {\n score += 8;\n reasons.push(`Category: ${s.meta.category}`);\n }\n\n const variantText = s.variants\n .map(v => `${v.name} ${v.description || ''}`.toLowerCase())\n .join(' ');\n const variantMatches = searchTerms.filter(term => variantText.includes(term));\n if (variantMatches.length > 0) {\n score += variantMatches.length * 3;\n reasons.push(`Variants match: ${variantMatches.join(', ')}`);\n }\n\n if (s.meta.status === 'stable') {\n score += 5;\n reasons.push('Stable component');\n } else if (s.meta.status === 'beta') {\n score += 2;\n }\n\n if (s.meta.status === 'deprecated') {\n score -= 25;\n reasons.push('Deprecated - consider alternatives');\n }\n\n const filteredWhen = filterPlaceholders(s.usage?.when).slice(0, 3);\n const filteredWhenNot = filterPlaceholders(s.usage?.whenNot).slice(0, 2);\n\n let confidence: 'high' | 'medium' | 'low';\n if (score >= 25) confidence = 'high';\n else if (score >= 15) confidence = 'medium';\n else confidence = 'low';\n\n return {\n component: s.meta.name,\n category: s.meta.category,\n description: s.meta.description,\n score,\n confidence,\n reasons,\n usage: { when: filteredWhen, whenNot: filteredWhenNot },\n variantCount: s.variants.length,\n status: s.meta.status,\n };\n });\n\n const MIN_SCORE = 8;\n const filtered = scored\n .filter((s) => s.score >= MIN_SCORE)\n .sort((a, b) => b.score - a.score);\n\n const suggestions: typeof filtered = [];\n const categoryCount: Record<string, number> = {};\n for (const item of filtered) {\n const cat = item.category || 'uncategorized';\n const count = categoryCount[cat] || 0;\n if (count < 2 || suggestions.length < 3) {\n suggestions.push(item);\n categoryCount[cat] = count + 1;\n if (suggestions.length >= 5) break;\n }\n }\n\n const compositionHint = suggestions.length >= 2\n ? `These components work well together. For example, ${suggestions[0].component} can be combined with ${suggestions.slice(1, 3).map(s => s.component).join(' and ')}.`\n : undefined;\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n useCase,\n context: context || undefined,\n suggestions: suggestions.map(({ score, ...rest }) => rest),\n recommendation: suggestions.length > 0\n ? `Best match: ${suggestions[0].component} (${suggestions[0].confidence} confidence) - ${suggestions[0].description}`\n : 'No matching components found. Try different keywords or browse with fragments_discover.',\n compositionHint,\n nextStep: suggestions.length > 0\n ? `Use fragments_inspect(\"${suggestions[0].component}\") for full details.`\n : undefined,\n }, null, 2),\n }],\n };\n }\n\n // --- Alternatives mode: component provided (no useCase) ---\n if (componentForAlts) {\n const segment = Object.values(data.segments).find(\n (s) => s.meta.name.toLowerCase() === componentForAlts.toLowerCase()\n );\n\n if (!segment) {\n throw new Error(`Component \"${componentForAlts}\" not found. Use fragments_discover to see available components.`);\n }\n\n const relations = segment.relations ?? [];\n\n const referencedBy = Object.values(data.segments)\n .filter((s) =>\n s.relations?.some((r) => r.component.toLowerCase() === componentForAlts.toLowerCase())\n )\n .map((s) => ({\n component: s.meta.name,\n relationship: s.relations?.find(\n (r) => r.component.toLowerCase() === componentForAlts.toLowerCase()\n )?.relationship,\n note: s.relations?.find(\n (r) => r.component.toLowerCase() === componentForAlts.toLowerCase()\n )?.note,\n }));\n\n const sameCategory = Object.values(data.segments)\n .filter(\n (s) =>\n s.meta.category === segment.meta.category &&\n s.meta.name.toLowerCase() !== componentForAlts.toLowerCase()\n )\n .map((s) => ({\n component: s.meta.name,\n description: s.meta.description,\n }));\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n component: segment.meta.name,\n category: segment.meta.category,\n directRelations: relations,\n referencedBy,\n sameCategory,\n suggestion: relations.find((r) => r.relationship === 'alternative')\n ? `Consider ${relations.find((r) => r.relationship === 'alternative')?.component}: ${relations.find((r) => r.relationship === 'alternative')?.note}`\n : undefined,\n }, null, 2),\n }],\n };\n }\n\n // --- Default: list mode ---\n const segments = Object.values(data.segments)\n .filter((s) => {\n if (category && s.meta.category !== category) return false;\n if (status && (s.meta.status ?? 'stable') !== status) return false;\n if (search) {\n const nameMatch = s.meta.name.toLowerCase().includes(search);\n const descMatch = s.meta.description?.toLowerCase().includes(search);\n const tagMatch = s.meta.tags?.some((t) => t.toLowerCase().includes(search));\n if (!nameMatch && !descMatch && !tagMatch) return false;\n }\n return true;\n })\n .map((s) => ({\n name: s.meta.name,\n category: s.meta.category,\n description: s.meta.description,\n status: s.meta.status ?? 'stable',\n variantCount: s.variants.length,\n tags: s.meta.tags ?? [],\n }));\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n total: segments.length,\n segments,\n categories: [...new Set(segments.map((s) => s.category))],\n hint: segments.length === 0\n ? 'No components found. Try broader search terms or check available categories.'\n : segments.length > 5\n ? 'Use fragments_discover with useCase for recommendations, or fragments_inspect for details on a specific component.'\n : undefined,\n }, null, 2),\n }],\n };\n }\n\n // ================================================================\n // INSPECT — get + guidelines + example in one call\n // ================================================================\n case TOOL_NAMES.inspect: {\n const data = await loadSegments();\n const componentName = args?.component as string;\n const fields = args?.fields as string[] | undefined;\n const variantName = (args?.variant as string) ?? undefined;\n const maxExamples = args?.maxExamples as number | undefined;\n const maxLines = args?.maxLines as number | undefined;\n\n if (!componentName) {\n throw new Error('component is required');\n }\n\n const segment = Object.values(data.segments).find(\n (s) => s.meta.name.toLowerCase() === componentName.toLowerCase()\n );\n\n if (!segment) {\n throw new Error(`Component \"${componentName}\" not found. Use fragments_discover to see available components.`);\n }\n\n // Build the full inspect result combining get + guidelines + example\n const pkgName = await getPackageName();\n\n // Filter variants for examples\n let variants = segment.variants;\n if (variantName) {\n const filtered = variants.filter(\n (v) => v.name.toLowerCase() === variantName.toLowerCase()\n );\n if (filtered.length > 0) {\n variants = filtered;\n }\n }\n if (maxExamples && maxExamples > 0) {\n variants = variants.slice(0, maxExamples);\n }\n\n const truncateCode = (code: string): string => {\n if (!maxLines || maxLines <= 0) return code;\n const lines = code.split('\\n');\n if (lines.length <= maxLines) return code;\n return lines.slice(0, maxLines).join('\\n') + '\\n// ... truncated';\n };\n\n const examples = variants.map((variant) => {\n if (variant.code) {\n return {\n variant: variant.name,\n description: variant.description,\n code: truncateCode(variant.code),\n };\n }\n return {\n variant: variant.name,\n description: variant.description,\n code: `<${segment.meta.name} />`,\n note: 'No code example provided in fragment. Refer to props for customization.',\n };\n });\n\n const propsReference = Object.entries(segment.props ?? {}).map(([propName, prop]) => ({\n name: propName,\n type: prop.type,\n required: prop.required,\n default: prop.default,\n description: prop.description,\n }));\n\n const propConstraints = Object.entries(segment.props ?? {})\n .filter(([, prop]) => prop.constraints && prop.constraints.length > 0)\n .map(([pName, prop]) => ({\n prop: pName,\n constraints: prop.constraints,\n }));\n\n const fullResult = {\n // Component data (from old \"get\")\n meta: segment.meta,\n props: segment.props,\n variants: segment.variants,\n relations: segment.relations,\n contract: segment.contract,\n generated: segment._generated,\n // Guidelines (from old \"guidelines\")\n guidelines: {\n when: filterPlaceholders(segment.usage?.when),\n whenNot: filterPlaceholders(segment.usage?.whenNot),\n guidelines: segment.usage?.guidelines ?? [],\n accessibility: segment.usage?.accessibility ?? [],\n propConstraints,\n alternatives: segment.relations\n ?.filter((r) => r.relationship === 'alternative')\n .map((r) => ({\n component: r.component,\n note: r.note,\n })) ?? [],\n },\n // Examples (from old \"example\")\n examples: {\n import: `import { ${segment.meta.name} } from '${pkgName}';`,\n code: examples,\n propsReference,\n },\n };\n\n // Apply field projection if specified\n const result = fields && fields.length > 0\n ? projectFields(fullResult as unknown as Record<string, unknown>, fields)\n : fullResult;\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify(result, null, 2),\n }],\n };\n }\n\n // ================================================================\n // RECIPE — unchanged\n // ================================================================\n case TOOL_NAMES.recipe: {\n const data = await loadSegments();\n const recipeName = args?.name as string | undefined;\n const search = (args?.search as string)?.toLowerCase() ?? undefined;\n const component = (args?.component as string)?.toLowerCase() ?? undefined;\n\n const allRecipes = Object.values(data.recipes ?? {});\n\n if (allRecipes.length === 0) {\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n total: 0,\n recipes: [],\n hint: `No recipes found. Run \\`${BRAND.cliCommand} build\\` after adding .recipe.ts files.`,\n }, null, 2),\n }],\n };\n }\n\n let filtered = allRecipes;\n\n if (recipeName) {\n filtered = filtered.filter(\n r => r.name.toLowerCase() === recipeName.toLowerCase()\n );\n }\n\n if (search) {\n filtered = filtered.filter(r => {\n const haystack = [\n r.name,\n r.description,\n ...(r.tags ?? []),\n ...r.components,\n r.category,\n ].join(' ').toLowerCase();\n return haystack.includes(search);\n });\n }\n\n if (component) {\n filtered = filtered.filter(r =>\n r.components.some(c => c.toLowerCase() === component)\n );\n }\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n total: filtered.length,\n recipes: filtered,\n }, null, 2),\n }],\n };\n }\n\n // ================================================================\n // RENDER — render + verify + compare\n // ================================================================\n case TOOL_NAMES.render: {\n const componentName = args?.component as string;\n const variantName = args?.variant as string | undefined;\n const props = (args?.props as Record<string, unknown>) ?? {};\n const viewport = args?.viewport as { width?: number; height?: number } | undefined;\n const useBaseline = (args?.baseline as boolean) ?? false;\n const figmaUrl = args?.figmaUrl as string | undefined;\n const theme = (args?.theme as Theme) ?? config.theme ?? DEFAULTS.theme;\n const threshold = (args?.threshold as number) ?? (figmaUrl ? 1.0 : config.threshold ?? DEFAULTS.diffThreshold);\n\n if (!componentName) {\n return {\n content: [{\n type: 'text' as const,\n text: 'Error: component name is required',\n }],\n isError: true,\n };\n }\n\n // --- Baseline verify mode ---\n if (useBaseline) {\n if (!variantName) {\n throw new Error('variant is required when baseline is true');\n }\n\n const { Timer, CaptureEngine: CE, bufferToBase64Url: toBase64 } = await getService();\n const timer = new Timer();\n\n const storage = await getStorageManager();\n const pool = await getBrowserPool();\n const diff = await getDiffEngine();\n\n const baseline = await storage.loadBaseline(componentName, variantName, theme);\n\n if (!baseline) {\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n verdict: 'error',\n matches: false,\n diffPercentage: 0,\n screenshot: '',\n baseline: '',\n notes: [],\n error: `No baseline found for ${componentName}/${variantName}. Run \\`${BRAND.cliCommand} screenshot\\` first.`,\n timing: { renderMs: 0, captureMs: 0, diffMs: 0, totalMs: timer.elapsed() },\n } satisfies VerifyResult, null, 2),\n }],\n };\n }\n\n const viewerUrl = config.viewerUrl ?? `http://localhost:${DEFAULTS.port}`;\n const captureEngine = new CE(pool, viewerUrl);\n\n const current = await captureEngine.captureVariant(componentName, variantName, {\n theme,\n delay: DEFAULTS.captureDelayMs,\n });\n\n let diffResult;\n let matches = false;\n\n if (diff.areIdentical(current, baseline)) {\n matches = true;\n diffResult = {\n matches: true,\n diffPercentage: 0,\n diffPixelCount: 0,\n totalPixels: current.viewport.width * current.viewport.height,\n changedRegions: [],\n diffTimeMs: 0,\n };\n } else {\n diffResult = diff.compare(current, baseline, { threshold });\n matches = diffResult.matches;\n }\n\n const result: VerifyResult = {\n verdict: matches ? 'pass' : 'fail',\n matches,\n diffPercentage: diffResult.diffPercentage,\n screenshot: toBase64(current.data),\n baseline: toBase64(baseline.data),\n diffImage: diffResult.diffImage\n ? toBase64(diffResult.diffImage)\n : undefined,\n notes: matches\n ? ['Screenshot matches baseline within threshold']\n : [\n `Diff percentage (${diffResult.diffPercentage}%) exceeds threshold (${threshold}%)`,\n `${diffResult.changedRegions.length} changed region(s) detected`,\n ],\n timing: {\n renderMs: current.metadata.renderTimeMs,\n captureMs: current.metadata.captureTimeMs,\n diffMs: diffResult.diffTimeMs,\n totalMs: timer.elapsed(),\n },\n };\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify(result, null, 2),\n }],\n };\n }\n\n // --- Figma compare mode ---\n if (figmaUrl) {\n const baseUrl = config.viewerUrl ?? 'http://localhost:6006';\n const compareUrl = `${baseUrl}/fragments/compare`;\n\n try {\n const response = await fetch(compareUrl, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n component: componentName,\n variant: variantName,\n props,\n figmaUrl,\n threshold,\n }),\n });\n\n interface CompareResult {\n match?: boolean;\n diffPercentage?: number;\n threshold?: number;\n rendered?: string;\n figma?: string;\n diff?: string;\n figmaUrl?: string;\n changedRegions?: Array<{ x: number; y: number; width: number; height: number }>;\n error?: string;\n suggestion?: string;\n }\n\n const result = await response.json() as CompareResult;\n\n if (!response.ok || result.error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Compare error: ${result.error ?? 'Unknown error'}${result.suggestion ? `\\nSuggestion: ${result.suggestion}` : ''}`,\n }],\n isError: true,\n };\n }\n\n const content: Array<{ type: string; text?: string; data?: string; mimeType?: string }> = [];\n\n const summaryText = result.match\n ? `MATCH: ${componentName} matches Figma design (${result.diffPercentage}% diff, threshold: ${result.threshold}%)`\n : `MISMATCH: ${componentName} differs from Figma design by ${result.diffPercentage}% (threshold: ${result.threshold}%)`;\n\n content.push({ type: 'text' as const, text: summaryText });\n\n if (result.diff && !result.match) {\n content.push({\n type: 'image' as const,\n data: result.diff.replace('data:image/png;base64,', ''),\n mimeType: 'image/png',\n });\n content.push({\n type: 'text' as const,\n text: `Diff image above shows visual differences (red highlights). Changed regions: ${result.changedRegions?.length ?? 0}`,\n });\n }\n\n content.push({\n type: 'text' as const,\n text: JSON.stringify({\n match: result.match,\n diffPercentage: result.diffPercentage,\n threshold: result.threshold,\n figmaUrl: result.figmaUrl,\n changedRegions: result.changedRegions,\n }, null, 2),\n });\n\n return { content };\n } catch (error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Failed to compare component: ${error instanceof Error ? error.message : 'Unknown error'}. Make sure the Fragments dev server is running and FIGMA_ACCESS_TOKEN is set.`,\n }],\n isError: true,\n };\n }\n }\n\n // --- Default: pure render mode ---\n const baseUrl = config.viewerUrl ?? 'http://localhost:6006';\n const renderUrl = `${baseUrl}/fragments/render`;\n\n try {\n const response = await fetch(renderUrl, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n component: componentName,\n props,\n viewport: viewport ?? { width: 800, height: 600 },\n }),\n });\n\n const result = await response.json() as { screenshot?: string; error?: string };\n\n if (!response.ok || result.error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Render error: ${result.error ?? 'Unknown error'}`,\n }],\n isError: true,\n };\n }\n\n return {\n content: [\n {\n type: 'image' as const,\n data: result.screenshot!.replace('data:image/png;base64,', ''),\n mimeType: 'image/png',\n },\n {\n type: 'text' as const,\n text: `Successfully rendered ${componentName} with props: ${JSON.stringify(props)}`,\n },\n ],\n };\n } catch (error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Failed to render component: ${error instanceof Error ? error.message : 'Unknown error'}. Make sure the Fragments dev server is running.`,\n }],\n isError: true,\n };\n }\n }\n\n // ================================================================\n // FIX — unchanged\n // ================================================================\n case TOOL_NAMES.fix: {\n const data = await loadSegments();\n const componentName = args?.component as string;\n const variantName = (args?.variant as string) ?? undefined;\n const fixType = (args?.fixType as 'token' | 'all') ?? 'all';\n\n if (!componentName) {\n throw new Error('component is required');\n }\n\n const segment = Object.values(data.segments).find(\n (s) => s.meta.name.toLowerCase() === componentName.toLowerCase()\n );\n\n if (!segment) {\n throw new Error(`Component \"${componentName}\" not found. Use fragments_discover to see available components.`);\n }\n\n const baseUrl = config.viewerUrl ?? 'http://localhost:6006';\n const fixUrl = `${baseUrl}/fragments/fix`;\n\n try {\n const response = await fetch(fixUrl, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n component: componentName,\n variant: variantName,\n fixType,\n }),\n });\n\n interface FixResult {\n patches: Array<{ file: string; diff: string }>;\n summary: string;\n error?: string;\n }\n\n const result = await response.json() as FixResult;\n\n if (!response.ok || result.error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Fix generation error: ${result.error ?? 'Unknown error'}`,\n }],\n isError: true,\n };\n }\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n component: componentName,\n variant: variantName ?? 'all',\n fixType,\n patches: result.patches,\n summary: result.summary,\n patchCount: result.patches.length,\n nextStep: result.patches.length > 0\n ? 'Apply patches using your editor or `patch` command, then run fragments_render with baseline:true to confirm fixes.'\n : undefined,\n }, null, 2),\n }],\n };\n } catch (error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Failed to generate fixes: ${error instanceof Error ? error.message : 'Unknown error'}. Make sure the Fragments dev server is running.`,\n }],\n isError: true,\n };\n }\n }\n\n default:\n throw new Error(`Unknown tool: ${name}`);\n }\n } catch (error) {\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({\n error: error instanceof Error ? error.message : String(error),\n }),\n },\n ],\n isError: true,\n };\n }\n });\n\n // Cleanup on close\n server.onclose = async () => {\n if (browserPool) {\n await browserPool.shutdown();\n }\n };\n\n return server;\n}\n\n/**\n * Start the MCP server with stdio transport\n */\nexport async function startMcpServer(config: McpServerConfig): Promise<void> {\n const server = createMcpServer(config);\n const transport = new StdioServerTransport();\n\n await server.connect(transport);\n}\n","/**\n * Utility functions for the MCP server\n */\n\n/**\n * Extract specific fields from an object using dot notation paths.\n * E.g., projectFields(obj, ['meta.name', 'usage.when']) returns { meta: { name: ... }, usage: { when: ... } }\n *\n * @param obj - The source object to extract fields from\n * @param fields - Array of field paths (supports dot notation for nested fields)\n * @returns A new object containing only the requested fields\n */\nexport function projectFields<T extends Record<string, unknown>>(\n obj: T,\n fields: string[]\n): Partial<T> {\n if (!fields || fields.length === 0) {\n return obj;\n }\n\n const result: Record<string, unknown> = {};\n\n for (const field of fields) {\n const parts = field.split('.');\n let source: unknown = obj;\n let target = result;\n\n for (let i = 0; i < parts.length; i++) {\n const part = parts[i];\n const isLast = i === parts.length - 1;\n\n if (source === null || source === undefined || typeof source !== 'object') {\n break;\n }\n\n const sourceObj = source as Record<string, unknown>;\n const value = sourceObj[part];\n\n if (isLast) {\n // Set the final value\n target[part] = value;\n } else {\n // Create nested object if needed\n if (!(part in target)) {\n target[part] = {};\n }\n target = target[part] as Record<string, unknown>;\n source = value;\n }\n }\n }\n\n return result as Partial<T>;\n}\n","#!/usr/bin/env node\nimport { startMcpServer } from './mcp/server.js';\n\n// Parse command line arguments\nconst args = process.argv.slice(2);\nlet projectRoot = process.cwd();\nlet viewerUrl: string | undefined;\n\nfor (let i = 0; i < args.length; i++) {\n const arg = args[i];\n\n if (arg === '--project-root' || arg === '-p') {\n projectRoot = args[++i] ?? projectRoot;\n } else if (arg === '--viewer-url' || arg === '-u') {\n viewerUrl = args[++i];\n } else if (arg === '--help' || arg === '-h') {\n console.log(`\nUsage: fragments-mcp [options]\n\nOptions:\n -p, --project-root <path> Project root directory (default: cwd)\n -u, --viewer-url <url> Viewer URL (default: http://localhost:6006)\n -h, --help Show this help message\n`);\n process.exit(0);\n }\n}\n\n// Start server\nstartMcpServer({\n projectRoot,\n viewerUrl,\n}).catch((error) => {\n console.error('Failed to start MCP server:', error);\n process.exit(1);\n});\n"],"mappings":"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","names":["args","segments","viewerUrl","baseUrl"]}
1
+ {"version":3,"sources":["../src/mcp/server.ts","../src/mcp/utils.ts","../src/mcp-bin.ts"],"sourcesContent":["import { Server } from '@modelcontextprotocol/sdk/server/index.js';\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\nimport {\n CallToolRequestSchema,\n ListToolsRequestSchema,\n type Tool,\n} from '@modelcontextprotocol/sdk/types.js';\nimport {\n BRAND,\n DEFAULTS,\n generateContext,\n type CompiledSegmentsFile,\n type VerifyResult,\n type Theme,\n} from '../core/index.js';\n// ../service is lazy-imported to avoid requiring playwright at startup.\n// Visual tools (render, fix) load it on first use.\ntype ServiceModule = typeof import('../service/index.js');\nlet _service: ServiceModule | null = null;\nasync function getService(): Promise<ServiceModule> {\n if (!_service) {\n try {\n _service = await import('../service/index.js');\n } catch {\n throw new Error(\n 'Visual tools require playwright. Install it with: npm install playwright'\n );\n }\n }\n return _service;\n}\nimport { readFile } from 'node:fs/promises';\nimport { existsSync, readFileSync, readdirSync } from 'node:fs';\nimport { join, dirname, resolve } from 'node:path';\nimport { createRequire } from 'node:module';\nimport { projectFields } from './utils.js';\n\n/**\n * MCP Tool names - derived from BRAND constants\n */\nconst TOOL_NAMES = {\n discover: `${BRAND.nameLower}_discover`,\n inspect: `${BRAND.nameLower}_inspect`,\n recipe: `${BRAND.nameLower}_recipe`,\n render: `${BRAND.nameLower}_render`,\n fix: `${BRAND.nameLower}_fix`,\n} as const;\n\n/**\n * Placeholder patterns to filter out from usage text.\n * These are auto-generated and provide no value to AI agents.\n */\nconst PLACEHOLDER_PATTERNS = [\n /^\\w+ component is needed$/i,\n /^Alternative component is more appropriate$/i,\n /^Use \\w+ when you need/i,\n];\n\n/**\n * Filter out placeholder text from usage arrays\n */\nfunction filterPlaceholders(items: string[] | undefined): string[] {\n if (!items) return [];\n return items.filter(item =>\n !PLACEHOLDER_PATTERNS.some(pattern => pattern.test(item.trim()))\n );\n}\n\n/**\n * Resolve workspace directory globs (e.g. \"apps/*\") into actual paths.\n */\nfunction resolveWorkspaceGlob(baseDir: string, pattern: string): string[] {\n const parts = pattern.split('/');\n let dirs = [baseDir];\n\n for (const part of parts) {\n if (part === '**') continue; // Skip double-star recursion\n const next: string[] = [];\n for (const d of dirs) {\n if (part === '*') {\n try {\n for (const entry of readdirSync(d, { withFileTypes: true })) {\n if (entry.isDirectory() && !entry.name.startsWith('.') && entry.name !== 'node_modules') {\n next.push(join(d, entry.name));\n }\n }\n } catch { /* unreadable dir */ }\n } else {\n const candidate = join(d, part);\n if (existsSync(candidate)) next.push(candidate);\n }\n }\n dirs = next;\n }\n\n return dirs;\n}\n\n/**\n * Detect workspace directories from package.json \"workspaces\" or pnpm-workspace.yaml.\n */\nfunction getWorkspaceDirs(rootDir: string): string[] {\n const dirs: string[] = [];\n\n // npm/yarn: package.json \"workspaces\" field\n const rootPkgPath = join(rootDir, 'package.json');\n if (existsSync(rootPkgPath)) {\n try {\n const rootPkg = JSON.parse(readFileSync(rootPkgPath, 'utf-8'));\n const workspaces = Array.isArray(rootPkg.workspaces)\n ? rootPkg.workspaces\n : rootPkg.workspaces?.packages;\n if (Array.isArray(workspaces)) {\n for (const pattern of workspaces) {\n dirs.push(...resolveWorkspaceGlob(rootDir, pattern));\n }\n return dirs;\n }\n } catch { /* unreadable */ }\n }\n\n // pnpm: pnpm-workspace.yaml\n const pnpmWsPath = join(rootDir, 'pnpm-workspace.yaml');\n if (existsSync(pnpmWsPath)) {\n try {\n const content = readFileSync(pnpmWsPath, 'utf-8');\n const lines = content.split('\\n');\n let inPackages = false;\n for (const line of lines) {\n if (/^packages\\s*:/.test(line)) {\n inPackages = true;\n continue;\n }\n if (inPackages) {\n const match = line.match(/^\\s+-\\s+['\"]?([^'\"#\\n]+)['\"]?/);\n if (match) {\n dirs.push(...resolveWorkspaceGlob(rootDir, match[1].trim()));\n } else if (/^\\S/.test(line) && line.trim()) {\n break; // New top-level key, stop\n }\n }\n }\n } catch { /* unreadable */ }\n }\n\n return dirs;\n}\n\n/**\n * Scan a directory's package.json deps for packages with a \"fragments\" field.\n * Uses Node.js module resolution (createRequire) to handle all package managers.\n */\nfunction findFragmentsInDeps(dir: string, found: string[]): void {\n const pkgJsonPath = join(dir, 'package.json');\n if (!existsSync(pkgJsonPath)) return;\n\n try {\n const pkgJson = JSON.parse(readFileSync(pkgJsonPath, 'utf-8'));\n const allDeps = {\n ...pkgJson.dependencies,\n ...pkgJson.devDependencies,\n };\n const localRequire = createRequire(join(dir, 'noop.js'));\n for (const depName of Object.keys(allDeps)) {\n try {\n const depPkgPath = localRequire.resolve(`${depName}/package.json`);\n const depPkg = JSON.parse(readFileSync(depPkgPath, 'utf-8'));\n if (depPkg.fragments) {\n const fragmentsPath = join(dirname(depPkgPath), depPkg.fragments);\n if (existsSync(fragmentsPath) && !found.includes(fragmentsPath)) {\n found.push(fragmentsPath);\n }\n }\n } catch {\n // Package not resolvable or unreadable, skip\n }\n }\n } catch {\n // No package.json or unreadable\n }\n}\n\n/**\n * Find fragments.json files:\n * 1. Walk up from startDir (for library authors with a local build)\n * 2. Read package.json deps and resolve packages with a \"fragments\" field\n * 3. Check workspace packages' deps (monorepo support)\n *\n * Uses Node.js module resolution (createRequire) to handle all package\n * managers: pnpm symlinks, yarn PnP, monorepo hoisting, etc.\n */\nexport function findFragmentsJson(startDir: string): string[] {\n const found: string[] = [];\n const resolvedStart = resolve(startDir);\n\n // 1. Walk upward from startDir (library author flow)\n let dir = resolvedStart;\n while (true) {\n const candidate = join(dir, BRAND.outFile);\n if (existsSync(candidate)) {\n found.push(candidate);\n break;\n }\n const parent = dirname(dir);\n if (parent === dir) break;\n dir = parent;\n }\n\n // 2. Check root package.json deps\n findFragmentsInDeps(resolvedStart, found);\n\n // 3. Check workspace packages' deps (monorepo support)\n if (found.length === 0 || existsSync(join(resolvedStart, 'pnpm-workspace.yaml'))) {\n const workspaceDirs = getWorkspaceDirs(resolvedStart);\n for (const wsDir of workspaceDirs) {\n findFragmentsInDeps(wsDir, found);\n }\n }\n\n return found;\n}\n\n/**\n * MCP Server configuration\n */\nexport interface McpServerConfig {\n /** Project root directory */\n projectRoot: string;\n\n /** Viewer base URL */\n viewerUrl?: string;\n\n /** Default theme for verification */\n theme?: Theme;\n\n /** Diff threshold percentage */\n threshold?: number;\n}\n\n/**\n * Tool definitions for the MCP server — 5 consolidated tools\n */\nconst TOOLS: Tool[] = [\n {\n name: TOOL_NAMES.discover,\n description: `Discover components in the design system. Use with no params to list all components. Use 'useCase' for AI-powered suggestions. Use 'component' to find alternatives. Use 'compact' for a token-efficient overview.`,\n inputSchema: {\n type: 'object' as const,\n properties: {\n useCase: {\n type: 'string',\n description: 'Description of what you want to build — returns ranked suggestions (e.g., \"form for user email input\", \"button to submit data\")',\n },\n component: {\n type: 'string',\n description: 'Component name to find alternatives for (e.g., \"Button\")',\n },\n category: {\n type: 'string',\n description: 'Filter by category (e.g., \"actions\", \"forms\", \"layout\")',\n },\n search: {\n type: 'string',\n description: 'Search term to filter by name, description, or tags',\n },\n status: {\n type: 'string',\n enum: ['stable', 'beta', 'deprecated', 'experimental'],\n description: 'Filter by component status',\n },\n format: {\n type: 'string',\n enum: ['markdown', 'json'],\n description: 'Output format for context mode (default: markdown)',\n },\n compact: {\n type: 'boolean',\n description: 'If true, returns minimal output (just component names and categories)',\n },\n includeCode: {\n type: 'boolean',\n description: 'If true, includes code examples for each variant',\n },\n includeRelations: {\n type: 'boolean',\n description: 'If true, includes component relationships',\n },\n },\n },\n },\n {\n name: TOOL_NAMES.inspect,\n description: `Get detailed information about a specific component: props, usage guidelines, code examples, accessibility — all in one call. Use 'fields' to request only specific data for token efficiency.`,\n inputSchema: {\n type: 'object' as const,\n properties: {\n component: {\n type: 'string',\n description: 'Component name (e.g., \"Button\", \"Input\")',\n },\n fields: {\n type: 'array',\n items: { type: 'string' },\n description: 'Specific fields to return (e.g., [\"meta\", \"usage.when\", \"contract.propsSummary\", \"props\", \"examples\", \"guidelines\"]). If omitted, returns everything. Supports dot notation.',\n },\n variant: {\n type: 'string',\n description: 'Filter examples to a specific variant name (e.g., \"Default\", \"Primary\")',\n },\n maxExamples: {\n type: 'number',\n description: 'Maximum number of code examples to return (default: all)',\n },\n maxLines: {\n type: 'number',\n description: 'Maximum lines per code example (truncates longer examples)',\n },\n },\n required: ['component'],\n },\n },\n {\n name: TOOL_NAMES.recipe,\n description: `Search and retrieve composition recipes — named patterns showing how design system components wire together for common use cases (e.g., \"Login Form\", \"Settings Page\"). Returns the recipe with its code pattern.`,\n inputSchema: {\n type: 'object' as const,\n properties: {\n name: {\n type: 'string',\n description: 'Exact recipe name to retrieve (e.g., \"Login Form\")',\n },\n search: {\n type: 'string',\n description: 'Free-text search across recipe names, descriptions, tags, and components',\n },\n component: {\n type: 'string',\n description: 'Filter recipes that use a specific component (e.g., \"Button\")',\n },\n },\n },\n },\n {\n name: TOOL_NAMES.render,\n description: `Render a component and return a screenshot. Optionally compare against a stored baseline ('baseline: true') or against a Figma design ('figmaUrl'). Use this to verify your implementation looks correct.`,\n inputSchema: {\n type: 'object' as const,\n properties: {\n component: {\n type: 'string',\n description: 'Component name (e.g., \"Button\", \"Card\", \"Input\")',\n },\n variant: {\n type: 'string',\n description: 'Variant name for baseline/compare modes',\n },\n props: {\n type: 'object',\n description: 'Props to pass to the component (e.g., { \"variant\": \"primary\", \"children\": \"Click me\" })',\n },\n viewport: {\n type: 'object',\n properties: {\n width: { type: 'number', description: 'Viewport width (default: 800)' },\n height: { type: 'number', description: 'Viewport height (default: 600)' },\n },\n description: 'Optional viewport size for the render',\n },\n baseline: {\n type: 'boolean',\n description: 'If true, compares the render against the stored baseline screenshot (requires variant)',\n },\n figmaUrl: {\n type: 'string',\n description: 'Figma frame URL — if provided, compares the render against the Figma design',\n },\n theme: {\n type: 'string',\n enum: ['light', 'dark'],\n description: 'Theme for baseline verification (default: light)',\n },\n threshold: {\n type: 'number',\n description: 'Diff threshold percentage (default: 5 for baseline, 1 for Figma)',\n },\n },\n required: ['component'],\n },\n },\n {\n name: TOOL_NAMES.fix,\n description: `Generate patches to fix token compliance issues in a component. Returns unified diff patches that replace hardcoded CSS values with design token references. Use this after fragments_render identifies issues to automatically fix them.`,\n inputSchema: {\n type: 'object' as const,\n properties: {\n component: {\n type: 'string',\n description: 'Component name to generate fixes for (e.g., \"Button\", \"Card\")',\n },\n variant: {\n type: 'string',\n description: 'Specific variant to fix (optional, fixes all variants if omitted)',\n },\n fixType: {\n type: 'string',\n enum: ['token', 'all'],\n description: 'Type of fixes to generate: \"token\" for hardcoded→token replacements, \"all\" for all available fixes (default: \"all\")',\n },\n },\n required: ['component'],\n },\n },\n];\n\n/**\n * Create and configure the MCP server\n */\nexport function createMcpServer(config: McpServerConfig): Server {\n const server = new Server(\n {\n name: `${BRAND.nameLower}-mcp`,\n version: '0.0.1',\n },\n {\n capabilities: {\n tools: {},\n },\n }\n );\n\n // Lazy-loaded resources\n let segmentsData: CompiledSegmentsFile | null = null;\n // Per-segment package name map (segment name → package name from its source fragments.json)\n const segmentPackageMap = new Map<string, string>();\n let defaultPackageName: string | null = null;\n\n let browserPool: any = null;\n let storageManager: any = null;\n let diffEngine: any = null;\n let isPoolWarming = false;\n\n async function loadSegments(): Promise<CompiledSegmentsFile> {\n if (segmentsData) {\n return segmentsData;\n }\n\n const paths = findFragmentsJson(config.projectRoot);\n\n if (paths.length === 0) {\n throw new Error(\n `No ${BRAND.outFile} found. Searched ${config.projectRoot} and package.json dependencies. ` +\n `Either run \\`${BRAND.cliCommand} build\\` or install a package with a \"fragments\" field in its package.json.`\n );\n }\n\n // Load and merge all found fragments files\n const content = await readFile(paths[0], 'utf-8');\n segmentsData = JSON.parse(content) as CompiledSegmentsFile;\n\n // Track per-segment package names from each source file\n if (segmentsData.packageName) {\n for (const name of Object.keys(segmentsData.segments)) {\n segmentPackageMap.set(name, segmentsData.packageName);\n }\n }\n\n for (let i = 1; i < paths.length; i++) {\n const extra = JSON.parse(await readFile(paths[i], 'utf-8')) as CompiledSegmentsFile;\n // Track package name for each segment from this file\n if (extra.packageName) {\n for (const name of Object.keys(extra.segments)) {\n segmentPackageMap.set(name, extra.packageName);\n }\n }\n Object.assign(segmentsData.segments, extra.segments);\n if (extra.recipes) {\n segmentsData.recipes = { ...segmentsData.recipes, ...extra.recipes };\n }\n }\n\n return segmentsData;\n }\n\n /**\n * Get the package name for import statements for a specific component.\n * Uses per-segment tracking when multiple fragments.json files are merged,\n * falls back to the first fragments.json packageName, then project package.json.\n */\n async function getPackageName(segmentName?: string): Promise<string> {\n // Check per-segment map first (handles multi-library merges correctly)\n if (segmentName) {\n const segPkg = segmentPackageMap.get(segmentName);\n if (segPkg) return segPkg;\n }\n\n if (defaultPackageName) {\n return defaultPackageName;\n }\n\n // Prefer packageName from compiled fragments.json\n const data = await loadSegments();\n if (data.packageName) {\n defaultPackageName = data.packageName;\n return defaultPackageName;\n }\n\n // Fallback to project package.json\n const packageJsonPath = join(config.projectRoot, 'package.json');\n if (existsSync(packageJsonPath)) {\n try {\n const content = await readFile(packageJsonPath, 'utf-8');\n const pkg = JSON.parse(content) as { name?: string };\n if (pkg.name) {\n defaultPackageName = pkg.name;\n return defaultPackageName;\n }\n } catch {\n // Fall through to default\n }\n }\n\n // Default fallback\n defaultPackageName = 'your-component-library';\n return defaultPackageName;\n }\n\n /**\n * Get or create browser pool with extended idle timeout for MCP\n */\n async function getBrowserPool() {\n if (!browserPool) {\n const { BrowserPool } = await getService();\n browserPool = new BrowserPool({\n viewport: DEFAULTS.viewport,\n // 30 minute idle timeout for MCP - server runs continuously\n idleTimeoutMs: 30 * 60 * 1000,\n poolSize: 2, // Keep 2 contexts warm for faster captures\n });\n }\n return browserPool;\n }\n\n /**\n * Pre-warm browser pool in background (non-blocking)\n */\n function warmBrowserPool(): void {\n if (isPoolWarming || browserPool?.isReady) {\n return;\n }\n isPoolWarming = true;\n\n // Warm in background - don't await\n getBrowserPool().then((pool) => {\n pool.warmup().then(() => {\n isPoolWarming = false;\n }).catch(() => {\n isPoolWarming = false;\n });\n }).catch(() => {\n isPoolWarming = false;\n });\n }\n\n /**\n * Get or create storage manager\n */\n async function getStorageManager() {\n if (!storageManager) {\n const { StorageManager } = await getService();\n storageManager = new StorageManager({\n projectRoot: config.projectRoot,\n });\n await storageManager.initialize();\n }\n return storageManager;\n }\n\n /**\n * Get or create diff engine\n */\n async function getDiffEngine() {\n if (!diffEngine) {\n const { DiffEngine } = await getService();\n diffEngine = new DiffEngine(config.threshold ?? DEFAULTS.diffThreshold);\n }\n return diffEngine;\n }\n\n // Register tool listing\n server.setRequestHandler(ListToolsRequestSchema, async () => {\n return { tools: TOOLS };\n });\n\n // Register tool execution\n server.setRequestHandler(CallToolRequestSchema, async (request) => {\n const { name, arguments: args } = request.params;\n\n try {\n switch (name) {\n // ================================================================\n // DISCOVER — list, suggest, context, alternatives\n // ================================================================\n case TOOL_NAMES.discover: {\n const data = await loadSegments();\n const useCase = (args?.useCase as string) ?? undefined;\n const componentForAlts = (args?.component as string) ?? undefined;\n const category = (args?.category as string) ?? undefined;\n const search = (args?.search as string)?.toLowerCase() ?? undefined;\n const status = (args?.status as string) ?? undefined;\n const format = (args?.format as 'markdown' | 'json') ?? 'markdown';\n const compact = (args?.compact as boolean) ?? false;\n const includeCode = (args?.includeCode as boolean) ?? false;\n const includeRelations = (args?.includeRelations as boolean) ?? false;\n\n // --- Context mode: compact or format specified with no specific query ---\n if (compact || (args?.format && !useCase && !componentForAlts && !category && !search && !status)) {\n const segments = Object.values(data.segments);\n const recipes = Object.values(data.recipes ?? {});\n\n const { content: ctxContent, tokenEstimate } = generateContext(segments, {\n format,\n compact,\n include: {\n code: includeCode,\n relations: includeRelations,\n },\n }, recipes);\n\n return {\n content: [{\n type: 'text' as const,\n text: ctxContent,\n }],\n _meta: { tokenEstimate },\n };\n }\n\n // --- Suggest mode: useCase provided ---\n if (useCase) {\n const useCaseLower = useCase.toLowerCase();\n const context = ((args as Record<string, unknown>)?.context as string)?.toLowerCase() ?? '';\n const searchTerms = `${useCaseLower} ${context}`.split(/\\s+/).filter(Boolean);\n\n const synonymMap: Record<string, string[]> = {\n 'form': ['input', 'field', 'submit', 'validation'],\n 'input': ['form', 'field', 'text', 'entry'],\n 'button': ['action', 'click', 'submit', 'trigger'],\n 'action': ['button', 'click', 'trigger'],\n 'alert': ['notification', 'message', 'warning', 'error', 'feedback'],\n 'notification': ['alert', 'message', 'toast'],\n 'card': ['container', 'panel', 'box', 'content'],\n 'toggle': ['switch', 'checkbox', 'boolean', 'on/off'],\n 'switch': ['toggle', 'checkbox', 'boolean'],\n 'badge': ['tag', 'label', 'status', 'indicator'],\n 'status': ['badge', 'indicator', 'state'],\n 'login': ['auth', 'signin', 'authentication', 'form'],\n 'auth': ['login', 'signin', 'authentication'],\n };\n\n const expandedTerms = new Set(searchTerms);\n searchTerms.forEach(term => {\n const synonyms = synonymMap[term];\n if (synonyms) {\n synonyms.forEach(syn => expandedTerms.add(syn));\n }\n });\n\n const scored = Object.values(data.segments).map((s) => {\n let score = 0;\n const reasons: string[] = [];\n\n const nameLower = s.meta.name.toLowerCase();\n if (searchTerms.some((term) => nameLower.includes(term))) {\n score += 15;\n reasons.push(`Name matches search`);\n } else if (Array.from(expandedTerms).some((term) => nameLower.includes(term))) {\n score += 8;\n reasons.push(`Name matches related term`);\n }\n\n const desc = s.meta.description?.toLowerCase() ?? '';\n const descMatches = searchTerms.filter((term) => desc.includes(term));\n if (descMatches.length > 0) {\n score += descMatches.length * 6;\n reasons.push(`Description matches: ${descMatches.join(', ')}`);\n }\n\n const tags = s.meta.tags?.map((t) => t.toLowerCase()) ?? [];\n const tagMatches = searchTerms.filter((term) =>\n tags.some((tag) => tag.includes(term))\n );\n if (tagMatches.length > 0) {\n score += tagMatches.length * 4;\n reasons.push(`Tags match: ${tagMatches.join(', ')}`);\n }\n\n const whenUsed = s.usage?.when?.join(' ').toLowerCase() ?? '';\n const whenMatches = searchTerms.filter((term) => whenUsed.includes(term));\n if (whenMatches.length > 0) {\n score += whenMatches.length * 10;\n reasons.push(`Use cases match: \"${whenMatches.join(', ')}\"`);\n }\n\n const expandedWhenMatches = Array.from(expandedTerms).filter(\n (term) => !searchTerms.includes(term) && whenUsed.includes(term)\n );\n if (expandedWhenMatches.length > 0) {\n score += expandedWhenMatches.length * 5;\n reasons.push(`Related use cases: \"${expandedWhenMatches.join(', ')}\"`);\n }\n\n const cat = s.meta.category?.toLowerCase() ?? '';\n if (searchTerms.some((term) => cat.includes(term))) {\n score += 8;\n reasons.push(`Category: ${s.meta.category}`);\n }\n\n const variantText = s.variants\n .map(v => `${v.name} ${v.description || ''}`.toLowerCase())\n .join(' ');\n const variantMatches = searchTerms.filter(term => variantText.includes(term));\n if (variantMatches.length > 0) {\n score += variantMatches.length * 3;\n reasons.push(`Variants match: ${variantMatches.join(', ')}`);\n }\n\n if (s.meta.status === 'stable') {\n score += 5;\n reasons.push('Stable component');\n } else if (s.meta.status === 'beta') {\n score += 2;\n }\n\n if (s.meta.status === 'deprecated') {\n score -= 25;\n reasons.push('Deprecated - consider alternatives');\n }\n\n const filteredWhen = filterPlaceholders(s.usage?.when).slice(0, 3);\n const filteredWhenNot = filterPlaceholders(s.usage?.whenNot).slice(0, 2);\n\n let confidence: 'high' | 'medium' | 'low';\n if (score >= 25) confidence = 'high';\n else if (score >= 15) confidence = 'medium';\n else confidence = 'low';\n\n return {\n component: s.meta.name,\n category: s.meta.category,\n description: s.meta.description,\n score,\n confidence,\n reasons,\n usage: { when: filteredWhen, whenNot: filteredWhenNot },\n variantCount: s.variants.length,\n status: s.meta.status,\n };\n });\n\n const MIN_SCORE = 8;\n const filtered = scored\n .filter((s) => s.score >= MIN_SCORE)\n .sort((a, b) => b.score - a.score);\n\n const suggestions: typeof filtered = [];\n const categoryCount: Record<string, number> = {};\n for (const item of filtered) {\n const cat = item.category || 'uncategorized';\n const count = categoryCount[cat] || 0;\n if (count < 2 || suggestions.length < 3) {\n suggestions.push(item);\n categoryCount[cat] = count + 1;\n if (suggestions.length >= 5) break;\n }\n }\n\n const compositionHint = suggestions.length >= 2\n ? `These components work well together. For example, ${suggestions[0].component} can be combined with ${suggestions.slice(1, 3).map(s => s.component).join(' and ')}.`\n : undefined;\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n useCase,\n context: context || undefined,\n suggestions: suggestions.map(({ score, ...rest }) => rest),\n recommendation: suggestions.length > 0\n ? `Best match: ${suggestions[0].component} (${suggestions[0].confidence} confidence) - ${suggestions[0].description}`\n : 'No matching components found. Try different keywords or browse with fragments_discover.',\n compositionHint,\n nextStep: suggestions.length > 0\n ? `Use fragments_inspect(\"${suggestions[0].component}\") for full details.`\n : undefined,\n }, null, 2),\n }],\n };\n }\n\n // --- Alternatives mode: component provided (no useCase) ---\n if (componentForAlts) {\n const segment = Object.values(data.segments).find(\n (s) => s.meta.name.toLowerCase() === componentForAlts.toLowerCase()\n );\n\n if (!segment) {\n throw new Error(`Component \"${componentForAlts}\" not found. Use fragments_discover to see available components.`);\n }\n\n const relations = segment.relations ?? [];\n\n const referencedBy = Object.values(data.segments)\n .filter((s) =>\n s.relations?.some((r) => r.component.toLowerCase() === componentForAlts.toLowerCase())\n )\n .map((s) => ({\n component: s.meta.name,\n relationship: s.relations?.find(\n (r) => r.component.toLowerCase() === componentForAlts.toLowerCase()\n )?.relationship,\n note: s.relations?.find(\n (r) => r.component.toLowerCase() === componentForAlts.toLowerCase()\n )?.note,\n }));\n\n const sameCategory = Object.values(data.segments)\n .filter(\n (s) =>\n s.meta.category === segment.meta.category &&\n s.meta.name.toLowerCase() !== componentForAlts.toLowerCase()\n )\n .map((s) => ({\n component: s.meta.name,\n description: s.meta.description,\n }));\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n component: segment.meta.name,\n category: segment.meta.category,\n directRelations: relations,\n referencedBy,\n sameCategory,\n suggestion: relations.find((r) => r.relationship === 'alternative')\n ? `Consider ${relations.find((r) => r.relationship === 'alternative')?.component}: ${relations.find((r) => r.relationship === 'alternative')?.note}`\n : undefined,\n }, null, 2),\n }],\n };\n }\n\n // --- Default: list mode ---\n const segments = Object.values(data.segments)\n .filter((s) => {\n if (category && s.meta.category !== category) return false;\n if (status && (s.meta.status ?? 'stable') !== status) return false;\n if (search) {\n const nameMatch = s.meta.name.toLowerCase().includes(search);\n const descMatch = s.meta.description?.toLowerCase().includes(search);\n const tagMatch = s.meta.tags?.some((t) => t.toLowerCase().includes(search));\n if (!nameMatch && !descMatch && !tagMatch) return false;\n }\n return true;\n })\n .map((s) => ({\n name: s.meta.name,\n category: s.meta.category,\n description: s.meta.description,\n status: s.meta.status ?? 'stable',\n variantCount: s.variants.length,\n tags: s.meta.tags ?? [],\n }));\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n total: segments.length,\n segments,\n categories: [...new Set(segments.map((s) => s.category))],\n hint: segments.length === 0\n ? 'No components found. Try broader search terms or check available categories.'\n : segments.length > 5\n ? 'Use fragments_discover with useCase for recommendations, or fragments_inspect for details on a specific component.'\n : undefined,\n }, null, 2),\n }],\n };\n }\n\n // ================================================================\n // INSPECT — get + guidelines + example in one call\n // ================================================================\n case TOOL_NAMES.inspect: {\n const data = await loadSegments();\n const componentName = args?.component as string;\n const fields = args?.fields as string[] | undefined;\n const variantName = (args?.variant as string) ?? undefined;\n const maxExamples = args?.maxExamples as number | undefined;\n const maxLines = args?.maxLines as number | undefined;\n\n if (!componentName) {\n throw new Error('component is required');\n }\n\n const segment = Object.values(data.segments).find(\n (s) => s.meta.name.toLowerCase() === componentName.toLowerCase()\n );\n\n if (!segment) {\n throw new Error(`Component \"${componentName}\" not found. Use fragments_discover to see available components.`);\n }\n\n // Build the full inspect result combining get + guidelines + example\n const pkgName = await getPackageName(segment.meta.name);\n\n // Filter variants for examples — fuzzy match: exact → prefix → contains\n let variants = segment.variants;\n if (variantName) {\n const query = variantName.toLowerCase();\n // 1. Exact match\n let filtered = variants.filter(\n (v) => v.name.toLowerCase() === query\n );\n // 2. Prefix match (e.g. \"Dots\" matches \"Dots (Default)\")\n if (filtered.length === 0) {\n filtered = variants.filter(\n (v) => v.name.toLowerCase().startsWith(query)\n );\n }\n // 3. Contains match (e.g. \"elapsed\" matches \"With Elapsed Time\")\n if (filtered.length === 0) {\n filtered = variants.filter(\n (v) => v.name.toLowerCase().includes(query)\n );\n }\n if (filtered.length > 0) {\n variants = filtered;\n } else {\n throw new Error(\n `Variant \"${variantName}\" not found for ${componentName}. ` +\n `Available: ${segment.variants.map((v) => v.name).join(', ')}`\n );\n }\n }\n if (maxExamples && maxExamples > 0) {\n variants = variants.slice(0, maxExamples);\n }\n\n const truncateCode = (code: string): string => {\n if (!maxLines || maxLines <= 0) return code;\n const lines = code.split('\\n');\n if (lines.length <= maxLines) return code;\n return lines.slice(0, maxLines).join('\\n') + '\\n// ... truncated';\n };\n\n const examples = variants.map((variant) => {\n if (variant.code) {\n return {\n variant: variant.name,\n description: variant.description,\n code: truncateCode(variant.code),\n };\n }\n return {\n variant: variant.name,\n description: variant.description,\n code: `<${segment.meta.name} />`,\n note: 'No code example provided in fragment. Refer to props for customization.',\n };\n });\n\n const propsReference = Object.entries(segment.props ?? {}).map(([propName, prop]) => ({\n name: propName,\n type: prop.type,\n required: prop.required,\n default: prop.default,\n description: prop.description,\n }));\n\n const propConstraints = Object.entries(segment.props ?? {})\n .filter(([, prop]) => prop.constraints && prop.constraints.length > 0)\n .map(([pName, prop]) => ({\n prop: pName,\n constraints: prop.constraints,\n }));\n\n const fullResult = {\n // Component data (from old \"get\")\n meta: segment.meta,\n props: segment.props,\n variants: segment.variants,\n relations: segment.relations,\n contract: segment.contract,\n generated: segment._generated,\n // Guidelines (from old \"guidelines\")\n guidelines: {\n when: filterPlaceholders(segment.usage?.when),\n whenNot: filterPlaceholders(segment.usage?.whenNot),\n guidelines: segment.usage?.guidelines ?? [],\n accessibility: segment.usage?.accessibility ?? [],\n propConstraints,\n alternatives: segment.relations\n ?.filter((r) => r.relationship === 'alternative')\n .map((r) => ({\n component: r.component,\n note: r.note,\n })) ?? [],\n },\n // Examples (from old \"example\")\n examples: {\n import: `import { ${segment.meta.name} } from '${pkgName}';`,\n code: examples,\n propsReference,\n },\n };\n\n // Apply field projection if specified\n const result = fields && fields.length > 0\n ? projectFields(fullResult as unknown as Record<string, unknown>, fields)\n : fullResult;\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify(result, null, 2),\n }],\n };\n }\n\n // ================================================================\n // RECIPE — unchanged\n // ================================================================\n case TOOL_NAMES.recipe: {\n const data = await loadSegments();\n const recipeName = args?.name as string | undefined;\n const search = (args?.search as string)?.toLowerCase() ?? undefined;\n const component = (args?.component as string)?.toLowerCase() ?? undefined;\n\n const allRecipes = Object.values(data.recipes ?? {});\n\n if (allRecipes.length === 0) {\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n total: 0,\n recipes: [],\n hint: `No recipes found. Run \\`${BRAND.cliCommand} build\\` after adding .recipe.ts files.`,\n }, null, 2),\n }],\n };\n }\n\n let filtered = allRecipes;\n\n if (recipeName) {\n filtered = filtered.filter(\n r => r.name.toLowerCase() === recipeName.toLowerCase()\n );\n }\n\n if (search) {\n filtered = filtered.filter(r => {\n const haystack = [\n r.name,\n r.description,\n ...(r.tags ?? []),\n ...r.components,\n r.category,\n ].join(' ').toLowerCase();\n return haystack.includes(search);\n });\n }\n\n if (component) {\n filtered = filtered.filter(r =>\n r.components.some(c => c.toLowerCase() === component)\n );\n }\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n total: filtered.length,\n recipes: filtered,\n }, null, 2),\n }],\n };\n }\n\n // ================================================================\n // RENDER — render + verify + compare\n // ================================================================\n case TOOL_NAMES.render: {\n const componentName = args?.component as string;\n const variantName = args?.variant as string | undefined;\n const props = (args?.props as Record<string, unknown>) ?? {};\n const viewport = args?.viewport as { width?: number; height?: number } | undefined;\n const useBaseline = (args?.baseline as boolean) ?? false;\n const figmaUrl = args?.figmaUrl as string | undefined;\n const theme = (args?.theme as Theme) ?? config.theme ?? DEFAULTS.theme;\n const threshold = (args?.threshold as number) ?? (figmaUrl ? 1.0 : config.threshold ?? DEFAULTS.diffThreshold);\n\n if (!componentName) {\n return {\n content: [{\n type: 'text' as const,\n text: 'Error: component name is required',\n }],\n isError: true,\n };\n }\n\n // --- Baseline verify mode ---\n if (useBaseline) {\n if (!variantName) {\n throw new Error('variant is required when baseline is true');\n }\n\n const { Timer, CaptureEngine: CE, bufferToBase64Url: toBase64 } = await getService();\n const timer = new Timer();\n\n const storage = await getStorageManager();\n const pool = await getBrowserPool();\n const diff = await getDiffEngine();\n\n const baseline = await storage.loadBaseline(componentName, variantName, theme);\n\n if (!baseline) {\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n verdict: 'error',\n matches: false,\n diffPercentage: 0,\n screenshot: '',\n baseline: '',\n notes: [],\n error: `No baseline found for ${componentName}/${variantName}. Run \\`${BRAND.cliCommand} screenshot\\` first.`,\n timing: { renderMs: 0, captureMs: 0, diffMs: 0, totalMs: timer.elapsed() },\n } satisfies VerifyResult, null, 2),\n }],\n };\n }\n\n const viewerUrl = config.viewerUrl ?? `http://localhost:${DEFAULTS.port}`;\n const captureEngine = new CE(pool, viewerUrl);\n\n const current = await captureEngine.captureVariant(componentName, variantName, {\n theme,\n delay: DEFAULTS.captureDelayMs,\n });\n\n let diffResult;\n let matches = false;\n\n if (diff.areIdentical(current, baseline)) {\n matches = true;\n diffResult = {\n matches: true,\n diffPercentage: 0,\n diffPixelCount: 0,\n totalPixels: current.viewport.width * current.viewport.height,\n changedRegions: [],\n diffTimeMs: 0,\n };\n } else {\n diffResult = diff.compare(current, baseline, { threshold });\n matches = diffResult.matches;\n }\n\n const result: VerifyResult = {\n verdict: matches ? 'pass' : 'fail',\n matches,\n diffPercentage: diffResult.diffPercentage,\n screenshot: toBase64(current.data),\n baseline: toBase64(baseline.data),\n diffImage: diffResult.diffImage\n ? toBase64(diffResult.diffImage)\n : undefined,\n notes: matches\n ? ['Screenshot matches baseline within threshold']\n : [\n `Diff percentage (${diffResult.diffPercentage}%) exceeds threshold (${threshold}%)`,\n `${diffResult.changedRegions.length} changed region(s) detected`,\n ],\n timing: {\n renderMs: current.metadata.renderTimeMs,\n captureMs: current.metadata.captureTimeMs,\n diffMs: diffResult.diffTimeMs,\n totalMs: timer.elapsed(),\n },\n };\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify(result, null, 2),\n }],\n };\n }\n\n // --- Figma compare mode ---\n if (figmaUrl) {\n const baseUrl = config.viewerUrl ?? 'http://localhost:6006';\n const compareUrl = `${baseUrl}/fragments/compare`;\n\n try {\n const response = await fetch(compareUrl, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n component: componentName,\n variant: variantName,\n props,\n figmaUrl,\n threshold,\n }),\n });\n\n interface CompareResult {\n match?: boolean;\n diffPercentage?: number;\n threshold?: number;\n rendered?: string;\n figma?: string;\n diff?: string;\n figmaUrl?: string;\n changedRegions?: Array<{ x: number; y: number; width: number; height: number }>;\n error?: string;\n suggestion?: string;\n }\n\n const result = await response.json() as CompareResult;\n\n if (!response.ok || result.error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Compare error: ${result.error ?? 'Unknown error'}${result.suggestion ? `\\nSuggestion: ${result.suggestion}` : ''}`,\n }],\n isError: true,\n };\n }\n\n const content: Array<{ type: string; text?: string; data?: string; mimeType?: string }> = [];\n\n const summaryText = result.match\n ? `MATCH: ${componentName} matches Figma design (${result.diffPercentage}% diff, threshold: ${result.threshold}%)`\n : `MISMATCH: ${componentName} differs from Figma design by ${result.diffPercentage}% (threshold: ${result.threshold}%)`;\n\n content.push({ type: 'text' as const, text: summaryText });\n\n if (result.diff && !result.match) {\n content.push({\n type: 'image' as const,\n data: result.diff.replace('data:image/png;base64,', ''),\n mimeType: 'image/png',\n });\n content.push({\n type: 'text' as const,\n text: `Diff image above shows visual differences (red highlights). Changed regions: ${result.changedRegions?.length ?? 0}`,\n });\n }\n\n content.push({\n type: 'text' as const,\n text: JSON.stringify({\n match: result.match,\n diffPercentage: result.diffPercentage,\n threshold: result.threshold,\n figmaUrl: result.figmaUrl,\n changedRegions: result.changedRegions,\n }, null, 2),\n });\n\n return { content };\n } catch (error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Failed to compare component: ${error instanceof Error ? error.message : 'Unknown error'}. Make sure the Fragments dev server is running and FIGMA_ACCESS_TOKEN is set.`,\n }],\n isError: true,\n };\n }\n }\n\n // --- Default: pure render mode ---\n const baseUrl = config.viewerUrl ?? 'http://localhost:6006';\n const renderUrl = `${baseUrl}/fragments/render`;\n\n try {\n const response = await fetch(renderUrl, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n component: componentName,\n props,\n viewport: viewport ?? { width: 800, height: 600 },\n }),\n });\n\n const result = await response.json() as { screenshot?: string; error?: string };\n\n if (!response.ok || result.error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Render error: ${result.error ?? 'Unknown error'}`,\n }],\n isError: true,\n };\n }\n\n return {\n content: [\n {\n type: 'image' as const,\n data: result.screenshot!.replace('data:image/png;base64,', ''),\n mimeType: 'image/png',\n },\n {\n type: 'text' as const,\n text: `Successfully rendered ${componentName} with props: ${JSON.stringify(props)}`,\n },\n ],\n };\n } catch (error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Failed to render component: ${error instanceof Error ? error.message : 'Unknown error'}. Make sure the Fragments dev server is running.`,\n }],\n isError: true,\n };\n }\n }\n\n // ================================================================\n // FIX — unchanged\n // ================================================================\n case TOOL_NAMES.fix: {\n const data = await loadSegments();\n const componentName = args?.component as string;\n const variantName = (args?.variant as string) ?? undefined;\n const fixType = (args?.fixType as 'token' | 'all') ?? 'all';\n\n if (!componentName) {\n throw new Error('component is required');\n }\n\n const segment = Object.values(data.segments).find(\n (s) => s.meta.name.toLowerCase() === componentName.toLowerCase()\n );\n\n if (!segment) {\n throw new Error(`Component \"${componentName}\" not found. Use fragments_discover to see available components.`);\n }\n\n const baseUrl = config.viewerUrl ?? 'http://localhost:6006';\n const fixUrl = `${baseUrl}/fragments/fix`;\n\n try {\n const response = await fetch(fixUrl, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n component: componentName,\n variant: variantName,\n fixType,\n }),\n });\n\n interface FixResult {\n patches: Array<{ file: string; diff: string }>;\n summary: string;\n error?: string;\n }\n\n const result = await response.json() as FixResult;\n\n if (!response.ok || result.error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Fix generation error: ${result.error ?? 'Unknown error'}`,\n }],\n isError: true,\n };\n }\n\n return {\n content: [{\n type: 'text' as const,\n text: JSON.stringify({\n component: componentName,\n variant: variantName ?? 'all',\n fixType,\n patches: result.patches,\n summary: result.summary,\n patchCount: result.patches.length,\n nextStep: result.patches.length > 0\n ? 'Apply patches using your editor or `patch` command, then run fragments_render with baseline:true to confirm fixes.'\n : undefined,\n }, null, 2),\n }],\n };\n } catch (error) {\n return {\n content: [{\n type: 'text' as const,\n text: `Failed to generate fixes: ${error instanceof Error ? error.message : 'Unknown error'}. Make sure the Fragments dev server is running.`,\n }],\n isError: true,\n };\n }\n }\n\n default:\n throw new Error(`Unknown tool: ${name}`);\n }\n } catch (error) {\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({\n error: error instanceof Error ? error.message : String(error),\n }),\n },\n ],\n isError: true,\n };\n }\n });\n\n // Cleanup on close\n server.onclose = async () => {\n if (browserPool) {\n await browserPool.shutdown();\n }\n };\n\n return server;\n}\n\n/**\n * Start the MCP server with stdio transport\n */\nexport async function startMcpServer(config: McpServerConfig): Promise<void> {\n const server = createMcpServer(config);\n const transport = new StdioServerTransport();\n\n await server.connect(transport);\n}\n","/**\n * Utility functions for the MCP server\n */\n\n/**\n * Extract specific fields from an object using dot notation paths.\n * E.g., projectFields(obj, ['meta.name', 'usage.when']) returns { meta: { name: ... }, usage: { when: ... } }\n *\n * @param obj - The source object to extract fields from\n * @param fields - Array of field paths (supports dot notation for nested fields)\n * @returns A new object containing only the requested fields\n */\nexport function projectFields<T extends Record<string, unknown>>(\n obj: T,\n fields: string[]\n): Partial<T> {\n if (!fields || fields.length === 0) {\n return obj;\n }\n\n const result: Record<string, unknown> = {};\n\n for (const field of fields) {\n const parts = field.split('.');\n let source: unknown = obj;\n let target = result;\n\n for (let i = 0; i < parts.length; i++) {\n const part = parts[i];\n const isLast = i === parts.length - 1;\n\n if (source === null || source === undefined || typeof source !== 'object') {\n break;\n }\n\n const sourceObj = source as Record<string, unknown>;\n const value = sourceObj[part];\n\n if (isLast) {\n // Set the final value\n target[part] = value;\n } else {\n // Create nested object if needed\n if (!(part in target)) {\n target[part] = {};\n }\n target = target[part] as Record<string, unknown>;\n source = value;\n }\n }\n }\n\n return result as Partial<T>;\n}\n","#!/usr/bin/env node\nimport { startMcpServer } from './mcp/server.js';\n\n// Parse command line arguments\nconst args = process.argv.slice(2);\nlet projectRoot = process.cwd();\nlet viewerUrl: string | undefined;\n\nfor (let i = 0; i < args.length; i++) {\n const arg = args[i];\n\n if (arg === '--project-root' || arg === '-p') {\n projectRoot = args[++i] ?? projectRoot;\n } else if (arg === '--viewer-url' || arg === '-u') {\n viewerUrl = args[++i];\n } else if (arg === '--help' || arg === '-h') {\n console.log(`\nUsage: fragments-mcp [options]\n\nOptions:\n -p, --project-root <path> Project root directory (default: cwd)\n -u, --viewer-url <url> Viewer URL (default: http://localhost:6006)\n -h, --help Show this help message\n`);\n process.exit(0);\n }\n}\n\n// Start server\nstartMcpServer({\n projectRoot,\n viewerUrl,\n}).catch((error) => {\n console.error('Failed to start MCP server:', error);\n process.exit(1);\n});\n"],"mappings":"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","names":["args","segments","viewerUrl","baseUrl"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@fragments-sdk/cli",
3
- "version": "0.4.2",
3
+ "version": "0.4.4",
4
4
  "description": "CLI, MCP server, and dev tools for Fragments design system",
5
5
  "type": "module",
6
6
  "bin": {
@@ -305,4 +305,160 @@ describe('findFragmentsJson', () => {
305
305
  expect(result[0]).toBe(join(depDir, 'fragments.json'));
306
306
  });
307
307
  });
308
+
309
+ describe('Phase 3: monorepo workspace discovery', () => {
310
+ it('finds fragments via npm/yarn workspaces field', () => {
311
+ // Root has workspaces but does NOT list @my-scope/ui as a dep
312
+ const appDir = join(root, 'apps', 'web');
313
+ const depDir = join(appDir, 'node_modules', '@my-scope', 'ui');
314
+ mkdirSync(depDir, { recursive: true });
315
+
316
+ writeJson(join(root, 'package.json'), {
317
+ workspaces: ['apps/*'],
318
+ });
319
+ writeJson(join(appDir, 'package.json'), {
320
+ name: 'web',
321
+ dependencies: { '@my-scope/ui': '^1.0.0' },
322
+ });
323
+ writeJson(join(depDir, 'package.json'), {
324
+ name: '@my-scope/ui',
325
+ fragments: 'fragments.json',
326
+ });
327
+ writeFileSync(join(depDir, 'fragments.json'), '{}');
328
+
329
+ const result = findFragmentsJson(root);
330
+ expect(result.length).toBe(1);
331
+ expect(result[0]).toBe(join(depDir, 'fragments.json'));
332
+ });
333
+
334
+ it('finds fragments via yarn workspaces object format', () => {
335
+ const appDir = join(root, 'packages', 'app');
336
+ const depDir = join(appDir, 'node_modules', 'ui-lib');
337
+ mkdirSync(depDir, { recursive: true });
338
+
339
+ writeJson(join(root, 'package.json'), {
340
+ workspaces: { packages: ['packages/*'] },
341
+ });
342
+ writeJson(join(appDir, 'package.json'), {
343
+ name: 'app',
344
+ dependencies: { 'ui-lib': '^1.0.0' },
345
+ });
346
+ writeJson(join(depDir, 'package.json'), {
347
+ name: 'ui-lib',
348
+ fragments: 'fragments.json',
349
+ });
350
+ writeFileSync(join(depDir, 'fragments.json'), '{}');
351
+
352
+ const result = findFragmentsJson(root);
353
+ expect(result.length).toBe(1);
354
+ expect(result[0]).toBe(join(depDir, 'fragments.json'));
355
+ });
356
+
357
+ it('finds fragments via pnpm-workspace.yaml', () => {
358
+ const appDir = join(root, 'apps', 'web');
359
+ const depDir = join(appDir, 'node_modules', 'ui-lib');
360
+ mkdirSync(depDir, { recursive: true });
361
+
362
+ writeJson(join(root, 'package.json'), {});
363
+ writeFileSync(join(root, 'pnpm-workspace.yaml'), 'packages:\n - apps/*\n - packages/*\n');
364
+ writeJson(join(appDir, 'package.json'), {
365
+ name: 'web',
366
+ dependencies: { 'ui-lib': '^1.0.0' },
367
+ });
368
+ writeJson(join(depDir, 'package.json'), {
369
+ name: 'ui-lib',
370
+ fragments: 'fragments.json',
371
+ });
372
+ writeFileSync(join(depDir, 'fragments.json'), '{}');
373
+
374
+ const result = findFragmentsJson(root);
375
+ expect(result.length).toBe(1);
376
+ expect(result[0]).toBe(join(depDir, 'fragments.json'));
377
+ });
378
+
379
+ it('finds fragments via pnpm-workspace.yaml with quoted patterns', () => {
380
+ const appDir = join(root, 'apps', 'web');
381
+ const depDir = join(appDir, 'node_modules', 'ui-lib');
382
+ mkdirSync(depDir, { recursive: true });
383
+
384
+ writeJson(join(root, 'package.json'), {});
385
+ writeFileSync(join(root, 'pnpm-workspace.yaml'), "packages:\n - 'apps/*'\n - \"packages/*\"\n");
386
+ writeJson(join(appDir, 'package.json'), {
387
+ name: 'web',
388
+ dependencies: { 'ui-lib': '^1.0.0' },
389
+ });
390
+ writeJson(join(depDir, 'package.json'), {
391
+ name: 'ui-lib',
392
+ fragments: 'fragments.json',
393
+ });
394
+ writeFileSync(join(depDir, 'fragments.json'), '{}');
395
+
396
+ const result = findFragmentsJson(root);
397
+ expect(result.length).toBe(1);
398
+ expect(result[0]).toBe(join(depDir, 'fragments.json'));
399
+ });
400
+
401
+ it('scans multiple workspace directories', () => {
402
+ const app1 = join(root, 'apps', 'web');
403
+ const app2 = join(root, 'apps', 'admin');
404
+ const dep1 = join(app1, 'node_modules', 'ui-lib-a');
405
+ const dep2 = join(app2, 'node_modules', 'ui-lib-b');
406
+ mkdirSync(dep1, { recursive: true });
407
+ mkdirSync(dep2, { recursive: true });
408
+
409
+ writeJson(join(root, 'package.json'), {
410
+ workspaces: ['apps/*'],
411
+ });
412
+ writeJson(join(app1, 'package.json'), {
413
+ name: 'web',
414
+ dependencies: { 'ui-lib-a': '^1.0.0' },
415
+ });
416
+ writeJson(join(dep1, 'package.json'), {
417
+ name: 'ui-lib-a',
418
+ fragments: 'fragments.json',
419
+ });
420
+ writeFileSync(join(dep1, 'fragments.json'), '{}');
421
+ writeJson(join(app2, 'package.json'), {
422
+ name: 'admin',
423
+ dependencies: { 'ui-lib-b': '^1.0.0' },
424
+ });
425
+ writeJson(join(dep2, 'package.json'), {
426
+ name: 'ui-lib-b',
427
+ fragments: 'fragments.json',
428
+ });
429
+ writeFileSync(join(dep2, 'fragments.json'), '{}');
430
+
431
+ const result = findFragmentsJson(root);
432
+ expect(result.length).toBe(2);
433
+ expect(result).toContain(join(dep1, 'fragments.json'));
434
+ expect(result).toContain(join(dep2, 'fragments.json'));
435
+ });
436
+
437
+ it('deduplicates across root deps and workspace deps', () => {
438
+ // Dep is in both root and workspace — should only appear once
439
+ const appDir = join(root, 'apps', 'web');
440
+ const depDir = join(root, 'node_modules', 'ui-lib');
441
+ mkdirSync(appDir, { recursive: true });
442
+ mkdirSync(depDir, { recursive: true });
443
+
444
+ writeJson(join(root, 'package.json'), {
445
+ workspaces: ['apps/*'],
446
+ dependencies: { 'ui-lib': '^1.0.0' },
447
+ });
448
+ writeJson(join(appDir, 'package.json'), {
449
+ name: 'web',
450
+ dependencies: { 'ui-lib': '^1.0.0' },
451
+ });
452
+ writeJson(join(depDir, 'package.json'), {
453
+ name: 'ui-lib',
454
+ fragments: 'fragments.json',
455
+ });
456
+ writeFileSync(join(depDir, 'fragments.json'), '{}');
457
+
458
+ const result = findFragmentsJson(root);
459
+ // Root phase finds it, workspace phase should not duplicate
460
+ expect(result.length).toBe(1);
461
+ expect(result[0]).toBe(join(depDir, 'fragments.json'));
462
+ });
463
+ });
308
464
  });
package/src/mcp/server.ts CHANGED
@@ -30,7 +30,7 @@ async function getService(): Promise<ServiceModule> {
30
30
  return _service;
31
31
  }
32
32
  import { readFile } from 'node:fs/promises';
33
- import { existsSync, readFileSync } from 'node:fs';
33
+ import { existsSync, readFileSync, readdirSync } from 'node:fs';
34
34
  import { join, dirname, resolve } from 'node:path';
35
35
  import { createRequire } from 'node:module';
36
36
  import { projectFields } from './utils.js';
@@ -66,10 +66,125 @@ function filterPlaceholders(items: string[] | undefined): string[] {
66
66
  );
67
67
  }
68
68
 
69
+ /**
70
+ * Resolve workspace directory globs (e.g. "apps/*") into actual paths.
71
+ */
72
+ function resolveWorkspaceGlob(baseDir: string, pattern: string): string[] {
73
+ const parts = pattern.split('/');
74
+ let dirs = [baseDir];
75
+
76
+ for (const part of parts) {
77
+ if (part === '**') continue; // Skip double-star recursion
78
+ const next: string[] = [];
79
+ for (const d of dirs) {
80
+ if (part === '*') {
81
+ try {
82
+ for (const entry of readdirSync(d, { withFileTypes: true })) {
83
+ if (entry.isDirectory() && !entry.name.startsWith('.') && entry.name !== 'node_modules') {
84
+ next.push(join(d, entry.name));
85
+ }
86
+ }
87
+ } catch { /* unreadable dir */ }
88
+ } else {
89
+ const candidate = join(d, part);
90
+ if (existsSync(candidate)) next.push(candidate);
91
+ }
92
+ }
93
+ dirs = next;
94
+ }
95
+
96
+ return dirs;
97
+ }
98
+
99
+ /**
100
+ * Detect workspace directories from package.json "workspaces" or pnpm-workspace.yaml.
101
+ */
102
+ function getWorkspaceDirs(rootDir: string): string[] {
103
+ const dirs: string[] = [];
104
+
105
+ // npm/yarn: package.json "workspaces" field
106
+ const rootPkgPath = join(rootDir, 'package.json');
107
+ if (existsSync(rootPkgPath)) {
108
+ try {
109
+ const rootPkg = JSON.parse(readFileSync(rootPkgPath, 'utf-8'));
110
+ const workspaces = Array.isArray(rootPkg.workspaces)
111
+ ? rootPkg.workspaces
112
+ : rootPkg.workspaces?.packages;
113
+ if (Array.isArray(workspaces)) {
114
+ for (const pattern of workspaces) {
115
+ dirs.push(...resolveWorkspaceGlob(rootDir, pattern));
116
+ }
117
+ return dirs;
118
+ }
119
+ } catch { /* unreadable */ }
120
+ }
121
+
122
+ // pnpm: pnpm-workspace.yaml
123
+ const pnpmWsPath = join(rootDir, 'pnpm-workspace.yaml');
124
+ if (existsSync(pnpmWsPath)) {
125
+ try {
126
+ const content = readFileSync(pnpmWsPath, 'utf-8');
127
+ const lines = content.split('\n');
128
+ let inPackages = false;
129
+ for (const line of lines) {
130
+ if (/^packages\s*:/.test(line)) {
131
+ inPackages = true;
132
+ continue;
133
+ }
134
+ if (inPackages) {
135
+ const match = line.match(/^\s+-\s+['"]?([^'"#\n]+)['"]?/);
136
+ if (match) {
137
+ dirs.push(...resolveWorkspaceGlob(rootDir, match[1].trim()));
138
+ } else if (/^\S/.test(line) && line.trim()) {
139
+ break; // New top-level key, stop
140
+ }
141
+ }
142
+ }
143
+ } catch { /* unreadable */ }
144
+ }
145
+
146
+ return dirs;
147
+ }
148
+
149
+ /**
150
+ * Scan a directory's package.json deps for packages with a "fragments" field.
151
+ * Uses Node.js module resolution (createRequire) to handle all package managers.
152
+ */
153
+ function findFragmentsInDeps(dir: string, found: string[]): void {
154
+ const pkgJsonPath = join(dir, 'package.json');
155
+ if (!existsSync(pkgJsonPath)) return;
156
+
157
+ try {
158
+ const pkgJson = JSON.parse(readFileSync(pkgJsonPath, 'utf-8'));
159
+ const allDeps = {
160
+ ...pkgJson.dependencies,
161
+ ...pkgJson.devDependencies,
162
+ };
163
+ const localRequire = createRequire(join(dir, 'noop.js'));
164
+ for (const depName of Object.keys(allDeps)) {
165
+ try {
166
+ const depPkgPath = localRequire.resolve(`${depName}/package.json`);
167
+ const depPkg = JSON.parse(readFileSync(depPkgPath, 'utf-8'));
168
+ if (depPkg.fragments) {
169
+ const fragmentsPath = join(dirname(depPkgPath), depPkg.fragments);
170
+ if (existsSync(fragmentsPath) && !found.includes(fragmentsPath)) {
171
+ found.push(fragmentsPath);
172
+ }
173
+ }
174
+ } catch {
175
+ // Package not resolvable or unreadable, skip
176
+ }
177
+ }
178
+ } catch {
179
+ // No package.json or unreadable
180
+ }
181
+ }
182
+
69
183
  /**
70
184
  * Find fragments.json files:
71
185
  * 1. Walk up from startDir (for library authors with a local build)
72
186
  * 2. Read package.json deps and resolve packages with a "fragments" field
187
+ * 3. Check workspace packages' deps (monorepo support)
73
188
  *
74
189
  * Uses Node.js module resolution (createRequire) to handle all package
75
190
  * managers: pnpm symlinks, yarn PnP, monorepo hoisting, etc.
@@ -91,32 +206,14 @@ export function findFragmentsJson(startDir: string): string[] {
91
206
  dir = parent;
92
207
  }
93
208
 
94
- // 2. Read package.json and resolve deps with "fragments" field
95
- const pkgJsonPath = join(resolvedStart, 'package.json');
96
- if (existsSync(pkgJsonPath)) {
97
- try {
98
- const pkgJson = JSON.parse(readFileSync(pkgJsonPath, 'utf-8'));
99
- const allDeps = {
100
- ...pkgJson.dependencies,
101
- ...pkgJson.devDependencies,
102
- };
103
- const localRequire = createRequire(join(resolvedStart, 'noop.js'));
104
- for (const depName of Object.keys(allDeps)) {
105
- try {
106
- const depPkgPath = localRequire.resolve(`${depName}/package.json`);
107
- const depPkg = JSON.parse(readFileSync(depPkgPath, 'utf-8'));
108
- if (depPkg.fragments) {
109
- const fragmentsPath = join(dirname(depPkgPath), depPkg.fragments);
110
- if (existsSync(fragmentsPath) && !found.includes(fragmentsPath)) {
111
- found.push(fragmentsPath);
112
- }
113
- }
114
- } catch {
115
- // Package not resolvable or unreadable, skip
116
- }
117
- }
118
- } catch {
119
- // No package.json or unreadable
209
+ // 2. Check root package.json deps
210
+ findFragmentsInDeps(resolvedStart, found);
211
+
212
+ // 3. Check workspace packages' deps (monorepo support)
213
+ if (found.length === 0 || existsSync(join(resolvedStart, 'pnpm-workspace.yaml'))) {
214
+ const workspaceDirs = getWorkspaceDirs(resolvedStart);
215
+ for (const wsDir of workspaceDirs) {
216
+ findFragmentsInDeps(wsDir, found);
120
217
  }
121
218
  }
122
219
 
@@ -333,8 +430,10 @@ export function createMcpServer(config: McpServerConfig): Server {
333
430
 
334
431
  // Lazy-loaded resources
335
432
  let segmentsData: CompiledSegmentsFile | null = null;
336
- let packageName: string | null = null;
337
-
433
+ // Per-segment package name map (segment name → package name from its source fragments.json)
434
+ const segmentPackageMap = new Map<string, string>();
435
+ let defaultPackageName: string | null = null;
436
+
338
437
  let browserPool: any = null;
339
438
  let storageManager: any = null;
340
439
  let diffEngine: any = null;
@@ -358,8 +457,21 @@ export function createMcpServer(config: McpServerConfig): Server {
358
457
  const content = await readFile(paths[0], 'utf-8');
359
458
  segmentsData = JSON.parse(content) as CompiledSegmentsFile;
360
459
 
460
+ // Track per-segment package names from each source file
461
+ if (segmentsData.packageName) {
462
+ for (const name of Object.keys(segmentsData.segments)) {
463
+ segmentPackageMap.set(name, segmentsData.packageName);
464
+ }
465
+ }
466
+
361
467
  for (let i = 1; i < paths.length; i++) {
362
468
  const extra = JSON.parse(await readFile(paths[i], 'utf-8')) as CompiledSegmentsFile;
469
+ // Track package name for each segment from this file
470
+ if (extra.packageName) {
471
+ for (const name of Object.keys(extra.segments)) {
472
+ segmentPackageMap.set(name, extra.packageName);
473
+ }
474
+ }
363
475
  Object.assign(segmentsData.segments, extra.segments);
364
476
  if (extra.recipes) {
365
477
  segmentsData.recipes = { ...segmentsData.recipes, ...extra.recipes };
@@ -370,20 +482,26 @@ export function createMcpServer(config: McpServerConfig): Server {
370
482
  }
371
483
 
372
484
  /**
373
- * Get the package name for import statements.
374
- * Prefers packageName from fragments.json (set at build time),
375
- * falls back to the project's package.json name.
485
+ * Get the package name for import statements for a specific component.
486
+ * Uses per-segment tracking when multiple fragments.json files are merged,
487
+ * falls back to the first fragments.json packageName, then project package.json.
376
488
  */
377
- async function getPackageName(): Promise<string> {
378
- if (packageName) {
379
- return packageName;
489
+ async function getPackageName(segmentName?: string): Promise<string> {
490
+ // Check per-segment map first (handles multi-library merges correctly)
491
+ if (segmentName) {
492
+ const segPkg = segmentPackageMap.get(segmentName);
493
+ if (segPkg) return segPkg;
494
+ }
495
+
496
+ if (defaultPackageName) {
497
+ return defaultPackageName;
380
498
  }
381
499
 
382
500
  // Prefer packageName from compiled fragments.json
383
501
  const data = await loadSegments();
384
502
  if (data.packageName) {
385
- packageName = data.packageName;
386
- return packageName;
503
+ defaultPackageName = data.packageName;
504
+ return defaultPackageName;
387
505
  }
388
506
 
389
507
  // Fallback to project package.json
@@ -393,8 +511,8 @@ export function createMcpServer(config: McpServerConfig): Server {
393
511
  const content = await readFile(packageJsonPath, 'utf-8');
394
512
  const pkg = JSON.parse(content) as { name?: string };
395
513
  if (pkg.name) {
396
- packageName = pkg.name;
397
- return packageName;
514
+ defaultPackageName = pkg.name;
515
+ return defaultPackageName;
398
516
  }
399
517
  } catch {
400
518
  // Fall through to default
@@ -402,8 +520,8 @@ export function createMcpServer(config: McpServerConfig): Server {
402
520
  }
403
521
 
404
522
  // Default fallback
405
- packageName = 'your-component-library';
406
- return packageName;
523
+ defaultPackageName = 'your-component-library';
524
+ return defaultPackageName;
407
525
  }
408
526
 
409
527
  /**
@@ -796,16 +914,35 @@ export function createMcpServer(config: McpServerConfig): Server {
796
914
  }
797
915
 
798
916
  // Build the full inspect result combining get + guidelines + example
799
- const pkgName = await getPackageName();
917
+ const pkgName = await getPackageName(segment.meta.name);
800
918
 
801
- // Filter variants for examples
919
+ // Filter variants for examples — fuzzy match: exact → prefix → contains
802
920
  let variants = segment.variants;
803
921
  if (variantName) {
804
- const filtered = variants.filter(
805
- (v) => v.name.toLowerCase() === variantName.toLowerCase()
922
+ const query = variantName.toLowerCase();
923
+ // 1. Exact match
924
+ let filtered = variants.filter(
925
+ (v) => v.name.toLowerCase() === query
806
926
  );
927
+ // 2. Prefix match (e.g. "Dots" matches "Dots (Default)")
928
+ if (filtered.length === 0) {
929
+ filtered = variants.filter(
930
+ (v) => v.name.toLowerCase().startsWith(query)
931
+ );
932
+ }
933
+ // 3. Contains match (e.g. "elapsed" matches "With Elapsed Time")
934
+ if (filtered.length === 0) {
935
+ filtered = variants.filter(
936
+ (v) => v.name.toLowerCase().includes(query)
937
+ );
938
+ }
807
939
  if (filtered.length > 0) {
808
940
  variants = filtered;
941
+ } else {
942
+ throw new Error(
943
+ `Variant "${variantName}" not found for ${componentName}. ` +
944
+ `Available: ${segment.variants.map((v) => v.name).join(', ')}`
945
+ );
809
946
  }
810
947
  }
811
948
  if (maxExamples && maxExamples > 0) {