@decaf-ts/mcp-server 0.0.4 → 0.2.0

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 (210) hide show
  1. package/README.md +18 -2
  2. package/dist/mcp-server.cjs +1986 -340
  3. package/dist/mcp-server.esm.cjs +1960 -337
  4. package/lib/McpWrapper.cjs +9 -9
  5. package/lib/McpWrapper.d.ts +1 -1
  6. package/lib/bin/validate-modules.cjs +24 -0
  7. package/lib/bin/validate-modules.d.ts +2 -0
  8. package/lib/constants.cjs +22 -2
  9. package/lib/constants.d.ts +16 -0
  10. package/lib/esm/McpWrapper.d.ts +1 -1
  11. package/lib/esm/McpWrapper.js +9 -9
  12. package/lib/esm/bin/validate-modules.d.ts +2 -0
  13. package/lib/esm/bin/validate-modules.js +22 -0
  14. package/lib/esm/constants.d.ts +16 -0
  15. package/lib/esm/constants.js +21 -1
  16. package/lib/esm/mcp/aggregateModules.d.ts +26 -0
  17. package/lib/esm/mcp/aggregateModules.js +185 -0
  18. package/lib/esm/mcp/code.d.ts +23 -0
  19. package/lib/esm/mcp/code.js +70 -0
  20. package/lib/esm/mcp/decorator-tools.js +237 -0
  21. package/lib/esm/mcp/fastmcp-wiring.d.ts +14 -0
  22. package/lib/esm/mcp/fastmcp-wiring.js +56 -0
  23. package/lib/esm/mcp/index.d.ts +7 -1
  24. package/lib/esm/mcp/index.js +26 -2
  25. package/lib/esm/mcp/mcp-module.d.ts +11 -0
  26. package/lib/esm/mcp/mcp-module.js +31 -0
  27. package/lib/esm/mcp/moduleRegistry.d.ts +12 -0
  28. package/lib/esm/mcp/moduleRegistry.js +46 -0
  29. package/lib/esm/mcp/prompts/index.d.ts +4 -0
  30. package/lib/esm/mcp/prompts/index.js +7 -0
  31. package/lib/esm/mcp/prompts/prompts.d.ts +22 -0
  32. package/lib/esm/mcp/prompts/prompts.js +197 -0
  33. package/lib/esm/mcp/resources/index.d.ts +1 -0
  34. package/lib/esm/mcp/resources/index.js +2 -0
  35. package/lib/esm/mcp/resources/resources.d.ts +2 -0
  36. package/lib/esm/mcp/resources/resources.js +69 -0
  37. package/lib/esm/mcp/schemas.d.ts +53 -0
  38. package/lib/esm/mcp/schemas.js +97 -0
  39. package/lib/esm/mcp/templates/codex-templates.d.ts +3 -0
  40. package/lib/esm/mcp/templates/codex-templates.js +33 -0
  41. package/lib/esm/mcp/templates/index.d.ts +71 -0
  42. package/lib/esm/mcp/templates/index.js +66 -0
  43. package/lib/esm/mcp/templates/resource-templates.d.ts +3 -0
  44. package/lib/esm/mcp/templates/resource-templates.js +60 -0
  45. package/lib/esm/mcp/templates/workspace-templates.d.ts +3 -0
  46. package/lib/esm/mcp/templates/workspace-templates.js +66 -0
  47. package/lib/esm/mcp/tools/codex-tools.d.ts +5 -0
  48. package/lib/esm/mcp/tools/codex-tools.js +244 -0
  49. package/lib/esm/mcp/tools/generateMcpModule.d.ts +9 -0
  50. package/lib/esm/mcp/tools/generateMcpModule.js +133 -0
  51. package/lib/esm/mcp/tools/index.d.ts +321 -0
  52. package/lib/esm/mcp/tools/index.js +29 -0
  53. package/lib/esm/mcp/tools/tools.d.ts +10 -0
  54. package/lib/esm/mcp/tools/tools.js +273 -0
  55. package/lib/esm/mcp/types.d.ts +66 -0
  56. package/lib/esm/mcp/types.js +2 -0
  57. package/lib/esm/mcp/utils.d.ts +4 -0
  58. package/lib/esm/mcp/utils.js +46 -0
  59. package/lib/esm/mcp/validation/index.d.ts +13 -0
  60. package/lib/esm/mcp/validation/index.js +116 -0
  61. package/lib/esm/mcp/validation/scaffoldModule.d.ts +9 -0
  62. package/lib/esm/mcp/validation/scaffoldModule.js +88 -0
  63. package/lib/esm/mcp/workspace.d.ts +9 -0
  64. package/lib/esm/mcp/workspace.js +73 -0
  65. package/lib/esm/metadata.d.ts +1 -1
  66. package/lib/esm/metadata.js +1 -1
  67. package/lib/esm/modules/_template/index.d.ts +32 -0
  68. package/lib/esm/modules/_template/index.js +16 -0
  69. package/lib/esm/modules/_template/prompts/index.d.ts +6 -0
  70. package/lib/esm/modules/_template/prompts/index.js +9 -0
  71. package/lib/esm/modules/_template/resources/index.d.ts +6 -0
  72. package/lib/esm/modules/_template/resources/index.js +9 -0
  73. package/lib/esm/modules/_template/templates/index.d.ts +7 -0
  74. package/lib/esm/modules/_template/templates/index.js +10 -0
  75. package/lib/esm/modules/_template/tools/index.d.ts +6 -0
  76. package/lib/esm/modules/_template/tools/index.js +15 -0
  77. package/lib/esm/modules/decoration/index.d.ts +46 -0
  78. package/lib/esm/modules/decoration/index.js +10 -2
  79. package/lib/esm/modules/decoration/prompts/index.d.ts +1 -0
  80. package/lib/esm/modules/decoration/prompts/index.js +2 -0
  81. package/lib/esm/modules/decoration/resources/index.d.ts +7 -0
  82. package/lib/esm/modules/decoration/resources/index.js +10 -0
  83. package/lib/esm/modules/decoration/templates/index.d.ts +6 -0
  84. package/lib/esm/modules/decoration/templates/index.js +9 -0
  85. package/lib/esm/modules/decoration/tools/index.d.ts +26 -0
  86. package/lib/esm/modules/decoration/tools/index.js +7 -0
  87. package/lib/esm/modules/index.d.ts +2 -0
  88. package/lib/esm/modules/index.js +10 -0
  89. package/lib/esm/modules/mcp/decoration-assist.d.ts +3 -38
  90. package/lib/esm/modules/mcp/decoration-assist.js +5 -352
  91. package/lib/esm/modules/mcp/index.d.ts +6 -2
  92. package/lib/esm/modules/mcp/index.js +16 -3
  93. package/lib/esm/modules/mcp/prompts/index.d.ts +2 -0
  94. package/lib/esm/modules/mcp/prompts/index.js +9 -0
  95. package/lib/esm/modules/mcp/resources/index.d.ts +2 -0
  96. package/lib/esm/modules/mcp/resources/index.js +24 -0
  97. package/lib/esm/modules/mcp/templates/index.d.ts +2 -0
  98. package/lib/esm/modules/mcp/templates/index.js +28 -0
  99. package/lib/esm/modules/mcp/tools/index.d.ts +6 -0
  100. package/lib/esm/modules/mcp/tools/index.js +15 -0
  101. package/lib/esm/types.d.ts +41 -1
  102. package/lib/esm/types.js +1 -1
  103. package/lib/esm/utils/modulePaths.d.ts +6 -0
  104. package/lib/esm/utils/modulePaths.js +33 -0
  105. package/lib/esm/utils/moduleValidator.d.ts +14 -0
  106. package/lib/esm/utils/moduleValidator.js +176 -0
  107. package/lib/esm/utils.d.ts +1 -0
  108. package/lib/esm/utils.js +2 -1
  109. package/lib/mcp/aggregateModules.cjs +225 -0
  110. package/lib/mcp/aggregateModules.d.ts +26 -0
  111. package/lib/mcp/code.cjs +81 -0
  112. package/lib/mcp/code.d.ts +23 -0
  113. package/lib/mcp/decorator-tools.cjs +243 -0
  114. package/lib/mcp/fastmcp-wiring.cjs +59 -0
  115. package/lib/mcp/fastmcp-wiring.d.ts +14 -0
  116. package/lib/mcp/index.cjs +47 -12
  117. package/lib/mcp/index.d.ts +7 -1
  118. package/lib/mcp/mcp-module.cjs +53 -0
  119. package/lib/mcp/mcp-module.d.ts +11 -0
  120. package/lib/mcp/moduleRegistry.cjs +50 -0
  121. package/lib/mcp/moduleRegistry.d.ts +12 -0
  122. package/lib/mcp/prompts/index.cjs +25 -0
  123. package/lib/mcp/prompts/index.d.ts +4 -0
  124. package/lib/mcp/prompts/prompts.cjs +211 -0
  125. package/lib/mcp/prompts/prompts.d.ts +22 -0
  126. package/lib/mcp/resources/index.cjs +18 -0
  127. package/lib/mcp/resources/index.d.ts +1 -0
  128. package/lib/mcp/resources/resources.cjs +72 -0
  129. package/lib/mcp/resources/resources.d.ts +2 -0
  130. package/lib/mcp/schemas.cjs +100 -0
  131. package/lib/mcp/schemas.d.ts +53 -0
  132. package/lib/mcp/templates/codex-templates.cjs +40 -0
  133. package/lib/mcp/templates/codex-templates.d.ts +3 -0
  134. package/lib/mcp/templates/index.cjs +76 -0
  135. package/lib/mcp/templates/index.d.ts +71 -0
  136. package/lib/mcp/templates/resource-templates.cjs +67 -0
  137. package/lib/mcp/templates/resource-templates.d.ts +3 -0
  138. package/lib/mcp/templates/workspace-templates.cjs +70 -0
  139. package/lib/mcp/templates/workspace-templates.d.ts +3 -0
  140. package/lib/mcp/tools/codex-tools.cjs +250 -0
  141. package/lib/mcp/tools/codex-tools.d.ts +5 -0
  142. package/lib/mcp/tools/generateMcpModule.cjs +139 -0
  143. package/lib/mcp/tools/generateMcpModule.d.ts +9 -0
  144. package/lib/mcp/tools/index.cjs +46 -0
  145. package/lib/mcp/tools/index.d.ts +321 -0
  146. package/lib/mcp/tools/tools.cjs +282 -0
  147. package/lib/mcp/tools/tools.d.ts +10 -0
  148. package/lib/mcp/types.cjs +3 -0
  149. package/lib/mcp/types.d.ts +66 -0
  150. package/lib/mcp/utils.cjs +54 -0
  151. package/lib/mcp/utils.d.ts +4 -0
  152. package/lib/mcp/validation/index.cjs +123 -0
  153. package/lib/mcp/validation/index.d.ts +13 -0
  154. package/lib/mcp/validation/scaffoldModule.cjs +94 -0
  155. package/lib/mcp/validation/scaffoldModule.d.ts +9 -0
  156. package/lib/mcp/workspace.cjs +119 -0
  157. package/lib/mcp/workspace.d.ts +9 -0
  158. package/lib/metadata.cjs +1 -1
  159. package/lib/metadata.d.ts +1 -1
  160. package/lib/modules/_template/index.cjs +23 -0
  161. package/lib/modules/_template/index.d.ts +32 -0
  162. package/lib/modules/_template/prompts/index.cjs +12 -0
  163. package/lib/modules/_template/prompts/index.d.ts +6 -0
  164. package/lib/modules/_template/resources/index.cjs +12 -0
  165. package/lib/modules/_template/resources/index.d.ts +6 -0
  166. package/lib/modules/_template/templates/index.cjs +13 -0
  167. package/lib/modules/_template/templates/index.d.ts +7 -0
  168. package/lib/modules/_template/tools/index.cjs +18 -0
  169. package/lib/modules/_template/tools/index.d.ts +6 -0
  170. package/lib/modules/decoration/index.cjs +16 -1
  171. package/lib/modules/decoration/index.d.ts +46 -0
  172. package/lib/modules/decoration/prompts/index.cjs +5 -0
  173. package/lib/modules/decoration/prompts/index.d.ts +1 -0
  174. package/lib/modules/decoration/resources/index.cjs +13 -0
  175. package/lib/modules/decoration/resources/index.d.ts +7 -0
  176. package/lib/modules/decoration/templates/index.cjs +12 -0
  177. package/lib/modules/decoration/templates/index.d.ts +6 -0
  178. package/lib/modules/decoration/tools/index.cjs +10 -0
  179. package/lib/modules/decoration/tools/index.d.ts +26 -0
  180. package/lib/modules/index.cjs +13 -0
  181. package/lib/modules/index.d.ts +2 -0
  182. package/lib/modules/mcp/decoration-assist.cjs +8 -355
  183. package/lib/modules/mcp/decoration-assist.d.ts +3 -38
  184. package/lib/modules/mcp/index.cjs +21 -22
  185. package/lib/modules/mcp/index.d.ts +6 -2
  186. package/lib/modules/mcp/prompts/index.cjs +12 -0
  187. package/lib/modules/mcp/prompts/index.d.ts +2 -0
  188. package/lib/modules/mcp/resources/index.cjs +27 -0
  189. package/lib/modules/mcp/resources/index.d.ts +2 -0
  190. package/lib/modules/mcp/templates/index.cjs +31 -0
  191. package/lib/modules/mcp/templates/index.d.ts +2 -0
  192. package/lib/modules/mcp/tools/index.cjs +18 -0
  193. package/lib/modules/mcp/tools/index.d.ts +6 -0
  194. package/lib/types.cjs +1 -1
  195. package/lib/types.d.ts +41 -1
  196. package/lib/utils/modulePaths.cjs +43 -0
  197. package/lib/utils/modulePaths.d.ts +6 -0
  198. package/lib/utils/moduleValidator.cjs +184 -0
  199. package/lib/utils/moduleValidator.d.ts +14 -0
  200. package/lib/utils.cjs +5 -1
  201. package/lib/utils.d.ts +1 -0
  202. package/package.json +17 -11
  203. package/lib/esm/modules/mcp/decorator-tools.js +0 -237
  204. package/lib/esm/modules/mcp/mcp-module.d.ts +0 -230
  205. package/lib/esm/modules/mcp/mcp-module.js +0 -406
  206. package/lib/modules/mcp/decorator-tools.cjs +0 -243
  207. package/lib/modules/mcp/mcp-module.cjs +0 -452
  208. package/lib/modules/mcp/mcp-module.d.ts +0 -230
  209. /package/lib/esm/{modules/mcp → mcp}/decorator-tools.d.ts +0 -0
  210. /package/lib/{modules/mcp → mcp}/decorator-tools.d.ts +0 -0
