@decaf-ts/mcp-server 0.0.2 → 0.0.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.
Files changed (93) hide show
  1. package/LICENSE.md +86 -21
  2. package/README.md +13 -13
  3. package/dist/mcp-server.cjs +893 -187
  4. package/dist/mcp-server.esm.cjs +888 -180
  5. package/lib/McpWrapper.cjs +189 -0
  6. package/lib/McpWrapper.d.ts +101 -0
  7. package/lib/bin/cli.cjs +30 -54
  8. package/lib/bin/cli.d.ts +18 -44
  9. package/lib/constants.cjs +12 -0
  10. package/lib/constants.d.ts +8 -0
  11. package/lib/esm/McpWrapper.d.ts +101 -0
  12. package/lib/esm/McpWrapper.js +182 -0
  13. package/lib/esm/bin/cli.d.ts +18 -44
  14. package/lib/esm/bin/cli.js +28 -55
  15. package/lib/esm/constants.d.ts +8 -0
  16. package/lib/esm/constants.js +9 -0
  17. package/lib/esm/index.d.ts +5 -26
  18. package/lib/esm/index.js +6 -27
  19. package/lib/esm/mcp/index.d.ts +1 -0
  20. package/lib/esm/mcp/index.js +2 -0
  21. package/lib/esm/metadata.d.ts +9 -0
  22. package/lib/esm/metadata.js +22 -0
  23. package/lib/esm/modules/decoration/index.d.ts +0 -0
  24. package/lib/esm/modules/decoration/index.js +2 -0
  25. package/lib/esm/modules/mcp/decoration-assist.d.ts +39 -0
  26. package/lib/esm/modules/mcp/decoration-assist.js +353 -0
  27. package/lib/esm/modules/mcp/decorator-tools.d.ts +118 -0
  28. package/lib/esm/modules/mcp/decorator-tools.js +237 -0
  29. package/lib/esm/modules/mcp/index.d.ts +2 -0
  30. package/lib/esm/modules/mcp/index.js +3 -0
  31. package/lib/esm/modules/mcp/mcp-module.d.ts +230 -0
  32. package/lib/esm/modules/mcp/mcp-module.js +406 -0
  33. package/lib/esm/types.d.ts +15 -0
  34. package/lib/esm/types.js +2 -0
  35. package/lib/esm/utils.d.ts +54 -13
  36. package/lib/esm/utils.js +78 -15
  37. package/lib/index.cjs +6 -28
  38. package/lib/index.d.ts +5 -26
  39. package/lib/mcp/index.cjs +17 -0
  40. package/lib/mcp/index.d.ts +1 -0
  41. package/lib/metadata.cjs +25 -0
  42. package/lib/metadata.d.ts +9 -0
  43. package/lib/modules/decoration/index.cjs +2 -0
  44. package/lib/modules/decoration/index.d.ts +0 -0
  45. package/lib/modules/mcp/decoration-assist.cjs +360 -0
  46. package/lib/modules/mcp/decoration-assist.d.ts +39 -0
  47. package/lib/modules/mcp/decorator-tools.cjs +243 -0
  48. package/lib/modules/mcp/decorator-tools.d.ts +118 -0
  49. package/lib/modules/mcp/index.cjs +24 -0
  50. package/lib/modules/mcp/index.d.ts +2 -0
  51. package/lib/modules/mcp/mcp-module.cjs +452 -0
  52. package/lib/modules/mcp/mcp-module.d.ts +230 -0
  53. package/lib/types.cjs +3 -0
  54. package/lib/types.d.ts +15 -0
  55. package/lib/utils.cjs +116 -16
  56. package/lib/utils.d.ts +54 -13
  57. package/package.json +35 -7
  58. package/lib/esm/namespace/Class.d.ts +0 -74
  59. package/lib/esm/namespace/Class.js +0 -73
  60. package/lib/esm/namespace/Interface.d.ts +0 -17
  61. package/lib/esm/namespace/Interface.js +0 -2
  62. package/lib/esm/namespace/children/ChildClass.d.ts +0 -44
  63. package/lib/esm/namespace/children/ChildClass.js +0 -43
  64. package/lib/esm/namespace/children/ChildInterface.d.ts +0 -22
  65. package/lib/esm/namespace/children/ChildInterface.js +0 -2
  66. package/lib/esm/namespace/children/Enum.d.ts +0 -14
  67. package/lib/esm/namespace/children/Enum.js +0 -16
  68. package/lib/esm/namespace/children/function.d.ts +0 -31
  69. package/lib/esm/namespace/children/function.js +0 -33
  70. package/lib/esm/namespace/children/index.d.ts +0 -25
  71. package/lib/esm/namespace/children/index.js +0 -26
  72. package/lib/esm/namespace/index.d.ts +0 -18
  73. package/lib/esm/namespace/index.js +0 -19
  74. package/lib/esm/namespace/type.d.ts +0 -28
  75. package/lib/esm/namespace/type.js +0 -2
  76. package/lib/namespace/Class.cjs +0 -77
  77. package/lib/namespace/Class.d.ts +0 -74
  78. package/lib/namespace/Interface.cjs +0 -3
  79. package/lib/namespace/Interface.d.ts +0 -17
  80. package/lib/namespace/children/ChildClass.cjs +0 -47
  81. package/lib/namespace/children/ChildClass.d.ts +0 -44
  82. package/lib/namespace/children/ChildInterface.cjs +0 -3
  83. package/lib/namespace/children/ChildInterface.d.ts +0 -22
  84. package/lib/namespace/children/Enum.cjs +0 -19
  85. package/lib/namespace/children/Enum.d.ts +0 -14
  86. package/lib/namespace/children/function.cjs +0 -36
  87. package/lib/namespace/children/function.d.ts +0 -31
  88. package/lib/namespace/children/index.cjs +0 -42
  89. package/lib/namespace/children/index.d.ts +0 -25
  90. package/lib/namespace/index.cjs +0 -35
  91. package/lib/namespace/index.d.ts +0 -18
  92. package/lib/namespace/type.cjs +0 -3
  93. package/lib/namespace/type.d.ts +0 -28
