@decaf-ts/mcp-server 0.0.3 → 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.
@@ -6,7 +6,7 @@ import { Metadata } from "@decaf-ts/decoration";
6
6
  * @summary The actual version number is replaced during the build process.
7
7
  * @type {string}
8
8
  */
9
- export const VERSION = "0.0.2";
9
+ export const VERSION = "0.0.3";
10
10
  export const PACKAGE_NAME = "##PACKAGE_NAME##";
11
11
  try {
12
12
  Metadata.registerLibrary(PACKAGE_NAME, VERSION);
@@ -34,6 +34,6 @@ export type AnalyzeRepoArgs = z.infer<typeof analyzeRepoSchema>;
34
34
  export type EnumerateCapabilitiesArgs = z.infer<typeof enumerateCapabilitiesSchema>;
35
35
  export type PlanFeatureArgs = z.infer<typeof planFeatureSchema>;
36
36
  export default function enrich(mcp: FastMCP): FastMCP<Record<string, unknown> | undefined>;
37
- export declare const VERSION = "0.0.2";
37
+ export declare const VERSION = "0.0.3";
38
38
  export declare const PACKAGE_NAME = "##PACKAGE_NAME##/decoration-assist";
39
39
  export {};
@@ -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"]}
@@ -221,7 +221,7 @@ export declare const tools: {
221
221
  export declare function enrich(mcp: FastMCP): FastMCP;
222
222
  export default enrich;
223
223
  export declare const PACKAGE_NAME = "##PACKAGE_NAME##";
224
- export declare const VERSION = "0.0.2";
224
+ export declare const VERSION = "0.0.3";
225
225
  export declare function setWorkspaceRoot(root: string): void;
226
226
  export declare function getWorkspaceRoot(): string;
227
227
  export declare function buildResourceTemplates(): WorkspaceResourceTemplate[];
package/lib/metadata.cjs CHANGED
@@ -9,7 +9,7 @@ const decoration_1 = require("@decaf-ts/decoration");
9
9
  * @summary The actual version number is replaced during the build process.
10
10
  * @type {string}
11
11
  */
12
- exports.VERSION = "0.0.2";
12
+ exports.VERSION = "0.0.3";
13
13
  exports.PACKAGE_NAME = "##PACKAGE_NAME##";
14
14
  try {
15
15
  decoration_1.Metadata.registerLibrary(exports.PACKAGE_NAME, exports.VERSION);
package/lib/metadata.d.ts CHANGED
@@ -5,5 +5,5 @@
5
5
  * @summary The actual version number is replaced during the build process.
6
6
  * @type {string}
7
7
  */
8
- export declare const VERSION = "0.0.2";
8
+ export declare const VERSION = "0.0.3";
9
9
  export declare const PACKAGE_NAME = "##PACKAGE_NAME##";