@@ -0,0 +1,250 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.readmeImprovementTool = exports.coverageEnforcerTool = exports.documentObjectTool = void 0;
7
+ const fs_1 = __importDefault(require("fs"));
8
+ const path_1 = __importDefault(require("path"));
9
+ const child_process_1 = require("child_process");
10
+ const schemas_1 = require("./../schemas.cjs");
11
+ const workspace_1 = require("./../workspace.cjs");
12
+ const prompts_1 = require("./../prompts/prompts.cjs");
13
+ const utils_1 = require("./../utils.cjs");
14
+ const code_1 = require("./../code.cjs");
15
+ function relativeFiles(root, files) {
16
+ return files.map((file) => path_1.default.relative(root, file)).sort();
17
+ }
18
+ function collectPromptSections(names) {
19
+ const root = (0, workspace_1.getWorkspaceRoot)();
20
+ const promptIndex = new Map((0, prompts_1.discoverDocPrompts)(root).map((prompt) => [prompt.name, prompt]));
21
+ return names
22
+ .map((name) => promptIndex.get(name))
23
+ .filter((prompt) => Boolean(prompt))
24
+ .map((prompt) => ({
25
+ name: prompt.name,
26
+ title: prompt.title,
27
+ description: prompt.description,
28
+ content: prompt.content,
29
+ absolutePath: prompt.absolutePath,
30
+ }));
31
+ }
32
+ function parseTaskLines(content) {
33
+ return content
34
+ .split(/\r?\n/)
35
+ .map((line) => line.trim())
36
+ .filter((line) => /^task\s+\d+/i.test(line));
37
+ }
38
+ function computeCoverageFromFinal(coveragePath) {
39
+ const payload = JSON.parse(fs_1.default.readFileSync(coveragePath, "utf8"));
40
+ const totals = {
41
+ statements: { covered: 0, total: 0 },
42
+ functions: { covered: 0, total: 0 },
43
+ branches: { covered: 0, total: 0 },
44
+ };
45
+ const files = Object.entries(payload).map(([filePath, info]) => {
46
+ const statementCounts = Object.values(info.s);
47
+ const functionCounts = Object.values(info.f);
48
+ const branchCounts = Object.values(info.b).flatMap((value) => Array.isArray(value) ? value : [value]);
49
+ const statementTotal = statementCounts.length;
50
+ const functionTotal = functionCounts.length;
51
+ const branchTotal = branchCounts.length;
52
+ const statementCovered = statementCounts.filter((count) => count > 0).length;
53
+ const functionCovered = functionCounts.filter((count) => count > 0).length;
54
+ const branchCovered = branchCounts.filter((count) => count > 0).length;
55
+ totals.statements.covered += statementCovered;
56
+ totals.statements.total += statementTotal;
57
+ totals.functions.covered += functionCovered;
58
+ totals.functions.total += functionTotal;
59
+ totals.branches.covered += branchCovered;
60
+ totals.branches.total += branchTotal;
61
+ const pct = (covered, total) => total === 0 ? 100 : Number(((covered / total) * 100).toFixed(2));
62
+ return {
63
+ path: filePath,
64
+ statements: pct(statementCovered, statementTotal),
65
+ functions: pct(functionCovered, functionTotal),
66
+ branches: pct(branchCovered, branchTotal),
67
+ };
68
+ });
69
+ const pct = (covered, total) => total === 0 ? 100 : Number(((covered / total) * 100).toFixed(2));
70
+ return {
71
+ totals: {
72
+ statements: {
73
+ ...totals.statements,
74
+ pct: pct(totals.statements.covered, totals.statements.total),
75
+ },
76
+ functions: {
77
+ ...totals.functions,
78
+ pct: pct(totals.functions.covered, totals.functions.total),
79
+ },
80
+ branches: {
81
+ ...totals.branches,
82
+ pct: pct(totals.branches.covered, totals.branches.total),
83
+ },
84
+ },
85
+ files,
86
+ };
87
+ }
88
+ function normalizePromptSections(sections) {
89
+ return sections.map((section) => ({
90
+ name: section.name,
91
+ title: section.title,
92
+ tasks: parseTaskLines(section.content),
93
+ content: section.content,
94
+ }));
95
+ }
96
+ async function resolveRepoRoot(basePath) {
97
+ const root = (0, workspace_1.getWorkspaceRoot)();
98
+ try {
99
+ return (0, workspace_1.resolveInWorkspace)(root, basePath);
100
+ }
101
+ catch (error) {
102
+ if (error instanceof workspace_1.WorkspaceError) {
103
+ await (0, workspace_1.throwUserError)(error.message);
104
+ }
105
+ throw error;
106
+ }
107
+ }
108
+ exports.documentObjectTool = {
109
+ name: "document-object",
110
+ description: "Create a documentation plan for a specific object type using .codex prompts and repository analysis.",
111
+ parameters: schemas_1.documentObjectSchema,
112
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
113
+ execute: async (input, _context) => {
114
+ const args = schemas_1.documentObjectSchema.parse(input);
115
+ const repoRoot = await resolveRepoRoot(args.basePath);
116
+ const dependencies = (0, prompts_1.getObjectPromptDependencies)()[args.objectType] ?? [];
117
+ if (!dependencies.length) {
118
+ await (0, workspace_1.throwUserError)(`No prompt guidance configured for object type ${args.objectType}`);
119
+ }
120
+ const sections = normalizePromptSections(collectPromptSections(dependencies));
121
+ const srcDir = path_1.default.join(repoRoot, "src");
122
+ const testDir = path_1.default.join(repoRoot, "tests");
123
+ const sourceFiles = fs_1.default.existsSync(srcDir)
124
+ ? (0, utils_1.listFilesRecursive)(srcDir, code_1.isSourceFile)
125
+ : [];
126
+ const testFiles = fs_1.default.existsSync(testDir)
127
+ ? (0, utils_1.listFilesRecursive)(testDir, (file) => (0, code_1.isSourceFile)(file) && (0, code_1.isTestFile)(file))
128
+ : [];
129
+ let targetFileContent;
130
+ if (args.targetFile) {
131
+ try {
132
+ const absolute = (0, workspace_1.resolveInWorkspace)(repoRoot, args.targetFile);
133
+ targetFileContent = (0, utils_1.readFileSafe)(absolute) ?? undefined;
134
+ }
135
+ catch (error) {
136
+ if (error instanceof workspace_1.WorkspaceError) {
137
+ await (0, workspace_1.throwUserError)(error.message);
138
+ }
139
+ throw error;
140
+ }
141
+ }
142
+ const payload = {
143
+ basePath: path_1.default.relative((0, workspace_1.getWorkspaceRoot)(), repoRoot) || ".",
144
+ objectType: args.objectType,
145
+ targetFile: args.targetFile,
146
+ guidance: sections,
147
+ files: {
148
+ source: relativeFiles(repoRoot, sourceFiles),
149
+ tests: relativeFiles(repoRoot, testFiles),
150
+ },
151
+ targetFileContent: args.includeContent ? targetFileContent : undefined,
152
+ };
153
+ return {
154
+ content: [
155
+ {
156
+ type: "text",
157
+ text: JSON.stringify(payload, null, 2),
158
+ },
159
+ ],
160
+ };
161
+ },
162
+ };
163
+ exports.coverageEnforcerTool = {
164
+ name: "ensure-test-coverage",
165
+ description: "Run the configured coverage command and report whether the target percentage is met, highlighting weak files.",
166
+ parameters: schemas_1.coverageTaskSchema,
167
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
168
+ execute: async (input, _context) => {
169
+ const args = schemas_1.coverageTaskSchema.parse(input);
170
+ const repoRoot = await resolveRepoRoot(args.basePath);
171
+ if (!args.dryRun) {
172
+ const env = {
173
+ ...process.env,
174
+ USE_WATCHMAN: "false",
175
+ WATCHMAN_DISABLE: "1",
176
+ JEST_DISABLE_WATCHMAN: "1",
177
+ };
178
+ const result = (0, child_process_1.spawnSync)("npm", ["run", "coverage", "--", "--watchman=false", "--runInBand"], { cwd: repoRoot, env, encoding: "utf8" });
179
+ if (result.status !== 0) {
180
+ const message = result.stderr || result.stdout || "Coverage command failed";
181
+ await (0, workspace_1.throwUserError)(message.trim());
182
+ }
183
+ }
184
+ const coveragePath = path_1.default.join(repoRoot, "workdocs", "reports", "coverage", "coverage-final.json");
185
+ if (!fs_1.default.existsSync(coveragePath)) {
186
+ await (0, workspace_1.throwUserError)(`Coverage report not found at ${path_1.default.relative(repoRoot, coveragePath)}`);
187
+ }
188
+ const summary = computeCoverageFromFinal(coveragePath);
189
+ const meetsThreshold = summary.totals.statements.pct >= args.coverage &&
190
+ summary.totals.functions.pct >= args.coverage &&
191
+ summary.totals.branches.pct >= args.coverage;
192
+ const weakest = [...summary.files]
193
+ .sort((a, b) => a.statements - b.statements)
194
+ .slice(0, 10);
195
+ const guidance = normalizePromptSections(collectPromptSections(["bulk-tests"]));
196
+ const payload = {
197
+ basePath: path_1.default.relative((0, workspace_1.getWorkspaceRoot)(), repoRoot) || ".",
198
+ target: args.coverage,
199
+ meetsThreshold,
200
+ totals: summary.totals,
201
+ weakest,
202
+ guidance,
203
+ };
204
+ return {
205
+ content: [
206
+ {
207
+ type: "text",
208
+ text: JSON.stringify(payload, null, 2),
209
+ },
210
+ ],
211
+ };
212
+ },
213
+ };
214
+ exports.readmeImprovementTool = {
215
+ name: "improve-readme",
216
+ description: "Summarize required steps to refresh README and workdocs content using .codex guidance and repository analysis.",
217
+ parameters: schemas_1.readmeImprovementSchema,
218
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
219
+ execute: async (input, _context) => {
220
+ const args = schemas_1.readmeImprovementSchema.parse(input);
221
+ const repoRoot = await resolveRepoRoot(args.basePath);
222
+ const analysis = (0, code_1.analyzeRepo)(repoRoot);
223
+ const modules = analysis.files
224
+ .filter((file) => /index\.ts$/.test(file))
225
+ .map((file) => path_1.default.relative(repoRoot, file));
226
+ const promptSections = normalizePromptSections(collectPromptSections(["update-readme", "doc", "module"]));
227
+ const testExamples = Object.keys(analysis.tests ?? {});
228
+ const examples = args.includeExamples ? testExamples.slice(0, 20) : [];
229
+ const payload = {
230
+ basePath: path_1.default.relative((0, workspace_1.getWorkspaceRoot)(), repoRoot) || ".",
231
+ summary: {
232
+ modules,
233
+ totalSourceFiles: analysis.files.length,
234
+ totalTestFiles: analysis.testFiles.length,
235
+ hasReadme: Boolean(analysis.readme),
236
+ },
237
+ guidance: promptSections,
238
+ suggestedExamples: examples,
239
+ };
240
+ return {
241
+ content: [
242
+ {
243
+ type: "text",
244
+ text: JSON.stringify(payload, null, 2),
245
+ },
246
+ ],
247
+ };
248
+ },
249
+ };
250
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"codex-tools.js","sourceRoot":"","sources":["../../../src/mcp/tools/codex-tools.ts"],"names":[],"mappings":";;;;;;AAAA,4CAAoB;AACpB,gDAAwB;AACxB,iDAA0C;AAE1C,8CAIoB;AAMpB,kDAKsB;AACtB,sDAI4B;AAE5B,0CAA4D;AAC5D,wCAAgE;AAUhE,SAAS,aAAa,CAAC,IAAY,EAAE,KAAe;IAClD,OAAO,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,cAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;AAC/D,CAAC;AAED,SAAS,qBAAqB,CAAC,KAAwB;IACrD,MAAM,IAAI,GAAG,IAAA,4BAAgB,GAAE,CAAC;IAChC,MAAM,WAAW,GAAG,IAAI,GAAG,CACzB,IAAA,4BAAkB,EAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,CAChE,CAAC;IACF,OAAO,KAAK;SACT,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;SACpC,MAAM,CAAC,CAAC,MAAM,EAAuB,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;SACxD,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;QAChB,IAAI,EAAE,MAAM,CAAC,IAAI;QACjB,KAAK,EAAE,MAAM,CAAC,KAAK;QACnB,WAAW,EAAE,MAAM,CAAC,WAAW;QAC/B,OAAO,EAAE,MAAM,CAAC,OAAO;QACvB,YAAY,EAAE,MAAM,CAAC,YAAY;KAClC,CAAC,CAAC,CAAC;AACR,CAAC;AAED,SAAS,cAAc,CAAC,OAAe;IACrC,OAAO,OAAO;SACX,KAAK,CAAC,OAAO,CAAC;SACd,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;SAC1B,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;AACjD,CAAC;AAED,SAAS,wBAAwB,CAAC,YAAoB;IACpD,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,YAAE,CAAC,YAAY,CAAC,YAAY,EAAE,MAAM,CAAC,CAO/D,CAAC;IAEF,MAAM,MAAM,GAAG;QACb,UAAU,EAAE,EAAE,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE;QACpC,SAAS,EAAE,EAAE,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE;QACnC,QAAQ,EAAE,EAAE,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE;KACnC,CAAC;IAEF,MAAM,KAAK,GAAG,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,QAAQ,EAAE,IAAI,CAAC,EAAE,EAAE;QAC7D,MAAM,eAAe,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAC9C,MAAM,cAAc,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAC7C,MAAM,YAAY,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE,CAC3D,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CACvC,CAAC;QAEF,MAAM,cAAc,GAAG,eAAe,CAAC,MAAM,CAAC;QAC9C,MAAM,aAAa,GAAG,cAAc,CAAC,MAAM,CAAC;QAC5C,MAAM,WAAW,GAAG,YAAY,CAAC,MAAM,CAAC;QAExC,MAAM,gBAAgB,GAAG,eAAe,CAAC,MAAM,CAC7C,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,GAAG,CAAC,CACrB,CAAC,MAAM,CAAC;QACT,MAAM,eAAe,GAAG,cAAc,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC;QAC3E,MAAM,aAAa,GAAG,YAAY,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC;QAEvE,MAAM,CAAC,UAAU,CAAC,OAAO,IAAI,gBAAgB,CAAC;QAC9C,MAAM,CAAC,UAAU,CAAC,KAAK,IAAI,cAAc,CAAC;QAC1C,MAAM,CAAC,SAAS,CAAC,OAAO,IAAI,eAAe,CAAC;QAC5C,MAAM,CAAC,SAAS,CAAC,KAAK,IAAI,aAAa,CAAC;QACxC,MAAM,CAAC,QAAQ,CAAC,OAAO,IAAI,aAAa,CAAC;QACzC,MAAM,CAAC,QAAQ,CAAC,KAAK,IAAI,WAAW,CAAC;QAErC,MAAM,GAAG,GAAG,CAAC,OAAe,EAAE,KAAa,EAAE,EAAE,CAC7C,KAAK,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,OAAO,GAAG,KAAK,CAAC,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;QAEnE,OAAO;YACL,IAAI,EAAE,QAAQ;YACd,UAAU,EAAE,GAAG,CAAC,gBAAgB,EAAE,cAAc,CAAC;YACjD,SAAS,EAAE,GAAG,CAAC,eAAe,EAAE,aAAa,CAAC;YAC9C,QAAQ,EAAE,GAAG,CAAC,aAAa,EAAE,WAAW,CAAC;SAC1C,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,MAAM,GAAG,GAAG,CAAC,OAAe,EAAE,KAAa,EAAE,EAAE,CAC7C,KAAK,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,OAAO,GAAG,KAAK,CAAC,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;IAEnE,OAAO;QACL,MAAM,EAAE;YACN,UAAU,EAAE;gBACV,GAAG,MAAM,CAAC,UAAU;gBACpB,GAAG,EAAE,GAAG,CAAC,MAAM,CAAC,UAAU,CAAC,OAAO,EAAE,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC;aAC7D;YACD,SAAS,EAAE;gBACT,GAAG,MAAM,CAAC,SAAS;gBACnB,GAAG,EAAE,GAAG,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,EAAE,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC;aAC3D;YACD,QAAQ,EAAE;gBACR,GAAG,MAAM,CAAC,QAAQ;gBAClB,GAAG,EAAE,GAAG,CAAC,MAAM,CAAC,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC;aACzD;SACF;QACD,KAAK;KACN,CAAC;AACJ,CAAC;AAED,SAAS,uBAAuB,CAAC,QAAyB;IACxD,OAAO,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;QAChC,IAAI,EAAE,OAAO,CAAC,IAAI;QAClB,KAAK,EAAE,OAAO,CAAC,KAAK;QACpB,KAAK,EAAE,cAAc,CAAC,OAAO,CAAC,OAAO,CAAC;QACtC,OAAO,EAAE,OAAO,CAAC,OAAO;KACzB,CAAC,CAAC,CAAC;AACN,CAAC;AAED,KAAK,UAAU,eAAe,CAAC,QAAgB;IAC7C,MAAM,IAAI,GAAG,IAAA,4BAAgB,GAAE,CAAC;IAChC,IAAI,CAAC;QACH,OAAO,IAAA,8BAAkB,EAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;IAC5C,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAI,KAAK,YAAY,0BAAc,EAAE,CAAC;YACpC,MAAM,IAAA,0BAAc,EAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QACtC,CAAC;QACD,MAAM,KAAK,CAAC;IACd,CAAC;AACH,CAAC;AAEY,QAAA,kBAAkB,GAC7B;IACE,IAAI,EAAE,iBAAiB;IACvB,WAAW,EACT,sGAAsG;IACxG,UAAU,EAAE,8BAAoB;IAChC,6DAA6D;IAC7D,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,QAAQ,EAA0B,EAAE;QACzD,MAAM,IAAI,GAAG,8BAAoB,CAAC,KAAK,CAAC,KAA2B,CAAC,CAAC;QACrE,MAAM,QAAQ,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEtD,MAAM,YAAY,GAAG,IAAA,qCAA2B,GAAE,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;QAC1E,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,CAAC;YACzB,MAAM,IAAA,0BAAc,EAClB,iDAAiD,IAAI,CAAC,UAAU,EAAE,CACnE,CAAC;QACJ,CAAC;QAED,MAAM,QAAQ,GAAG,uBAAuB,CACtC,qBAAqB,CAAC,YAAY,CAAC,CACpC,CAAC;QAEF,MAAM,MAAM,GAAG,cAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;QAC1C,MAAM,OAAO,GAAG,cAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;QAE7C,MAAM,WAAW,GAAG,YAAE,CAAC,UAAU,CAAC,MAAM,CAAC;YACvC,CAAC,CAAC,IAAA,0BAAkB,EAAC,MAAM,EAAE,mBAAY,CAAC;YAC1C,CAAC,CAAC,EAAE,CAAC;QACP,MAAM,SAAS,GAAG,YAAE,CAAC,UAAU,CAAC,OAAO,CAAC;YACtC,CAAC,CAAC,IAAA,0BAAkB,EAChB,OAAO,EACP,CAAC,IAAI,EAAE,EAAE,CAAC,IAAA,mBAAY,EAAC,IAAI,CAAC,IAAI,IAAA,iBAAU,EAAC,IAAI,CAAC,CACjD;YACH,CAAC,CAAC,EAAE,CAAC;QAEP,IAAI,iBAAqC,CAAC;QAC1C,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,IAAI,CAAC;gBACH,MAAM,QAAQ,GAAG,IAAA,8BAAkB,EAAC,QAAQ,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;gBAC/D,iBAAiB,GAAG,IAAA,oBAAY,EAAC,QAAQ,CAAC,IAAI,SAAS,CAAC;YAC1D,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,IAAI,KAAK,YAAY,0BAAc,EAAE,CAAC;oBACpC,MAAM,IAAA,0BAAc,EAAC,KAAK,CAAC,OAAO,CAAC,CAAC;gBACtC,CAAC;gBACD,MAAM,KAAK,CAAC;YACd,CAAC;QACH,CAAC;QAED,MAAM,OAAO,GAAG;YACd,QAAQ,EAAE,cAAI,CAAC,QAAQ,CAAC,IAAA,4BAAgB,GAAE,EAAE,QAAQ,CAAC,IAAI,GAAG;YAC5D,UAAU,EAAE,IAAI,CAAC,UAAU;YAC3B,UAAU,EAAE,IAAI,CAAC,UAAU;YAC3B,QAAQ,EAAE,QAAQ;YAClB,KAAK,EAAE;gBACL,MAAM,EAAE,aAAa,CAAC,QAAQ,EAAE,WAAW,CAAC;gBAC5C,KAAK,EAAE,aAAa,CAAC,QAAQ,EAAE,SAAS,CAAC;aAC1C;YACD,iBAAiB,EAAE,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,iBAAiB,CAAC,CAAC,CAAC,SAAS;SACvE,CAAC;QAEF,OAAO;YACL,OAAO,EAAE;gBACP;oBACE,IAAI,EAAE,MAAM;oBACZ,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC;iBACvC;aACF;SACsB,CAAC;IAC5B,CAAC;CACF,CAAC;AAES,QAAA,oBAAoB,GAC/B;IACE,IAAI,EAAE,sBAAsB;IAC5B,WAAW,EACT,+GAA+G;IACjH,UAAU,EAAE,4BAAkB;IAC9B,6DAA6D;IAC7D,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,QAAQ,EAA0B,EAAE;QACzD,MAAM,IAAI,GAAG,4BAAkB,CAAC,KAAK,CAAC,KAAyB,CAAC,CAAC;QACjE,MAAM,QAAQ,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEtD,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;YACjB,MAAM,GAAG,GAAG;gBACV,GAAG,OAAO,CAAC,GAAG;gBACd,YAAY,EAAE,OAAO;gBACrB,gBAAgB,EAAE,GAAG;gBACrB,qBAAqB,EAAE,GAAG;aAC3B,CAAC;YACF,MAAM,MAAM,GAAG,IAAA,yBAAS,EACtB,KAAK,EACL,CAAC,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,kBAAkB,EAAE,aAAa,CAAC,EAC5D,EAAE,GAAG,EAAE,QAAQ,EAAE,GAAG,EAAE,QAAQ,EAAE,MAAM,EAAE,CACzC,CAAC;YAEF,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACxB,MAAM,OAAO,GACX,MAAM,CAAC,MAAM,IAAI,MAAM,CAAC,MAAM,IAAI,yBAAyB,CAAC;gBAC9D,MAAM,IAAA,0BAAc,EAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;YACvC,CAAC;QACH,CAAC;QAED,MAAM,YAAY,GAAG,cAAI,CAAC,IAAI,CAC5B,QAAQ,EACR,UAAU,EACV,SAAS,EACT,UAAU,EACV,qBAAqB,CACtB,CAAC;QAEF,IAAI,CAAC,YAAE,CAAC,UAAU,CAAC,YAAY,CAAC,EAAE,CAAC;YACjC,MAAM,IAAA,0BAAc,EAClB,gCAAgC,cAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,YAAY,CAAC,EAAE,CACxE,CAAC;QACJ,CAAC;QAED,MAAM,OAAO,GAAG,wBAAwB,CAAC,YAAY,CAAC,CAAC;QACvD,MAAM,cAAc,GAClB,OAAO,CAAC,MAAM,CAAC,UAAU,CAAC,GAAG,IAAI,IAAI,CAAC,QAAQ;YAC9C,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,GAAG,IAAI,IAAI,CAAC,QAAQ;YAC7C,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,IAAI,IAAI,CAAC,QAAQ,CAAC;QAE/C,MAAM,OAAO,GAAG,CAAC,GAAG,OAAO,CAAC,KAAK,CAAC;aAC/B,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,UAAU,GAAG,CAAC,CAAC,UAAU,CAAC;aAC3C,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QAEhB,MAAM,QAAQ,GAAG,uBAAuB,CACtC,qBAAqB,CAAC,CAAC,YAAY,CAAC,CAAC,CACtC,CAAC;QAEF,MAAM,OAAO,GAAG;YACd,QAAQ,EAAE,cAAI,CAAC,QAAQ,CAAC,IAAA,4BAAgB,GAAE,EAAE,QAAQ,CAAC,IAAI,GAAG;YAC5D,MAAM,EAAE,IAAI,CAAC,QAAQ;YACrB,cAAc;YACd,MAAM,EAAE,OAAO,CAAC,MAAM;YACtB,OAAO;YACP,QAAQ;SACT,CAAC;QAEF,OAAO;YACL,OAAO,EAAE;gBACP;oBACE,IAAI,EAAE,MAAM;oBACZ,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC;iBACvC;aACF;SACsB,CAAC;IAC5B,CAAC;CACF,CAAC;AAES,QAAA,qBAAqB,GAG9B;IACF,IAAI,EAAE,gBAAgB;IACtB,WAAW,EACT,gHAAgH;IAClH,UAAU,EAAE,iCAAuB;IACnC,6DAA6D;IAC7D,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,QAAQ,EAA0B,EAAE;QACzD,MAAM,IAAI,GAAG,iCAAuB,CAAC,KAAK,CAAC,KAA8B,CAAC,CAAC;QAC3E,MAAM,QAAQ,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEtD,MAAM,QAAQ,GAAG,IAAA,kBAAW,EAAC,QAAQ,CAAC,CAAC;QACvC,MAAM,OAAO,GAAG,QAAQ,CAAC,KAAK;aAC3B,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;aACzC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,cAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC,CAAC;QAEhD,MAAM,cAAc,GAAG,uBAAuB,CAC5C,qBAAqB,CAAC,CAAC,eAAe,EAAE,KAAK,EAAE,QAAQ,CAAC,CAAC,CAC1D,CAAC;QAEF,MAAM,YAAY,GAAG,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC;QACvD,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QAEvE,MAAM,OAAO,GAAG;YACd,QAAQ,EAAE,cAAI,CAAC,QAAQ,CAAC,IAAA,4BAAgB,GAAE,EAAE,QAAQ,CAAC,IAAI,GAAG;YAC5D,OAAO,EAAE;gBACP,OAAO;gBACP,gBAAgB,EAAE,QAAQ,CAAC,KAAK,CAAC,MAAM;gBACvC,cAAc,EAAE,QAAQ,CAAC,SAAS,CAAC,MAAM;gBACzC,SAAS,EAAE,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC;aACpC;YACD,QAAQ,EAAE,cAAc;YACxB,iBAAiB,EAAE,QAAQ;SAC5B,CAAC;QAEF,OAAO;YACL,OAAO,EAAE;gBACP;oBACE,IAAI,EAAE,MAAM;oBACZ,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC;iBACvC;aACF;SACsB,CAAC;IAC5B,CAAC;CACF,CAAC","sourcesContent":["import fs from \"fs\";\nimport path from \"path\";\nimport { spawnSync } from \"child_process\";\nimport type { ContentResult, Tool } from \"fastmcp\";\nimport {\n  documentObjectSchema,\n  coverageTaskSchema,\n  readmeImprovementSchema,\n} from \"../schemas\";\nimport {\n  DocumentObjectArgs,\n  CoverageTaskArgs,\n  ReadmeImprovementArgs,\n} from \"../types\";\nimport {\n  getWorkspaceRoot,\n  resolveInWorkspace,\n  throwUserError,\n  WorkspaceError,\n} from \"../workspace\";\nimport {\n  buildObjectPrompts,\n  discoverDocPrompts,\n  getObjectPromptDependencies,\n} from \"../prompts/prompts\";\nimport type { DocPrompt } from \"../types\";\nimport { listFilesRecursive, readFileSafe } from \"../utils\";\nimport { analyzeRepo, isSourceFile, isTestFile } from \"../code\";\n\ntype PromptSection = {\n  name: string;\n  title: string;\n  description: string;\n  content: string;\n  absolutePath?: string;\n};\n\nfunction relativeFiles(root: string, files: string[]): string[] {\n  return files.map((file) => path.relative(root, file)).sort();\n}\n\nfunction collectPromptSections(names: readonly string[]): PromptSection[] {\n  const root = getWorkspaceRoot();\n  const promptIndex = new Map<string, DocPrompt>(\n    discoverDocPrompts(root).map((prompt) => [prompt.name, prompt])\n  );\n  return names\n    .map((name) => promptIndex.get(name))\n    .filter((prompt): prompt is DocPrompt => Boolean(prompt))\n    .map((prompt) => ({\n      name: prompt.name,\n      title: prompt.title,\n      description: prompt.description,\n      content: prompt.content,\n      absolutePath: prompt.absolutePath,\n    }));\n}\n\nfunction parseTaskLines(content: string): string[] {\n  return content\n    .split(/\\r?\\n/)\n    .map((line) => line.trim())\n    .filter((line) => /^task\\s+\\d+/i.test(line));\n}\n\nfunction computeCoverageFromFinal(coveragePath: string) {\n  const payload = JSON.parse(fs.readFileSync(coveragePath, \"utf8\")) as Record<\n    string,\n    {\n      s: Record<string, number>;\n      f: Record<string, number>;\n      b: Record<string, number | number[]>;\n    }\n  >;\n\n  const totals = {\n    statements: { covered: 0, total: 0 },\n    functions: { covered: 0, total: 0 },\n    branches: { covered: 0, total: 0 },\n  };\n\n  const files = Object.entries(payload).map(([filePath, info]) => {\n    const statementCounts = Object.values(info.s);\n    const functionCounts = Object.values(info.f);\n    const branchCounts = Object.values(info.b).flatMap((value) =>\n      Array.isArray(value) ? value : [value]\n    );\n\n    const statementTotal = statementCounts.length;\n    const functionTotal = functionCounts.length;\n    const branchTotal = branchCounts.length;\n\n    const statementCovered = statementCounts.filter(\n      (count) => count > 0\n    ).length;\n    const functionCovered = functionCounts.filter((count) => count > 0).length;\n    const branchCovered = branchCounts.filter((count) => count > 0).length;\n\n    totals.statements.covered += statementCovered;\n    totals.statements.total += statementTotal;\n    totals.functions.covered += functionCovered;\n    totals.functions.total += functionTotal;\n    totals.branches.covered += branchCovered;\n    totals.branches.total += branchTotal;\n\n    const pct = (covered: number, total: number) =>\n      total === 0 ? 100 : Number(((covered / total) * 100).toFixed(2));\n\n    return {\n      path: filePath,\n      statements: pct(statementCovered, statementTotal),\n      functions: pct(functionCovered, functionTotal),\n      branches: pct(branchCovered, branchTotal),\n    };\n  });\n\n  const pct = (covered: number, total: number) =>\n    total === 0 ? 100 : Number(((covered / total) * 100).toFixed(2));\n\n  return {\n    totals: {\n      statements: {\n        ...totals.statements,\n        pct: pct(totals.statements.covered, totals.statements.total),\n      },\n      functions: {\n        ...totals.functions,\n        pct: pct(totals.functions.covered, totals.functions.total),\n      },\n      branches: {\n        ...totals.branches,\n        pct: pct(totals.branches.covered, totals.branches.total),\n      },\n    },\n    files,\n  };\n}\n\nfunction normalizePromptSections(sections: PromptSection[]) {\n  return sections.map((section) => ({\n    name: section.name,\n    title: section.title,\n    tasks: parseTaskLines(section.content),\n    content: section.content,\n  }));\n}\n\nasync function resolveRepoRoot(basePath: string) {\n  const root = getWorkspaceRoot();\n  try {\n    return resolveInWorkspace(root, basePath);\n  } catch (error) {\n    if (error instanceof WorkspaceError) {\n      await throwUserError(error.message);\n    }\n    throw error;\n  }\n}\n\nexport const documentObjectTool: Tool<undefined, typeof documentObjectSchema> =\n  {\n    name: \"document-object\",\n    description:\n      \"Create a documentation plan for a specific object type using .codex prompts and repository analysis.\",\n    parameters: documentObjectSchema,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    execute: async (input, _context): Promise<ContentResult> => {\n      const args = documentObjectSchema.parse(input as DocumentObjectArgs);\n      const repoRoot = await resolveRepoRoot(args.basePath);\n\n      const dependencies = getObjectPromptDependencies()[args.objectType] ?? [];\n      if (!dependencies.length) {\n        await throwUserError(\n          `No prompt guidance configured for object type ${args.objectType}`\n        );\n      }\n\n      const sections = normalizePromptSections(\n        collectPromptSections(dependencies)\n      );\n\n      const srcDir = path.join(repoRoot, \"src\");\n      const testDir = path.join(repoRoot, \"tests\");\n\n      const sourceFiles = fs.existsSync(srcDir)\n        ? listFilesRecursive(srcDir, isSourceFile)\n        : [];\n      const testFiles = fs.existsSync(testDir)\n        ? listFilesRecursive(\n            testDir,\n            (file) => isSourceFile(file) && isTestFile(file)\n          )\n        : [];\n\n      let targetFileContent: string | undefined;\n      if (args.targetFile) {\n        try {\n          const absolute = resolveInWorkspace(repoRoot, args.targetFile);\n          targetFileContent = readFileSafe(absolute) ?? undefined;\n        } catch (error) {\n          if (error instanceof WorkspaceError) {\n            await throwUserError(error.message);\n          }\n          throw error;\n        }\n      }\n\n      const payload = {\n        basePath: path.relative(getWorkspaceRoot(), repoRoot) || \".\",\n        objectType: args.objectType,\n        targetFile: args.targetFile,\n        guidance: sections,\n        files: {\n          source: relativeFiles(repoRoot, sourceFiles),\n          tests: relativeFiles(repoRoot, testFiles),\n        },\n        targetFileContent: args.includeContent ? targetFileContent : undefined,\n      };\n\n      return {\n        content: [\n          {\n            type: \"text\",\n            text: JSON.stringify(payload, null, 2),\n          },\n        ],\n      } satisfies ContentResult;\n    },\n  };\n\nexport const coverageEnforcerTool: Tool<undefined, typeof coverageTaskSchema> =\n  {\n    name: \"ensure-test-coverage\",\n    description:\n      \"Run the configured coverage command and report whether the target percentage is met, highlighting weak files.\",\n    parameters: coverageTaskSchema,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    execute: async (input, _context): Promise<ContentResult> => {\n      const args = coverageTaskSchema.parse(input as CoverageTaskArgs);\n      const repoRoot = await resolveRepoRoot(args.basePath);\n\n      if (!args.dryRun) {\n        const env = {\n          ...process.env,\n          USE_WATCHMAN: \"false\",\n          WATCHMAN_DISABLE: \"1\",\n          JEST_DISABLE_WATCHMAN: \"1\",\n        };\n        const result = spawnSync(\n          \"npm\",\n          [\"run\", \"coverage\", \"--\", \"--watchman=false\", \"--runInBand\"],\n          { cwd: repoRoot, env, encoding: \"utf8\" }\n        );\n\n        if (result.status !== 0) {\n          const message =\n            result.stderr || result.stdout || \"Coverage command failed\";\n          await throwUserError(message.trim());\n        }\n      }\n\n      const coveragePath = path.join(\n        repoRoot,\n        \"workdocs\",\n        \"reports\",\n        \"coverage\",\n        \"coverage-final.json\"\n      );\n\n      if (!fs.existsSync(coveragePath)) {\n        await throwUserError(\n          `Coverage report not found at ${path.relative(repoRoot, coveragePath)}`\n        );\n      }\n\n      const summary = computeCoverageFromFinal(coveragePath);\n      const meetsThreshold =\n        summary.totals.statements.pct >= args.coverage &&\n        summary.totals.functions.pct >= args.coverage &&\n        summary.totals.branches.pct >= args.coverage;\n\n      const weakest = [...summary.files]\n        .sort((a, b) => a.statements - b.statements)\n        .slice(0, 10);\n\n      const guidance = normalizePromptSections(\n        collectPromptSections([\"bulk-tests\"])\n      );\n\n      const payload = {\n        basePath: path.relative(getWorkspaceRoot(), repoRoot) || \".\",\n        target: args.coverage,\n        meetsThreshold,\n        totals: summary.totals,\n        weakest,\n        guidance,\n      };\n\n      return {\n        content: [\n          {\n            type: \"text\",\n            text: JSON.stringify(payload, null, 2),\n          },\n        ],\n      } satisfies ContentResult;\n    },\n  };\n\nexport const readmeImprovementTool: Tool<\n  undefined,\n  typeof readmeImprovementSchema\n> = {\n  name: \"improve-readme\",\n  description:\n    \"Summarize required steps to refresh README and workdocs content using .codex guidance and repository analysis.\",\n  parameters: readmeImprovementSchema,\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  execute: async (input, _context): Promise<ContentResult> => {\n    const args = readmeImprovementSchema.parse(input as ReadmeImprovementArgs);\n    const repoRoot = await resolveRepoRoot(args.basePath);\n\n    const analysis = analyzeRepo(repoRoot);\n    const modules = analysis.files\n      .filter((file) => /index\\.ts$/.test(file))\n      .map((file) => path.relative(repoRoot, file));\n\n    const promptSections = normalizePromptSections(\n      collectPromptSections([\"update-readme\", \"doc\", \"module\"])\n    );\n\n    const testExamples = Object.keys(analysis.tests ?? {});\n    const examples = args.includeExamples ? testExamples.slice(0, 20) : [];\n\n    const payload = {\n      basePath: path.relative(getWorkspaceRoot(), repoRoot) || \".\",\n      summary: {\n        modules,\n        totalSourceFiles: analysis.files.length,\n        totalTestFiles: analysis.testFiles.length,\n        hasReadme: Boolean(analysis.readme),\n      },\n      guidance: promptSections,\n      suggestedExamples: examples,\n    };\n\n    return {\n      content: [\n        {\n          type: \"text\",\n          text: JSON.stringify(payload, null, 2),\n        },\n      ],\n    } satisfies ContentResult;\n  },\n};\n"]}
@@ -0,0 +1,5 @@
1
+ import type { Tool } from "fastmcp";
2
+ import { documentObjectSchema, coverageTaskSchema, readmeImprovementSchema } from "../schemas";
3
+ export declare const documentObjectTool: Tool<undefined, typeof documentObjectSchema>;
4
+ export declare const coverageEnforcerTool: Tool<undefined, typeof coverageTaskSchema>;
5
+ export declare const readmeImprovementTool: Tool<undefined, typeof readmeImprovementSchema>;
@@ -0,0 +1,139 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.generateMcpModuleTool = void 0;
7
+ // New tool: generate-mcp-module
8
+ const fs_1 = __importDefault(require("fs"));
9
+ const path_1 = __importDefault(require("path"));
10
+ const zod_1 = require("zod");
11
+ const workspace_1 = require("./../workspace.cjs");
12
+ const code_1 = require("./../code.cjs");
13
+ const scaffoldModule_1 = require("./../validation/scaffoldModule.cjs");
14
+ const generateSchema = zod_1.z.object({
15
+ repoPath: zod_1.z.string().optional().default("."),
16
+ moduleName: zod_1.z.string().optional(),
17
+ includeDocs: zod_1.z.boolean().default(true),
18
+ });
19
+ exports.generateMcpModuleTool = {
20
+ name: "generate-mcp-module",
21
+ description: "Generate a minimal MCP module under src/modules/<name> by analyzing a target repository and exporting prompts, resources, templates and tools.",
22
+ parameters: generateSchema,
23
+ execute: async (input) => {
24
+ const args = generateSchema.parse(input);
25
+ const root = (0, workspace_1.getWorkspaceRoot)();
26
+ let repoRoot = path_1.default.resolve(process.cwd(), args.repoPath || ".");
27
+ if (!fs_1.default.existsSync(repoRoot)) {
28
+ const alt = path_1.default.resolve(process.cwd(), "..", args.repoPath);
29
+ if (fs_1.default.existsSync(alt))
30
+ repoRoot = alt;
31
+ }
32
+ if (!fs_1.default.existsSync(repoRoot)) {
33
+ const alt2 = path_1.default.resolve(process.cwd(), "..", path_1.default.basename(args.repoPath));
34
+ if (fs_1.default.existsSync(alt2))
35
+ repoRoot = alt2;
36
+ }
37
+ if (!fs_1.default.existsSync(repoRoot))
38
+ throw new Error(`Repository not found at ${repoRoot}`);
39
+ const analysis = (0, code_1.analyzeRepo)(repoRoot);
40
+ const inferredName = args.moduleName ?? path_1.default.basename(path_1.default.resolve(repoRoot));
41
+ const moduleRoot = path_1.default.join(root, "src", "modules", inferredName);
42
+ // Use existing scaffold to create folders
43
+ (0, scaffoldModule_1.scaffoldModule)(root, inferredName);
44
+ // populate prompts: copy markdown prompts from README.md and docs/
45
+ const promptsDir = path_1.default.join(moduleRoot, "prompts");
46
+ const resourcesDir = path_1.default.join(moduleRoot, "resources");
47
+ const templatesDir = path_1.default.join(moduleRoot, "templates");
48
+ const toolsDir = path_1.default.join(moduleRoot, "tools");
49
+ // helper to write an index.ts that exports arrays
50
+ function writeIndex(dir, varName, items) {
51
+ const p = path_1.default.join(dir, "index.ts");
52
+ const content = `export const ${varName} = ${items} as const;\n`;
53
+ fs_1.default.writeFileSync(p, content, { encoding: "utf8" });
54
+ }
55
+ // Prompts: create a simple prompt from README and any .md in docs
56
+ const promptAssets = [];
57
+ if (args.includeDocs) {
58
+ const readme = path_1.default.join(repoRoot, "README.md");
59
+ if (fs_1.default.existsSync(readme)) {
60
+ const content = fs_1.default.readFileSync(readme, "utf8");
61
+ const id = "readme";
62
+ const promptPath = path_1.default.join(promptsDir, `${id}.md`);
63
+ fs_1.default.writeFileSync(promptPath, content, { encoding: "utf8" });
64
+ promptAssets.push({
65
+ id,
66
+ title: "README",
67
+ description: "Repository README",
68
+ absolutePath: promptPath,
69
+ load: () => content,
70
+ });
71
+ }
72
+ const docsDir = path_1.default.join(repoRoot, "docs");
73
+ if (fs_1.default.existsSync(docsDir) && fs_1.default.statSync(docsDir).isDirectory()) {
74
+ for (const f of fs_1.default.readdirSync(docsDir)) {
75
+ if (!f.endsWith(".md"))
76
+ continue;
77
+ const content = fs_1.default.readFileSync(path_1.default.join(docsDir, f), "utf8");
78
+ const id = path_1.default.parse(f).name;
79
+ const promptPath = path_1.default.join(promptsDir, `${id}.md`);
80
+ fs_1.default.writeFileSync(promptPath, content, { encoding: "utf8" });
81
+ promptAssets.push({
82
+ id,
83
+ title: id,
84
+ description: content.split(/\r?\n/)[0] || "",
85
+ absolutePath: promptPath,
86
+ load: () => content,
87
+ });
88
+ }
89
+ }
90
+ }
91
+ writeIndex(promptsDir, "prompts", JSON.stringify(promptAssets, null, 2));
92
+ // Resources: reference the repo root and docs
93
+ const resourceAssets = [
94
+ {
95
+ id: `${inferredName}.repo`,
96
+ name: `${inferredName} repository`,
97
+ description: "Source repository",
98
+ uri: `file://${repoRoot}`,
99
+ absolutePath: repoRoot,
100
+ },
101
+ ];
102
+ writeIndex(resourcesDir, "resources", JSON.stringify(resourceAssets, null, 2));
103
+ // Templates: create a placeholder template that references README
104
+ const templateAssets = [
105
+ {
106
+ name: "readme-template",
107
+ description: "README as guidance",
108
+ uriTemplate: `file://${path_1.default.join(repoRoot, "README.md")}`,
109
+ mimeType: "text/markdown",
110
+ },
111
+ ];
112
+ writeIndex(templatesDir, "templates", JSON.stringify(templateAssets, null, 2));
113
+ // Tools: create a wrapper tool that exposes analyze/enumerate for the module
114
+ const toolIndexPath = path_1.default.join(toolsDir, "index.ts");
115
+ const toolContent = `import type { Tool } from 'fastmcp';\nimport { buildAnalyzeRepositoryTool, buildEnumerateCapabilitiesTool, buildPlanFeatureTool } from '../../../mcp/tools/tools';\nexport const tools = [\n { id: '${inferredName}.analyze', title: 'Analyze ${inferredName}', description: 'Analyze the target repository', tool: buildAnalyzeRepositoryTool() },\n { id: '${inferredName}.enumerate', title: 'Enumerate capabilities for ${inferredName}', description: 'Enumerate capabilities', tool: buildEnumerateCapabilitiesTool() },\n { id: '${inferredName}.plan', title: 'Plan features for ${inferredName}', description: 'Plan feature implementation', tool: buildPlanFeatureTool() },\n] as const;\n`;
116
+ fs_1.default.writeFileSync(toolIndexPath, toolContent, { encoding: "utf8" });
117
+ // Write module index.ts
118
+ const moduleIndex = path_1.default.join(moduleRoot, "index.ts");
119
+ const moduleIndexContent = `import { prompts } from './prompts';\nimport { resources } from './resources';\nimport { templates } from './templates';\nimport { tools } from './tools';\nexport { prompts } from './prompts';\nexport { resources } from './resources';\nexport { templates } from './templates';\nexport { tools } from './tools';\nexport const modulePackage = { name: '${inferredName}', prompts, resources, templates, tools } as const;\n`;
120
+ fs_1.default.writeFileSync(moduleIndex, moduleIndexContent, { encoding: "utf8" });
121
+ return {
122
+ content: [
123
+ {
124
+ type: "text",
125
+ text: JSON.stringify({
126
+ moduleRoot,
127
+ inferredName,
128
+ analysisSummary: {
129
+ files: analysis.files.length,
130
+ testFiles: analysis.testFiles.length,
131
+ },
132
+ }, null, 2),
133
+ },
134
+ ],
135
+ };
136
+ },
137
+ };
138
+ exports.default = exports.generateMcpModuleTool;
139
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"generateMcpModule.js","sourceRoot":"","sources":["../../../src/mcp/tools/generateMcpModule.ts"],"names":[],"mappings":";;;;;;AAAA,gCAAgC;AAChC,4CAAoB;AACpB,gDAAwB;AAExB,6BAAwB;AACxB,kDAAgD;AAChD,wCAAsC;AACtC,uEAA8D;AAE9D,MAAM,cAAc,GAAG,OAAC,CAAC,MAAM,CAAC;IAC9B,QAAQ,EAAE,OAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC;IAC5C,UAAU,EAAE,OAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE;IACjC,WAAW,EAAE,OAAC,CAAC,OAAO,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC;CACvC,CAAC,CAAC;AAIU,QAAA,qBAAqB,GAA2C;IAC3E,IAAI,EAAE,qBAAqB;IAC3B,WAAW,EACT,gJAAgJ;IAClJ,UAAU,EAAE,cAAc;IAC1B,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,EAAE;QACvB,MAAM,IAAI,GAAG,cAAc,CAAC,KAAK,CAAC,KAAY,CAAC,CAAC;QAChD,MAAM,IAAI,GAAG,IAAA,4BAAgB,GAAE,CAAC;QAChC,IAAI,QAAQ,GAAG,cAAI,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,QAAQ,IAAI,GAAG,CAAC,CAAC;QACjE,IAAI,CAAC,YAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC7B,MAAM,GAAG,GAAG,cAAI,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC7D,IAAI,YAAE,CAAC,UAAU,CAAC,GAAG,CAAC;gBAAE,QAAQ,GAAG,GAAG,CAAC;QACzC,CAAC;QACD,IAAI,CAAC,YAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC7B,MAAM,IAAI,GAAG,cAAI,CAAC,OAAO,CACvB,OAAO,CAAC,GAAG,EAAE,EACb,IAAI,EACJ,cAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,CAC7B,CAAC;YACF,IAAI,YAAE,CAAC,UAAU,CAAC,IAAI,CAAC;gBAAE,QAAQ,GAAG,IAAI,CAAC;QAC3C,CAAC;QACD,IAAI,CAAC,YAAE,CAAC,UAAU,CAAC,QAAQ,CAAC;YAC1B,MAAM,IAAI,KAAK,CAAC,2BAA2B,QAAQ,EAAE,CAAC,CAAC;QAEzD,MAAM,QAAQ,GAAG,IAAA,kBAAW,EAAC,QAAQ,CAAC,CAAC;QACvC,MAAM,YAAY,GAChB,IAAI,CAAC,UAAU,IAAI,cAAI,CAAC,QAAQ,CAAC,cAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC;QAC3D,MAAM,UAAU,GAAG,cAAI,CAAC,IAAI,CAAC,IAAI,EAAE,KAAK,EAAE,SAAS,EAAE,YAAY,CAAC,CAAC;QAEnE,0CAA0C;QAC1C,IAAA,+BAAc,EAAC,IAAI,EAAE,YAAY,CAAC,CAAC;QAEnC,mEAAmE;QACnE,MAAM,UAAU,GAAG,cAAI,CAAC,IAAI,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;QACpD,MAAM,YAAY,GAAG,cAAI,CAAC,IAAI,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC;QACxD,MAAM,YAAY,GAAG,cAAI,CAAC,IAAI,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC;QACxD,MAAM,QAAQ,GAAG,cAAI,CAAC,IAAI,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;QAEhD,kDAAkD;QAClD,SAAS,UAAU,CAAC,GAAW,EAAE,OAAe,EAAE,KAAa;YAC7D,MAAM,CAAC,GAAG,cAAI,CAAC,IAAI,CAAC,GAAG,EAAE,UAAU,CAAC,CAAC;YACrC,MAAM,OAAO,GAAG,gBAAgB,OAAO,MAAM,KAAK,cAAc,CAAC;YACjE,YAAE,CAAC,aAAa,CAAC,CAAC,EAAE,OAAO,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;QACrD,CAAC;QAED,kEAAkE;QAClE,MAAM,YAAY,GAAU,EAAE,CAAC;QAC/B,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YACrB,MAAM,MAAM,GAAG,cAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC;YAChD,IAAI,YAAE,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;gBAC1B,MAAM,OAAO,GAAG,YAAE,CAAC,YAAY,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;gBAChD,MAAM,EAAE,GAAG,QAAQ,CAAC;gBACpB,MAAM,UAAU,GAAG,cAAI,CAAC,IAAI,CAAC,UAAU,EAAE,GAAG,EAAE,KAAK,CAAC,CAAC;gBACrD,YAAE,CAAC,aAAa,CAAC,UAAU,EAAE,OAAO,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;gBAC5D,YAAY,CAAC,IAAI,CAAC;oBAChB,EAAE;oBACF,KAAK,EAAE,QAAQ;oBACf,WAAW,EAAE,mBAAmB;oBAChC,YAAY,EAAE,UAAU;oBACxB,IAAI,EAAE,GAAG,EAAE,CAAC,OAAO;iBACpB,CAAC,CAAC;YACL,CAAC;YACD,MAAM,OAAO,GAAG,cAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YAC5C,IAAI,YAAE,CAAC,UAAU,CAAC,OAAO,CAAC,IAAI,YAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,EAAE,CAAC;gBACjE,KAAK,MAAM,CAAC,IAAI,YAAE,CAAC,WAAW,CAAC,OAAO,CAAC,EAAE,CAAC;oBACxC,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC;wBAAE,SAAS;oBACjC,MAAM,OAAO,GAAG,YAAE,CAAC,YAAY,CAAC,cAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC;oBAC/D,MAAM,EAAE,GAAG,cAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;oBAC9B,MAAM,UAAU,GAAG,cAAI,CAAC,IAAI,CAAC,UAAU,EAAE,GAAG,EAAE,KAAK,CAAC,CAAC;oBACrD,YAAE,CAAC,aAAa,CAAC,UAAU,EAAE,OAAO,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;oBAC5D,YAAY,CAAC,IAAI,CAAC;wBAChB,EAAE;wBACF,KAAK,EAAE,EAAE;wBACT,WAAW,EAAE,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE;wBAC5C,YAAY,EAAE,UAAU;wBACxB,IAAI,EAAE,GAAG,EAAE,CAAC,OAAO;qBACpB,CAAC,CAAC;gBACL,CAAC;YACH,CAAC;QACH,CAAC;QAED,UAAU,CAAC,UAAU,EAAE,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAEzE,8CAA8C;QAC9C,MAAM,cAAc,GAAG;YACrB;gBACE,EAAE,EAAE,GAAG,YAAY,OAAO;gBAC1B,IAAI,EAAE,GAAG,YAAY,aAAa;gBAClC,WAAW,EAAE,mBAAmB;gBAChC,GAAG,EAAE,UAAU,QAAQ,EAAE;gBACzB,YAAY,EAAE,QAAQ;aACvB;SACF,CAAC;QACF,UAAU,CACR,YAAY,EACZ,WAAW,EACX,IAAI,CAAC,SAAS,CAAC,cAAc,EAAE,IAAI,EAAE,CAAC,CAAC,CACxC,CAAC;QAEF,kEAAkE;QAClE,MAAM,cAAc,GAAG;YACrB;gBACE,IAAI,EAAE,iBAAiB;gBACvB,WAAW,EAAE,oBAAoB;gBACjC,WAAW,EAAE,UAAU,cAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,WAAW,CAAC,EAAE;gBACzD,QAAQ,EAAE,eAAe;aAC1B;SACF,CAAC;QACF,UAAU,CACR,YAAY,EACZ,WAAW,EACX,IAAI,CAAC,SAAS,CAAC,cAAc,EAAE,IAAI,EAAE,CAAC,CAAC,CACxC,CAAC;QAEF,6EAA6E;QAC7E,MAAM,aAAa,GAAG,cAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC;QACtD,MAAM,WAAW,GAAG,wMAAwM,YAAY,8BAA8B,YAAY,oGAAoG,YAAY,mDAAmD,YAAY,iGAAiG,YAAY,qCAAqC,YAAY,+FAA+F,CAAC;QAC/rB,YAAE,CAAC,aAAa,CAAC,aAAa,EAAE,WAAW,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;QAEnE,wBAAwB;QACxB,MAAM,WAAW,GAAG,cAAI,CAAC,IAAI,CAAC,UAAU,EAAE,UAAU,CAAC,CAAC;QACtD,MAAM,kBAAkB,GAAG,iWAAiW,YAAY,uDAAuD,CAAC;QAChc,YAAE,CAAC,aAAa,CAAC,WAAW,EAAE,kBAAkB,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;QAExE,OAAO;YACL,OAAO,EAAE;gBACP;oBACE,IAAI,EAAE,MAAM;oBACZ,IAAI,EAAE,IAAI,CAAC,SAAS,CAClB;wBACE,UAAU;wBACV,YAAY;wBACZ,eAAe,EAAE;4BACf,KAAK,EAAE,QAAQ,CAAC,KAAK,CAAC,MAAM;4BAC5B,SAAS,EAAE,QAAQ,CAAC,SAAS,CAAC,MAAM;yBACrC;qBACF,EACD,IAAI,EACJ,CAAC,CACF;iBACF;aACF;SACF,CAAC;IACJ,CAAC;CACF,CAAC;AAEF,kBAAe,6BAAqB,CAAC","sourcesContent":["// New tool: generate-mcp-module\nimport fs from \"fs\";\nimport path from \"path\";\nimport type { Tool } from \"fastmcp\";\nimport { z } from \"zod\";\nimport { getWorkspaceRoot } from \"../workspace\";\nimport { analyzeRepo } from \"../code\";\nimport { scaffoldModule } from \"../validation/scaffoldModule\";\n\nconst generateSchema = z.object({\n  repoPath: z.string().optional().default(\".\"),\n  moduleName: z.string().optional(),\n  includeDocs: z.boolean().default(true),\n});\n\ntype GenerateArgs = z.infer<typeof generateSchema>;\n\nexport const generateMcpModuleTool: Tool<undefined, typeof generateSchema> = {\n  name: \"generate-mcp-module\",\n  description:\n    \"Generate a minimal MCP module under src/modules/<name> by analyzing a target repository and exporting prompts, resources, templates and tools.\",\n  parameters: generateSchema,\n  execute: async (input) => {\n    const args = generateSchema.parse(input as any);\n    const root = getWorkspaceRoot();\n    let repoRoot = path.resolve(process.cwd(), args.repoPath || \".\");\n    if (!fs.existsSync(repoRoot)) {\n      const alt = path.resolve(process.cwd(), \"..\", args.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(args.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\n    const analysis = analyzeRepo(repoRoot);\n    const inferredName =\n      args.moduleName ?? path.basename(path.resolve(repoRoot));\n    const moduleRoot = path.join(root, \"src\", \"modules\", inferredName);\n\n    // Use existing scaffold to create folders\n    scaffoldModule(root, inferredName);\n\n    // populate prompts: copy markdown prompts from README.md and docs/\n    const promptsDir = path.join(moduleRoot, \"prompts\");\n    const resourcesDir = path.join(moduleRoot, \"resources\");\n    const templatesDir = path.join(moduleRoot, \"templates\");\n    const toolsDir = path.join(moduleRoot, \"tools\");\n\n    // helper to write an index.ts that exports arrays\n    function writeIndex(dir: string, varName: string, items: string) {\n      const p = path.join(dir, \"index.ts\");\n      const content = `export const ${varName} = ${items} as const;\\n`;\n      fs.writeFileSync(p, content, { encoding: \"utf8\" });\n    }\n\n    // Prompts: create a simple prompt from README and any .md in docs\n    const promptAssets: any[] = [];\n    if (args.includeDocs) {\n      const readme = path.join(repoRoot, \"README.md\");\n      if (fs.existsSync(readme)) {\n        const content = fs.readFileSync(readme, \"utf8\");\n        const id = \"readme\";\n        const promptPath = path.join(promptsDir, `${id}.md`);\n        fs.writeFileSync(promptPath, content, { encoding: \"utf8\" });\n        promptAssets.push({\n          id,\n          title: \"README\",\n          description: \"Repository README\",\n          absolutePath: promptPath,\n          load: () => content,\n        });\n      }\n      const docsDir = path.join(repoRoot, \"docs\");\n      if (fs.existsSync(docsDir) && fs.statSync(docsDir).isDirectory()) {\n        for (const f of fs.readdirSync(docsDir)) {\n          if (!f.endsWith(\".md\")) continue;\n          const content = fs.readFileSync(path.join(docsDir, f), \"utf8\");\n          const id = path.parse(f).name;\n          const promptPath = path.join(promptsDir, `${id}.md`);\n          fs.writeFileSync(promptPath, content, { encoding: \"utf8\" });\n          promptAssets.push({\n            id,\n            title: id,\n            description: content.split(/\\r?\\n/)[0] || \"\",\n            absolutePath: promptPath,\n            load: () => content,\n          });\n        }\n      }\n    }\n\n    writeIndex(promptsDir, \"prompts\", JSON.stringify(promptAssets, null, 2));\n\n    // Resources: reference the repo root and docs\n    const resourceAssets = [\n      {\n        id: `${inferredName}.repo`,\n        name: `${inferredName} repository`,\n        description: \"Source repository\",\n        uri: `file://${repoRoot}`,\n        absolutePath: repoRoot,\n      },\n    ];\n    writeIndex(\n      resourcesDir,\n      \"resources\",\n      JSON.stringify(resourceAssets, null, 2)\n    );\n\n    // Templates: create a placeholder template that references README\n    const templateAssets = [\n      {\n        name: \"readme-template\",\n        description: \"README as guidance\",\n        uriTemplate: `file://${path.join(repoRoot, \"README.md\")}`,\n        mimeType: \"text/markdown\",\n      },\n    ];\n    writeIndex(\n      templatesDir,\n      \"templates\",\n      JSON.stringify(templateAssets, null, 2)\n    );\n\n    // Tools: create a wrapper tool that exposes analyze/enumerate for the module\n    const toolIndexPath = path.join(toolsDir, \"index.ts\");\n    const toolContent = `import type { Tool } from 'fastmcp';\\nimport { buildAnalyzeRepositoryTool, buildEnumerateCapabilitiesTool, buildPlanFeatureTool } from '../../../mcp/tools/tools';\\nexport const tools = [\\n  { id: '${inferredName}.analyze', title: 'Analyze ${inferredName}', description: 'Analyze the target repository', tool: buildAnalyzeRepositoryTool() },\\n  { id: '${inferredName}.enumerate', title: 'Enumerate capabilities for ${inferredName}', description: 'Enumerate capabilities', tool: buildEnumerateCapabilitiesTool() },\\n  { id: '${inferredName}.plan', title: 'Plan features for ${inferredName}', description: 'Plan feature implementation', tool: buildPlanFeatureTool() },\\n] as const;\\n`;\n    fs.writeFileSync(toolIndexPath, toolContent, { encoding: \"utf8\" });\n\n    // Write module index.ts\n    const moduleIndex = path.join(moduleRoot, \"index.ts\");\n    const moduleIndexContent = `import { prompts } from './prompts';\\nimport { resources } from './resources';\\nimport { templates } from './templates';\\nimport { tools } from './tools';\\nexport { prompts } from './prompts';\\nexport { resources } from './resources';\\nexport { templates } from './templates';\\nexport { tools } from './tools';\\nexport const modulePackage = { name: '${inferredName}', prompts, resources, templates, tools } as const;\\n`;\n    fs.writeFileSync(moduleIndex, moduleIndexContent, { encoding: \"utf8\" });\n\n    return {\n      content: [\n        {\n          type: \"text\",\n          text: JSON.stringify(\n            {\n              moduleRoot,\n              inferredName,\n              analysisSummary: {\n                files: analysis.files.length,\n                testFiles: analysis.testFiles.length,\n              },\n            },\n            null,\n            2\n          ),\n        },\n      ],\n    };\n  },\n};\n\nexport default generateMcpModuleTool;\n"]}
@@ -0,0 +1,9 @@
1
+ import type { Tool } from "fastmcp";
2
+ import { z } from "zod";
3
+ declare const generateSchema: z.ZodObject<{
4
+ repoPath: z.ZodDefault<z.ZodOptional<z.ZodString>>;
5
+ moduleName: z.ZodOptional<z.ZodString>;
6
+ includeDocs: z.ZodDefault<z.ZodBoolean>;
7
+ }, z.core.$strip>;
8
+ export declare const generateMcpModuleTool: Tool<undefined, typeof generateSchema>;
9
+ export default generateMcpModuleTool;
@@ -0,0 +1,46 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ exports.decoratorTools = exports.tools = exports.decoratorToolList = exports.toolList = void 0;
18
+ const decorator_tools_1 = require("./../decorator-tools.cjs");
19
+ Object.defineProperty(exports, "decoratorTools", { enumerable: true, get: function () { return decorator_tools_1.decoratorTools; } });
20
+ const moduleRegistry_1 = require("./../moduleRegistry.cjs");
21
+ const tools_1 = require("./tools.cjs");
22
+ const codex_tools_1 = require("./codex-tools.cjs");
23
+ __exportStar(require("./tools.cjs"), exports);
24
+ __exportStar(require("./codex-tools.cjs"), exports);
25
+ __exportStar(require("./generateMcpModule.cjs"), exports);
26
+ const codexToolList = [
27
+ codex_tools_1.documentObjectTool,
28
+ codex_tools_1.coverageEnforcerTool,
29
+ codex_tools_1.readmeImprovementTool,
30
+ ];
31
+ const moduleToolList = moduleRegistry_1.moduleRegistry.listTools().map((asset) => asset.tool);
32
+ exports.toolList = [...tools_1.toolList, ...codexToolList, ...moduleToolList];
33
+ exports.decoratorToolList = Object.values(decorator_tools_1.decoratorTools);
34
+ const [analyzeRepositoryTool, enumerateCapabilitiesTool, planFeatureTool, documentCodeToolRef, applyCodeChangeToolRef,] = tools_1.toolList;
35
+ exports.tools = {
36
+ analyzeRepositoryTool,
37
+ enumerateCapabilitiesTool,
38
+ planFeatureTool,
39
+ documentCodeTool: documentCodeToolRef,
40
+ applyCodeChangeTool: applyCodeChangeToolRef,
41
+ documentObjectTool: codex_tools_1.documentObjectTool,
42
+ coverageEnforcerTool: codex_tools_1.coverageEnforcerTool,
43
+ readmeImprovementTool: codex_tools_1.readmeImprovementTool,
44
+ ...decorator_tools_1.decoratorTools,
45
+ };
46
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbWNwL3Rvb2xzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsOERBQW9EO0FBK0MzQywrRkEvQ0EsZ0NBQWMsT0ErQ0E7QUE5Q3ZCLDREQUFtRDtBQUNuRCx1Q0FJaUI7QUFDakIsbURBSXVCO0FBRXZCLDhDQUF3QjtBQUN4QixvREFBOEI7QUFDOUIsMERBQW9DO0FBRXBDLE1BQU0sYUFBYSxHQUFHO0lBQ3BCLGdDQUFrQjtJQUNsQixrQ0FBb0I7SUFDcEIsbUNBQXFCO0NBQ3RCLENBQUM7QUFFRixNQUFNLGNBQWMsR0FBRywrQkFBYyxDQUFDLFNBQVMsRUFBRSxDQUFDLEdBQUcsQ0FBQyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsS0FBSyxDQUFDLElBQVcsQ0FBQyxDQUFDO0FBRXZFLFFBQUEsUUFBUSxHQUFHLENBQUMsR0FBRyxnQkFBWSxFQUFFLEdBQUcsYUFBYSxFQUFFLEdBQUcsY0FBYyxDQUFDLENBQUM7QUFDbEUsUUFBQSxpQkFBaUIsR0FBRyxNQUFNLENBQUMsTUFBTSxDQUFDLGdDQUFjLENBQUMsQ0FBQztBQUMvRCxNQUFNLENBQ0oscUJBQXFCLEVBQ3JCLHlCQUF5QixFQUN6QixlQUFlLEVBQ2YsbUJBQW1CLEVBQ25CLHNCQUFzQixFQUN2QixHQUFHLGdCQUFZLENBQUM7QUFFSixRQUFBLEtBQUssR0FBRztJQUNuQixxQkFBcUI7SUFDckIseUJBQXlCO0lBQ3pCLGVBQWU7SUFDZixnQkFBZ0IsRUFBRSxtQkFBbUI7SUFDckMsbUJBQW1CLEVBQUUsc0JBQXNCO0lBQzNDLGtCQUFrQixFQUFsQixnQ0FBa0I7SUFDbEIsb0JBQW9CLEVBQXBCLGtDQUFvQjtJQUNwQixxQkFBcUIsRUFBckIsbUNBQXFCO0lBQ3JCLEdBQUcsZ0NBQWM7Q0FDbEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGRlY29yYXRvclRvb2xzIH0gZnJvbSBcIi4uL2RlY29yYXRvci10b29sc1wiO1xuaW1wb3J0IHsgbW9kdWxlUmVnaXN0cnkgfSBmcm9tIFwiLi4vbW9kdWxlUmVnaXN0cnlcIjtcbmltcG9ydCB7XG4gIGFwcGx5Q29kZUNoYW5nZVRvb2wsXG4gIGRvY3VtZW50Q29kZVRvb2wsXG4gIHRvb2xMaXN0IGFzIGNvcmVUb29sTGlzdCxcbn0gZnJvbSBcIi4vdG9vbHNcIjtcbmltcG9ydCB7XG4gIGNvdmVyYWdlRW5mb3JjZXJUb29sLFxuICBkb2N1bWVudE9iamVjdFRvb2wsXG4gIHJlYWRtZUltcHJvdmVtZW50VG9vbCxcbn0gZnJvbSBcIi4vY29kZXgtdG9vbHNcIjtcblxuZXhwb3J0ICogZnJvbSBcIi4vdG9vbHNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2NvZGV4LXRvb2xzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9nZW5lcmF0ZU1jcE1vZHVsZVwiO1xuXG5jb25zdCBjb2RleFRvb2xMaXN0ID0gW1xuICBkb2N1bWVudE9iamVjdFRvb2wsXG4gIGNvdmVyYWdlRW5mb3JjZXJUb29sLFxuICByZWFkbWVJbXByb3ZlbWVudFRvb2wsXG5dO1xuXG5jb25zdCBtb2R1bGVUb29sTGlzdCA9IG1vZHVsZVJlZ2lzdHJ5Lmxpc3RUb29scygpLm1hcCgoYXNzZXQpID0+IGFzc2V0LnRvb2wgYXMgYW55KTtcblxuZXhwb3J0IGNvbnN0IHRvb2xMaXN0ID0gWy4uLmNvcmVUb29sTGlzdCwgLi4uY29kZXhUb29sTGlzdCwgLi4ubW9kdWxlVG9vbExpc3RdO1xuZXhwb3J0IGNvbnN0IGRlY29yYXRvclRvb2xMaXN0ID0gT2JqZWN0LnZhbHVlcyhkZWNvcmF0b3JUb29scyk7XG5jb25zdCBbXG4gIGFuYWx5emVSZXBvc2l0b3J5VG9vbCxcbiAgZW51bWVyYXRlQ2FwYWJpbGl0aWVzVG9vbCxcbiAgcGxhbkZlYXR1cmVUb29sLFxuICBkb2N1bWVudENvZGVUb29sUmVmLFxuICBhcHBseUNvZGVDaGFuZ2VUb29sUmVmLFxuXSA9IGNvcmVUb29sTGlzdDtcblxuZXhwb3J0IGNvbnN0IHRvb2xzID0ge1xuICBhbmFseXplUmVwb3NpdG9yeVRvb2wsXG4gIGVudW1lcmF0ZUNhcGFiaWxpdGllc1Rvb2wsXG4gIHBsYW5GZWF0dXJlVG9vbCxcbiAgZG9jdW1lbnRDb2RlVG9vbDogZG9jdW1lbnRDb2RlVG9vbFJlZixcbiAgYXBwbHlDb2RlQ2hhbmdlVG9vbDogYXBwbHlDb2RlQ2hhbmdlVG9vbFJlZixcbiAgZG9jdW1lbnRPYmplY3RUb29sLFxuICBjb3ZlcmFnZUVuZm9yY2VyVG9vbCxcbiAgcmVhZG1lSW1wcm92ZW1lbnRUb29sLFxuICAuLi5kZWNvcmF0b3JUb29scyxcbn07XG5cbmV4cG9ydCB7IGRlY29yYXRvclRvb2xzIH07Il19