@@ -0,0 +1,353 @@
1
+ import fs from "fs";
2
+ import path from "path";
3
+ import { z } from "zod";
4
+ import { VERSION as V, PACKAGE_NAME as PKG } from "../../metadata";
5
+ // Utility: safe read file
6
+ function readFileSafe(filePath, encoding = "utf8") {
7
+ try {
8
+ return fs.readFileSync(filePath, { encoding });
9
+ }
10
+ catch {
11
+ return undefined;
12
+ }
13
+ }
14
+ function listFilesRecursive(root, matcher) {
15
+ const out = [];
16
+ const stack = [root];
17
+ while (stack.length) {
18
+ const cur = stack.pop();
19
+ const stat = fs.statSync(cur);
20
+ if (stat.isDirectory()) {
21
+ for (const f of fs.readdirSync(cur))
22
+ stack.push(path.join(cur, f));
23
+ }
24
+ else if (!matcher || matcher(cur)) {
25
+ out.push(cur);
26
+ }
27
+ }
28
+ return out.sort();
29
+ }
30
+ // Zod Schemas (with explicit descriptions)
31
+ const analyzeRepoSchema = z
32
+ .object({
33
+ repoPath: z
34
+ .string({
35
+ description: "Relative or absolute path to the target repository inside this monorepo, e.g. './decoration'.",
36
+ })
37
+ .min(1, "repoPath is required"),
38
+ includeTests: z
39
+ .boolean({
40
+ description: "If true, analyze the tests directory (if present) to derive expected behaviors.",
41
+ })
42
+ .default(true),
43
+ includeDocs: z
44
+ .boolean({
45
+ description: "If true, analyze README.md and docs directories to extract documented features.",
46
+ })
47
+ .default(true),
48
+ })
49
+ .strict()
50
+ .describe("Analyze a local repository (e.g. ./decoration) to extract APIs, features, tests, and documentation cues.");
51
+ const enumerateCapabilitiesSchema = z
52
+ .object({
53
+ repoPath: z
54
+ .string({
55
+ description: "Relative or absolute path to the target repository to enumerate developer-facing capabilities.",
56
+ })
57
+ .min(1, "repoPath is required"),
58
+ })
59
+ .strict()
60
+ .describe("Enumerate the complete set of capabilities a developer is expected to use from the given repository.");
61
+ const planFeatureSchema = z
62
+ .object({
63
+ feature: z
64
+ .string({
65
+ description: "Natural-language description of a developer's requested feature or task to implement using the repository and available MCP tools.",
66
+ })
67
+ .min(5, "feature must describe the goal clearly"),
68
+ repoPath: z
69
+ .string({
70
+ description: "Target repository path providing the library to use, e.g. './decoration'.",
71
+ })
72
+ .default("./decoration"),
73
+ })
74
+ .strict()
75
+ .describe("Plan which MCP tools to use and in what sequence to implement a requested feature using the repository.");
76
+ // Analysis helpers (minimal yet effective, text-based to avoid heavy AST deps)
77
+ function isSourceFile(p) {
78
+ return /\.(ts|tsx|js|jsx)$/.test(p) && !p.endsWith(".d.ts");
79
+ }
80
+ function isTestFile(p) {
81
+ return /(\.test\.|\.spec\.)/.test(p);
82
+ }
83
+ function extractExports(fileContent) {
84
+ const names = new Set();
85
+ const exportRe = /(export\s+(?:default\s+)?(?:class|function|const|let|var|interface|type|enum)\s+)([A-Za-z0-9_]+)/g;
86
+ const namedRe = /export\s*\{([^}]+)\}/g;
87
+ let m;
88
+ while ((m = exportRe.exec(fileContent)))
89
+ names.add(m[2]);
90
+ while ((m = namedRe.exec(fileContent))) {
91
+ m[1]
92
+ .split(",")
93
+ .map((s) => s.trim().split(" as ")[0].trim())
94
+ .forEach((n) => {
95
+ if (n)
96
+ names.add(n);
97
+ });
98
+ }
99
+ return [...names].sort();
100
+ }
101
+ function extractDecorators(fileContent) {
102
+ const decs = new Set();
103
+ const decRe = /@([A-Za-z_][A-Za-z0-9_]*)/g;
104
+ let m;
105
+ while ((m = decRe.exec(fileContent)))
106
+ decs.add(m[1]);
107
+ return [...decs].sort();
108
+ }
109
+ function summarizeReadme(readme) {
110
+ if (!readme)
111
+ return undefined;
112
+ const lines = readme.split(/\r?\n/).filter(Boolean);
113
+ const title = lines.find((l) => /^#\s+/.test(l))?.replace(/^#\s+/, "") || "README";
114
+ const bullets = lines.filter((l) => /^[-*]\s+/.test(l)).slice(0, 20);
115
+ return { title, bullets };
116
+ }
117
+ function analyzeRepo(root) {
118
+ const src = path.join(root, "src");
119
+ const testDir = path.join(root, "tests");
120
+ const readmePath = path.join(root, "README.md");
121
+ const readme = readFileSafe(readmePath);
122
+ const files = fs.existsSync(src) ? listFilesRecursive(src, isSourceFile) : [];
123
+ const testFiles = fs.existsSync(testDir)
124
+ ? listFilesRecursive(testDir, (f) => isSourceFile(f) && isTestFile(f))
125
+ : [];
126
+ const api = {};
127
+ for (const f of files) {
128
+ const content = readFileSafe(f) || "";
129
+ api[path.relative(root, f)] = {
130
+ exports: extractExports(content),
131
+ decorators: extractDecorators(content),
132
+ };
133
+ }
134
+ const tests = {};
135
+ for (const f of testFiles) {
136
+ const content = readFileSafe(f) || "";
137
+ const mentions = Array.from(new Set([...extractExports(content), ...extractDecorators(content)])).sort();
138
+ tests[path.relative(root, f)] = { mentions };
139
+ }
140
+ return { files, testFiles, api, tests, readme: summarizeReadme(readme) };
141
+ }
142
+ // Tools
143
+ function buildAnalyzeRepositoryTool() {
144
+ return {
145
+ name: "analyze-repository",
146
+ description: "Analyze a local repository's source, tests, and docs to extract exported APIs, decorators, and test mentions.",
147
+ parameters: analyzeRepoSchema,
148
+ execute: async (input) => {
149
+ let repoRoot = path.resolve(process.cwd(), input.repoPath);
150
+ if (!fs.existsSync(repoRoot)) {
151
+ // try resolving from monorepo root (parent of current cwd)
152
+ const alt = path.resolve(process.cwd(), "..", input.repoPath);
153
+ if (fs.existsSync(alt))
154
+ repoRoot = alt;
155
+ }
156
+ if (!fs.existsSync(repoRoot)) {
157
+ // if input was absolute and still not found, try ../<basename>
158
+ const alt2 = path.resolve(process.cwd(), "..", path.basename(input.repoPath));
159
+ if (fs.existsSync(alt2))
160
+ repoRoot = alt2;
161
+ }
162
+ if (!fs.existsSync(repoRoot))
163
+ throw new Error(`Repository not found at ${repoRoot}`);
164
+ const result = analyzeRepo(repoRoot);
165
+ return {
166
+ content: [{ type: "text", text: JSON.stringify(result, null, 2) }],
167
+ };
168
+ },
169
+ };
170
+ }
171
+ function deriveCapabilities(analysis) {
172
+ const cap = new Set();
173
+ // heuristics: if decorators like Decoration, flavouredAs, extend, override appear, add capabilities
174
+ const allDecs = new Set();
175
+ for (const k of Object.keys(analysis.api)) {
176
+ for (const d of analysis.api[k].decorators)
177
+ allDecs.add(d);
178
+ for (const e of analysis.api[k].exports)
179
+ if (/Decoration|decorate|Builder|Flavour/i.test(e))
180
+ cap.add("use-decoration-api");
181
+ }
182
+ if ([...allDecs].some((d) => /override|extend/i.test(d)))
183
+ cap.add("override-and-extend-decorations");
184
+ if (Object.keys(analysis.tests).length > 0)
185
+ cap.add("validate-with-tests");
186
+ if (analysis.readme)
187
+ cap.add("follow-readme-guides");
188
+ return [...cap].sort();
189
+ }
190
+ function buildEnumerateCapabilitiesTool() {
191
+ return {
192
+ name: "enumerate-capabilities",
193
+ description: "Enumerate developer-facing capabilities of the given repository, inferred from code, tests, and docs.",
194
+ parameters: enumerateCapabilitiesSchema,
195
+ execute: async (input) => {
196
+ let repoRoot = path.resolve(process.cwd(), input.repoPath);
197
+ if (!fs.existsSync(repoRoot)) {
198
+ const alt = path.resolve(process.cwd(), "..", input.repoPath);
199
+ if (fs.existsSync(alt))
200
+ repoRoot = alt;
201
+ }
202
+ if (!fs.existsSync(repoRoot)) {
203
+ const alt2 = path.resolve(process.cwd(), "..", path.basename(input.repoPath));
204
+ if (fs.existsSync(alt2))
205
+ repoRoot = alt2;
206
+ }
207
+ if (!fs.existsSync(repoRoot))
208
+ throw new Error(`Repository not found at ${repoRoot}`);
209
+ const analysis = analyzeRepo(repoRoot);
210
+ const capabilities = deriveCapabilities(analysis);
211
+ return {
212
+ content: [
213
+ {
214
+ type: "text",
215
+ text: JSON.stringify({
216
+ capabilities,
217
+ analysisSummary: {
218
+ files: analysis.files.length,
219
+ testFiles: analysis.testFiles.length,
220
+ readme: analysis.readme?.title,
221
+ },
222
+ }, null, 2),
223
+ },
224
+ ],
225
+ };
226
+ },
227
+ };
228
+ }
229
+ function buildPlanFeatureTool() {
230
+ return {
231
+ name: "plan-feature-implementation",
232
+ description: "Given a feature request, select appropriate MCP tools (including existing and new ones) and produce an execution plan.",
233
+ parameters: planFeatureSchema,
234
+ execute: async (input) => {
235
+ const steps = [];
236
+ let i = 1;
237
+ steps.push({
238
+ step: i++,
239
+ action: "Analyze repository to enumerate APIs and decorators",
240
+ tool: "analyze-repository",
241
+ arguments: { repoPath: input.repoPath },
242
+ rationale: "Understand available building blocks.",
243
+ });
244
+ steps.push({
245
+ step: i++,
246
+ action: "List capabilities expected for developers",
247
+ tool: "enumerate-capabilities",
248
+ arguments: { repoPath: input.repoPath },
249
+ rationale: "Align the plan with supported capabilities.",
250
+ });
251
+ // Suggest existing generic tools from mcp-module
252
+ steps.push({
253
+ step: i++,
254
+ action: "Select documentation prompt and gather relevant source file(s)",
255
+ tool: "document-code",
256
+ arguments: { filePath: "<target-file>" },
257
+ rationale: "Provide context and instructions for changes.",
258
+ });
259
+ steps.push({
260
+ step: i++,
261
+ action: "Apply code changes using unified diff patch",
262
+ tool: "apply-code-change",
263
+ arguments: {
264
+ filePath: "<target-file>",
265
+ patch: "<unified-diff>",
266
+ dryRun: true,
267
+ },
268
+ rationale: "Validate changes safely before committing.",
269
+ });
270
+ steps.push({
271
+ step: i++,
272
+ action: "Commit code changes",
273
+ tool: "apply-code-change",
274
+ arguments: {
275
+ filePath: "<target-file>",
276
+ patch: "<unified-diff>",
277
+ dryRun: false,
278
+ },
279
+ rationale: "Persist the update.",
280
+ });
281
+ // If decoration-related terms present, suggest decorator tools
282
+ if (/decorat|flavour|override|extend|builder/i.test(input.feature)) {
283
+ steps.unshift({
284
+ step: 0,
285
+ action: "Use decorator tooling to insert/remove/modify decorators",
286
+ tool: "decorator-tools",
287
+ arguments: { action: "help" },
288
+ rationale: "Leverage specialized utilities for decoration patterns.",
289
+ });
290
+ steps.forEach((s, idx) => (s.step = idx + 1));
291
+ }
292
+ return {
293
+ content: [
294
+ {
295
+ type: "text",
296
+ text: JSON.stringify({
297
+ plan: steps,
298
+ notes: "Replace placeholder arguments like <target-file> and <unified-diff> based on the analysis output.",
299
+ }, null, 2),
300
+ },
301
+ ],
302
+ };
303
+ },
304
+ };
305
+ }
306
+ function buildPrompts(repoPath) {
307
+ return [
308
+ {
309
+ name: "decoration-overview",
310
+ description: "High-level guidance on using the decoration library: key exports, decorators, and common workflows.",
311
+ load: async () => `You are assisting with the Decaf.ts decoration module located at ${repoPath}. Prefer using exported builders and decorators over ad-hoc patterns.\n\nProvide a concise, actionable overview of how to use the decoration APIs for extending and overriding behaviors.`,
312
+ },
313
+ ];
314
+ }
315
+ function buildResourceTemplates(repoPath) {
316
+ const root = path.resolve(process.cwd(), repoPath);
317
+ return [
318
+ {
319
+ name: "decoration-src",
320
+ description: "Read a file from the decoration/src tree by relative path.",
321
+ mimeType: "text/plain",
322
+ uriTemplate: "decoration://src/{path}",
323
+ arguments: [
324
+ {
325
+ name: "path",
326
+ description: "Path under decoration/src to load, e.g. 'decoration/types.ts'",
327
+ required: true,
328
+ },
329
+ ],
330
+ load: async ({ path: rel }) => {
331
+ const abs = path.join(root, "src", rel);
332
+ const text = readFileSafe(abs) ?? "";
333
+ return { text };
334
+ },
335
+ },
336
+ ];
337
+ }
338
+ export default function enrich(mcp) {
339
+ // Register tools
340
+ mcp.addTool(buildAnalyzeRepositoryTool());
341
+ mcp.addTool(buildEnumerateCapabilitiesTool());
342
+ mcp.addTool(buildPlanFeatureTool());
343
+ // Prompts/resources
344
+ const repoPath = "./decoration";
345
+ for (const p of buildPrompts(repoPath))
346
+ mcp.addPrompt(p);
347
+ for (const r of buildResourceTemplates(repoPath))
348
+ mcp.addResourceTemplate(r);
349
+ return mcp;
350
+ }
351
+ export const VERSION = V;
352
+ export const PACKAGE_NAME = `${PKG}/decoration-assist`;
353
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decoration-assist.js","sourceRoot":"","sources":["../../../../src/modules/mcp/decoration-assist.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from \"fs\";\nimport path from \"path\";\nimport { z } from \"zod\";\nimport type { FastMCP, Tool, InputPrompt } from \"fastmcp\";\nimport { VERSION as V, PACKAGE_NAME as PKG } from \"../../metadata\";\n\n// Utility: safe read file\nfunction readFileSafe(\n  filePath: string,\n  encoding: BufferEncoding = \"utf8\"\n): string | undefined {\n  try {\n    return fs.readFileSync(filePath, { encoding });\n  } catch {\n    return undefined;\n  }\n}\n\nfunction listFilesRecursive(\n  root: string,\n  matcher?: (p: string) => boolean\n): string[] {\n  const out: string[] = [];\n  const stack: string[] = [root];\n  while (stack.length) {\n    const cur = stack.pop()!;\n    const stat = fs.statSync(cur);\n    if (stat.isDirectory()) {\n      for (const f of fs.readdirSync(cur)) stack.push(path.join(cur, f));\n    } else if (!matcher || matcher(cur)) {\n      out.push(cur);\n    }\n  }\n  return out.sort();\n}\n\n// Zod Schemas (with explicit descriptions)\nconst analyzeRepoSchema = z\n  .object({\n    repoPath: z\n      .string({\n        description:\n          \"Relative or absolute path to the target repository inside this monorepo, e.g. './decoration'.\",\n      })\n      .min(1, \"repoPath is required\"),\n    includeTests: z\n      .boolean({\n        description:\n          \"If true, analyze the tests directory (if present) to derive expected behaviors.\",\n      })\n      .default(true),\n    includeDocs: z\n      .boolean({\n        description:\n          \"If true, analyze README.md and docs directories to extract documented features.\",\n      })\n      .default(true),\n  })\n  .strict()\n  .describe(\n    \"Analyze a local repository (e.g. ./decoration) to extract APIs, features, tests, and documentation cues.\"\n  );\n\nconst enumerateCapabilitiesSchema = z\n  .object({\n    repoPath: z\n      .string({\n        description:\n          \"Relative or absolute path to the target repository to enumerate developer-facing capabilities.\",\n      })\n      .min(1, \"repoPath is required\"),\n  })\n  .strict()\n  .describe(\n    \"Enumerate the complete set of capabilities a developer is expected to use from the given repository.\"\n  );\n\nconst planFeatureSchema = z\n  .object({\n    feature: z\n      .string({\n        description:\n          \"Natural-language description of a developer's requested feature or task to implement using the repository and available MCP tools.\",\n      })\n      .min(5, \"feature must describe the goal clearly\"),\n    repoPath: z\n      .string({\n        description:\n          \"Target repository path providing the library to use, e.g. './decoration'.\",\n      })\n      .default(\"./decoration\"),\n  })\n  .strict()\n  .describe(\n    \"Plan which MCP tools to use and in what sequence to implement a requested feature using the repository.\"\n  );\n\n// Types\nexport type AnalyzeRepoArgs = z.infer<typeof analyzeRepoSchema>;\nexport type EnumerateCapabilitiesArgs = z.infer<\n  typeof enumerateCapabilitiesSchema\n>;\nexport type PlanFeatureArgs = z.infer<typeof planFeatureSchema>;\n\n// Analysis helpers (minimal yet effective, text-based to avoid heavy AST deps)\nfunction isSourceFile(p: string) {\n  return /\\.(ts|tsx|js|jsx)$/.test(p) && !p.endsWith(\".d.ts\");\n}\nfunction isTestFile(p: string) {\n  return /(\\.test\\.|\\.spec\\.)/.test(p);\n}\n\nfunction extractExports(fileContent: string): string[] {\n  const names = new Set<string>();\n  const exportRe =\n    /(export\\s+(?:default\\s+)?(?:class|function|const|let|var|interface|type|enum)\\s+)([A-Za-z0-9_]+)/g;\n  const namedRe = /export\\s*\\{([^}]+)\\}/g;\n  let m: RegExpExecArray | null;\n  while ((m = exportRe.exec(fileContent))) names.add(m[2]);\n  while ((m = namedRe.exec(fileContent))) {\n    m[1]\n      .split(\",\")\n      .map((s) => s.trim().split(\" as \")[0].trim())\n      .forEach((n) => {\n        if (n) names.add(n);\n      });\n  }\n  return [...names].sort();\n}\n\nfunction extractDecorators(fileContent: string): string[] {\n  const decs = new Set<string>();\n  const decRe = /@([A-Za-z_][A-Za-z0-9_]*)/g;\n  let m: RegExpExecArray | null;\n  while ((m = decRe.exec(fileContent))) decs.add(m[1]);\n  return [...decs].sort();\n}\n\nfunction summarizeReadme(readme?: string) {\n  if (!readme) return undefined;\n  const lines = readme.split(/\\r?\\n/).filter(Boolean);\n  const title =\n    lines.find((l) => /^#\\s+/.test(l))?.replace(/^#\\s+/, \"\") || \"README\";\n  const bullets = lines.filter((l) => /^[-*]\\s+/.test(l)).slice(0, 20);\n  return { title, bullets };\n}\n\nfunction analyzeRepo(root: string) {\n  const src = path.join(root, \"src\");\n  const testDir = path.join(root, \"tests\");\n  const readmePath = path.join(root, \"README.md\");\n  const readme = readFileSafe(readmePath);\n\n  const files = fs.existsSync(src) ? listFilesRecursive(src, isSourceFile) : [];\n  const testFiles = fs.existsSync(testDir)\n    ? listFilesRecursive(testDir, (f) => isSourceFile(f) && isTestFile(f))\n    : [];\n\n  const api: Record<string, { exports: string[]; decorators: string[] }> = {};\n  for (const f of files) {\n    const content = readFileSafe(f) || \"\";\n    api[path.relative(root, f)] = {\n      exports: extractExports(content),\n      decorators: extractDecorators(content),\n    };\n  }\n  const tests: Record<string, { mentions: string[] }> = {};\n  for (const f of testFiles) {\n    const content = readFileSafe(f) || \"\";\n    const mentions = Array.from(\n      new Set([...extractExports(content), ...extractDecorators(content)])\n    ).sort();\n    tests[path.relative(root, f)] = { mentions };\n  }\n  return { files, testFiles, api, tests, readme: summarizeReadme(readme) };\n}\n\n// Tools\nfunction buildAnalyzeRepositoryTool(): Tool<\n  undefined,\n  typeof analyzeRepoSchema\n> {\n  return {\n    name: \"analyze-repository\",\n    description:\n      \"Analyze a local repository's source, tests, and docs to extract exported APIs, decorators, and test mentions.\",\n    parameters: analyzeRepoSchema,\n    execute: async (input) => {\n      let repoRoot = path.resolve(process.cwd(), input.repoPath);\n      if (!fs.existsSync(repoRoot)) {\n        // try resolving from monorepo root (parent of current cwd)\n        const alt = path.resolve(process.cwd(), \"..\", input.repoPath);\n        if (fs.existsSync(alt)) repoRoot = alt;\n      }\n      if (!fs.existsSync(repoRoot)) {\n        // if input was absolute and still not found, try ../<basename>\n        const alt2 = path.resolve(\n          process.cwd(),\n          \"..\",\n          path.basename(input.repoPath)\n        );\n        if (fs.existsSync(alt2)) repoRoot = alt2;\n      }\n      if (!fs.existsSync(repoRoot))\n        throw new Error(`Repository not found at ${repoRoot}`);\n      const result = analyzeRepo(repoRoot);\n      return {\n        content: [{ type: \"text\", text: JSON.stringify(result, null, 2) }],\n      };\n    },\n  };\n}\n\nfunction deriveCapabilities(\n  analysis: ReturnType<typeof analyzeRepo>\n): string[] {\n  const cap = new Set<string>();\n  // heuristics: if decorators like Decoration, flavouredAs, extend, override appear, add capabilities\n  const allDecs = new Set<string>();\n  for (const k of Object.keys(analysis.api)) {\n    for (const d of analysis.api[k].decorators) allDecs.add(d);\n    for (const e of analysis.api[k].exports)\n      if (/Decoration|decorate|Builder|Flavour/i.test(e))\n        cap.add(\"use-decoration-api\");\n  }\n  if ([...allDecs].some((d) => /override|extend/i.test(d)))\n    cap.add(\"override-and-extend-decorations\");\n  if (Object.keys(analysis.tests).length > 0) cap.add(\"validate-with-tests\");\n  if (analysis.readme) cap.add(\"follow-readme-guides\");\n  return [...cap].sort();\n}\n\nfunction buildEnumerateCapabilitiesTool(): Tool<\n  undefined,\n  typeof enumerateCapabilitiesSchema\n> {\n  return {\n    name: \"enumerate-capabilities\",\n    description:\n      \"Enumerate developer-facing capabilities of the given repository, inferred from code, tests, and docs.\",\n    parameters: enumerateCapabilitiesSchema,\n    execute: async (input) => {\n      let repoRoot = path.resolve(process.cwd(), input.repoPath);\n      if (!fs.existsSync(repoRoot)) {\n        const alt = path.resolve(process.cwd(), \"..\", input.repoPath);\n        if (fs.existsSync(alt)) repoRoot = alt;\n      }\n      if (!fs.existsSync(repoRoot)) {\n        const alt2 = path.resolve(\n          process.cwd(),\n          \"..\",\n          path.basename(input.repoPath)\n        );\n        if (fs.existsSync(alt2)) repoRoot = alt2;\n      }\n      if (!fs.existsSync(repoRoot))\n        throw new Error(`Repository not found at ${repoRoot}`);\n      const analysis = analyzeRepo(repoRoot);\n      const capabilities = deriveCapabilities(analysis);\n      return {\n        content: [\n          {\n            type: \"text\",\n            text: JSON.stringify(\n              {\n                capabilities,\n                analysisSummary: {\n                  files: analysis.files.length,\n                  testFiles: analysis.testFiles.length,\n                  readme: analysis.readme?.title,\n                },\n              },\n              null,\n              2\n            ),\n          },\n        ],\n      };\n    },\n  };\n}\n\nfunction buildPlanFeatureTool(): Tool<undefined, typeof planFeatureSchema> {\n  return {\n    name: \"plan-feature-implementation\",\n    description:\n      \"Given a feature request, select appropriate MCP tools (including existing and new ones) and produce an execution plan.\",\n    parameters: planFeatureSchema,\n    execute: async (input) => {\n      const steps: Array<{\n        step: number;\n        action: string;\n        tool?: string;\n        arguments?: Record<string, any>;\n        rationale: string;\n      }> = [];\n      let i = 1;\n      steps.push({\n        step: i++,\n        action: \"Analyze repository to enumerate APIs and decorators\",\n        tool: \"analyze-repository\",\n        arguments: { repoPath: input.repoPath },\n        rationale: \"Understand available building blocks.\",\n      });\n      steps.push({\n        step: i++,\n        action: \"List capabilities expected for developers\",\n        tool: \"enumerate-capabilities\",\n        arguments: { repoPath: input.repoPath },\n        rationale: \"Align the plan with supported capabilities.\",\n      });\n      // Suggest existing generic tools from mcp-module\n      steps.push({\n        step: i++,\n        action:\n          \"Select documentation prompt and gather relevant source file(s)\",\n        tool: \"document-code\",\n        arguments: { filePath: \"<target-file>\" },\n        rationale: \"Provide context and instructions for changes.\",\n      });\n      steps.push({\n        step: i++,\n        action: \"Apply code changes using unified diff patch\",\n        tool: \"apply-code-change\",\n        arguments: {\n          filePath: \"<target-file>\",\n          patch: \"<unified-diff>\",\n          dryRun: true,\n        },\n        rationale: \"Validate changes safely before committing.\",\n      });\n      steps.push({\n        step: i++,\n        action: \"Commit code changes\",\n        tool: \"apply-code-change\",\n        arguments: {\n          filePath: \"<target-file>\",\n          patch: \"<unified-diff>\",\n          dryRun: false,\n        },\n        rationale: \"Persist the update.\",\n      });\n      // If decoration-related terms present, suggest decorator tools\n      if (/decorat|flavour|override|extend|builder/i.test(input.feature)) {\n        steps.unshift({\n          step: 0,\n          action: \"Use decorator tooling to insert/remove/modify decorators\",\n          tool: \"decorator-tools\",\n          arguments: { action: \"help\" },\n          rationale: \"Leverage specialized utilities for decoration patterns.\",\n        });\n        steps.forEach((s, idx) => (s.step = idx + 1));\n      }\n      return {\n        content: [\n          {\n            type: \"text\",\n            text: JSON.stringify(\n              {\n                plan: steps,\n                notes:\n                  \"Replace placeholder arguments like <target-file> and <unified-diff> based on the analysis output.\",\n              },\n              null,\n              2\n            ),\n          },\n        ],\n      };\n    },\n  };\n}\n\nfunction buildPrompts(repoPath: string): InputPrompt<undefined>[] {\n  return [\n    {\n      name: \"decoration-overview\",\n      description:\n        \"High-level guidance on using the decoration library: key exports, decorators, and common workflows.\",\n      load: async () =>\n        `You are assisting with the Decaf.ts decoration module located at ${repoPath}. Prefer using exported builders and decorators over ad-hoc patterns.\\n\\nProvide a concise, actionable overview of how to use the decoration APIs for extending and overriding behaviors.`,\n    },\n  ];\n}\n\ntype DecorationResourceTemplate = {\n  name: string;\n  description: string;\n  uriTemplate: string;\n  mimeType: string;\n  arguments: ReadonlyArray<{\n    name: string;\n    description: string;\n    required: boolean;\n  }>;\n  load: (args: { path: string }) => Promise<{ text: string }>;\n};\nfunction buildResourceTemplates(\n  repoPath: string\n): DecorationResourceTemplate[] {\n  const root = path.resolve(process.cwd(), repoPath);\n  return [\n    {\n      name: \"decoration-src\",\n      description: \"Read a file from the decoration/src tree by relative path.\",\n      mimeType: \"text/plain\",\n      uriTemplate: \"decoration://src/{path}\",\n      arguments: [\n        {\n          name: \"path\",\n          description:\n            \"Path under decoration/src to load, e.g. 'decoration/types.ts'\",\n          required: true,\n        },\n      ],\n      load: async ({ path: rel }: { path: string }) => {\n        const abs = path.join(root, \"src\", rel);\n        const text = readFileSafe(abs) ?? \"\";\n        return { text };\n      },\n    },\n  ];\n}\n\nexport default function enrich(mcp: FastMCP) {\n  // Register tools\n  mcp.addTool(buildAnalyzeRepositoryTool() as any);\n  mcp.addTool(buildEnumerateCapabilitiesTool() as any);\n  mcp.addTool(buildPlanFeatureTool() as any);\n  // Prompts/resources\n  const repoPath = \"./decoration\";\n  for (const p of buildPrompts(repoPath)) mcp.addPrompt(p as any);\n  for (const r of buildResourceTemplates(repoPath))\n    mcp.addResourceTemplate(r as any);\n  return mcp;\n}\n\nexport const VERSION = V;\nexport const PACKAGE_NAME = `${PKG}/decoration-assist`;\n"]}
@@ -0,0 +1,118 @@
1
+ export type DecoratorSpec = {
2
+ name: string;
3
+ args?: unknown[];
4
+ };
5
+ export type AttributeSpec = {
6
+ name: string;
7
+ type: string;
8
+ decorators?: DecoratorSpec[];
9
+ };
10
+ export declare const decoratorTools: {
11
+ readonly createOrUpdateModelTool: {
12
+ readonly name: "create-or-update-model";
13
+ readonly description: "Create or update a validation-ready model class";
14
+ readonly execute: (args: {
15
+ filePath: string;
16
+ className: string;
17
+ classDecorators?: DecoratorSpec[];
18
+ properties: AttributeSpec[];
19
+ importsFrom: string;
20
+ overwrite?: boolean;
21
+ }) => Promise<{
22
+ filePath: string;
23
+ }>;
24
+ };
25
+ readonly addAttributeTool: {
26
+ readonly name: "add-attribute";
27
+ readonly description: "Add a decorated attribute to an existing model";
28
+ readonly execute: (args: {
29
+ filePath: string;
30
+ className: string;
31
+ attribute: AttributeSpec;
32
+ importsFrom: string;
33
+ }) => Promise<{
34
+ filePath: string;
35
+ }>;
36
+ };
37
+ readonly removeAttributeTool: {
38
+ readonly name: "remove-attribute";
39
+ readonly description: "Remove an attribute from a model class";
40
+ readonly execute: (args: {
41
+ filePath: string;
42
+ className: string;
43
+ attributeName: string;
44
+ }) => Promise<{
45
+ filePath: string;
46
+ }>;
47
+ };
48
+ readonly applyDecoratorTool: {
49
+ readonly name: "apply-decorator";
50
+ readonly description: "Apply a decorator to a class or property";
51
+ readonly execute: (args: {
52
+ filePath: string;
53
+ className: string;
54
+ target: {
55
+ kind: "class" | "property";
56
+ name?: string;
57
+ };
58
+ decorator: DecoratorSpec;
59
+ importsFrom: string;
60
+ }) => Promise<{
61
+ filePath: string;
62
+ }>;
63
+ };
64
+ readonly removeDecoratorTool: {
65
+ readonly name: "remove-decorator";
66
+ readonly description: "Remove a decorator from a class or property";
67
+ readonly execute: (args: {
68
+ filePath: string;
69
+ className: string;
70
+ target: {
71
+ kind: "class" | "property";
72
+ name?: string;
73
+ };
74
+ decoratorName: string;
75
+ }) => Promise<{
76
+ filePath: string;
77
+ }>;
78
+ };
79
+ readonly scaffoldValidatorTool: {
80
+ readonly name: "scaffold-validator";
81
+ readonly description: "Scaffold a validator class and optional decorator";
82
+ readonly execute: (args: {
83
+ validatorsDir: string;
84
+ decoratorDir?: string;
85
+ name: string;
86
+ }) => Promise<{
87
+ classFile: string;
88
+ decoratorFile: string | undefined;
89
+ }>;
90
+ };
91
+ readonly scaffoldSerializerTool: {
92
+ readonly name: "scaffold-serializer";
93
+ readonly description: "Scaffold a serializer class and optional registry";
94
+ readonly execute: (args: {
95
+ dir: string;
96
+ name: string;
97
+ registerDir?: string;
98
+ setDefault?: boolean;
99
+ }) => Promise<{
100
+ classFile: string;
101
+ registerFile: string | undefined;
102
+ }>;
103
+ };
104
+ readonly scaffoldHashingTool: {
105
+ readonly name: "scaffold-hashing";
106
+ readonly description: "Scaffold a hashing function and optional registry";
107
+ readonly execute: (args: {
108
+ dir: string;
109
+ name: string;
110
+ registerDir?: string;
111
+ setDefault?: boolean;
112
+ }) => Promise<{
113
+ functionFile: string;
114
+ registerFile: string | undefined;
115
+ }>;
116
+ };
117
+ };
118
+ export type DecoratorTools = typeof decoratorTools;