@decaf-ts/mcp-server 0.0.4 → 0.3.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.
- package/README.md +18 -2
- package/dist/mcp-server.cjs +1986 -340
- package/dist/mcp-server.esm.cjs +1960 -337
- package/lib/McpWrapper.cjs +9 -9
- package/lib/McpWrapper.d.ts +1 -1
- package/lib/bin/validate-modules.cjs +24 -0
- package/lib/bin/validate-modules.d.ts +2 -0
- package/lib/constants.cjs +22 -2
- package/lib/constants.d.ts +16 -0
- package/lib/esm/McpWrapper.d.ts +1 -1
- package/lib/esm/McpWrapper.js +9 -9
- package/lib/esm/bin/validate-modules.d.ts +2 -0
- package/lib/esm/bin/validate-modules.js +22 -0
- package/lib/esm/constants.d.ts +16 -0
- package/lib/esm/constants.js +21 -1
- package/lib/esm/mcp/aggregateModules.d.ts +26 -0
- package/lib/esm/mcp/aggregateModules.js +185 -0
- package/lib/esm/mcp/code.d.ts +23 -0
- package/lib/esm/mcp/code.js +70 -0
- package/lib/esm/mcp/decorator-tools.js +237 -0
- package/lib/esm/mcp/fastmcp-wiring.d.ts +14 -0
- package/lib/esm/mcp/fastmcp-wiring.js +56 -0
- package/lib/esm/mcp/index.d.ts +7 -1
- package/lib/esm/mcp/index.js +26 -2
- package/lib/esm/mcp/mcp-module.d.ts +11 -0
- package/lib/esm/mcp/mcp-module.js +31 -0
- package/lib/esm/mcp/moduleRegistry.d.ts +12 -0
- package/lib/esm/mcp/moduleRegistry.js +46 -0
- package/lib/esm/mcp/prompts/index.d.ts +4 -0
- package/lib/esm/mcp/prompts/index.js +7 -0
- package/lib/esm/mcp/prompts/prompts.d.ts +22 -0
- package/lib/esm/mcp/prompts/prompts.js +197 -0
- package/lib/esm/mcp/resources/index.d.ts +1 -0
- package/lib/esm/mcp/resources/index.js +2 -0
- package/lib/esm/mcp/resources/resources.d.ts +2 -0
- package/lib/esm/mcp/resources/resources.js +69 -0
- package/lib/esm/mcp/schemas.d.ts +53 -0
- package/lib/esm/mcp/schemas.js +97 -0
- package/lib/esm/mcp/templates/codex-templates.d.ts +3 -0
- package/lib/esm/mcp/templates/codex-templates.js +33 -0
- package/lib/esm/mcp/templates/index.d.ts +71 -0
- package/lib/esm/mcp/templates/index.js +66 -0
- package/lib/esm/mcp/templates/resource-templates.d.ts +3 -0
- package/lib/esm/mcp/templates/resource-templates.js +60 -0
- package/lib/esm/mcp/templates/workspace-templates.d.ts +3 -0
- package/lib/esm/mcp/templates/workspace-templates.js +66 -0
- package/lib/esm/mcp/tools/codex-tools.d.ts +5 -0
- package/lib/esm/mcp/tools/codex-tools.js +244 -0
- package/lib/esm/mcp/tools/generateMcpModule.d.ts +9 -0
- package/lib/esm/mcp/tools/generateMcpModule.js +133 -0
- package/lib/esm/mcp/tools/index.d.ts +321 -0
- package/lib/esm/mcp/tools/index.js +29 -0
- package/lib/esm/mcp/tools/tools.d.ts +10 -0
- package/lib/esm/mcp/tools/tools.js +273 -0
- package/lib/esm/mcp/types.d.ts +66 -0
- package/lib/esm/mcp/types.js +2 -0
- package/lib/esm/mcp/utils.d.ts +4 -0
- package/lib/esm/mcp/utils.js +46 -0
- package/lib/esm/mcp/validation/index.d.ts +13 -0
- package/lib/esm/mcp/validation/index.js +116 -0
- package/lib/esm/mcp/validation/scaffoldModule.d.ts +9 -0
- package/lib/esm/mcp/validation/scaffoldModule.js +88 -0
- package/lib/esm/mcp/workspace.d.ts +9 -0
- package/lib/esm/mcp/workspace.js +73 -0
- package/lib/esm/metadata.d.ts +1 -1
- package/lib/esm/metadata.js +1 -1
- package/lib/esm/modules/_template/index.d.ts +32 -0
- package/lib/esm/modules/_template/index.js +16 -0
- package/lib/esm/modules/_template/prompts/index.d.ts +6 -0
- package/lib/esm/modules/_template/prompts/index.js +9 -0
- package/lib/esm/modules/_template/resources/index.d.ts +6 -0
- package/lib/esm/modules/_template/resources/index.js +9 -0
- package/lib/esm/modules/_template/templates/index.d.ts +7 -0
- package/lib/esm/modules/_template/templates/index.js +10 -0
- package/lib/esm/modules/_template/tools/index.d.ts +6 -0
- package/lib/esm/modules/_template/tools/index.js +15 -0
- package/lib/esm/modules/decoration/index.d.ts +46 -0
- package/lib/esm/modules/decoration/index.js +10 -2
- package/lib/esm/modules/decoration/prompts/index.d.ts +1 -0
- package/lib/esm/modules/decoration/prompts/index.js +2 -0
- package/lib/esm/modules/decoration/resources/index.d.ts +7 -0
- package/lib/esm/modules/decoration/resources/index.js +10 -0
- package/lib/esm/modules/decoration/templates/index.d.ts +6 -0
- package/lib/esm/modules/decoration/templates/index.js +9 -0
- package/lib/esm/modules/decoration/tools/index.d.ts +26 -0
- package/lib/esm/modules/decoration/tools/index.js +7 -0
- package/lib/esm/modules/index.d.ts +2 -0
- package/lib/esm/modules/index.js +10 -0
- package/lib/esm/modules/mcp/decoration-assist.d.ts +3 -38
- package/lib/esm/modules/mcp/decoration-assist.js +5 -352
- package/lib/esm/modules/mcp/index.d.ts +6 -2
- package/lib/esm/modules/mcp/index.js +16 -3
- package/lib/esm/modules/mcp/prompts/index.d.ts +2 -0
- package/lib/esm/modules/mcp/prompts/index.js +9 -0
- package/lib/esm/modules/mcp/resources/index.d.ts +2 -0
- package/lib/esm/modules/mcp/resources/index.js +24 -0
- package/lib/esm/modules/mcp/templates/index.d.ts +2 -0
- package/lib/esm/modules/mcp/templates/index.js +28 -0
- package/lib/esm/modules/mcp/tools/index.d.ts +6 -0
- package/lib/esm/modules/mcp/tools/index.js +15 -0
- package/lib/esm/types.d.ts +41 -1
- package/lib/esm/types.js +1 -1
- package/lib/esm/utils/modulePaths.d.ts +6 -0
- package/lib/esm/utils/modulePaths.js +33 -0
- package/lib/esm/utils/moduleValidator.d.ts +14 -0
- package/lib/esm/utils/moduleValidator.js +176 -0
- package/lib/esm/utils.d.ts +1 -0
- package/lib/esm/utils.js +2 -1
- package/lib/mcp/aggregateModules.cjs +225 -0
- package/lib/mcp/aggregateModules.d.ts +26 -0
- package/lib/mcp/code.cjs +81 -0
- package/lib/mcp/code.d.ts +23 -0
- package/lib/mcp/decorator-tools.cjs +243 -0
- package/lib/mcp/fastmcp-wiring.cjs +59 -0
- package/lib/mcp/fastmcp-wiring.d.ts +14 -0
- package/lib/mcp/index.cjs +47 -12
- package/lib/mcp/index.d.ts +7 -1
- package/lib/mcp/mcp-module.cjs +53 -0
- package/lib/mcp/mcp-module.d.ts +11 -0
- package/lib/mcp/moduleRegistry.cjs +50 -0
- package/lib/mcp/moduleRegistry.d.ts +12 -0
- package/lib/mcp/prompts/index.cjs +25 -0
- package/lib/mcp/prompts/index.d.ts +4 -0
- package/lib/mcp/prompts/prompts.cjs +211 -0
- package/lib/mcp/prompts/prompts.d.ts +22 -0
- package/lib/mcp/resources/index.cjs +18 -0
- package/lib/mcp/resources/index.d.ts +1 -0
- package/lib/mcp/resources/resources.cjs +72 -0
- package/lib/mcp/resources/resources.d.ts +2 -0
- package/lib/mcp/schemas.cjs +100 -0
- package/lib/mcp/schemas.d.ts +53 -0
- package/lib/mcp/templates/codex-templates.cjs +40 -0
- package/lib/mcp/templates/codex-templates.d.ts +3 -0
- package/lib/mcp/templates/index.cjs +76 -0
- package/lib/mcp/templates/index.d.ts +71 -0
- package/lib/mcp/templates/resource-templates.cjs +67 -0
- package/lib/mcp/templates/resource-templates.d.ts +3 -0
- package/lib/mcp/templates/workspace-templates.cjs +70 -0
- package/lib/mcp/templates/workspace-templates.d.ts +3 -0
- package/lib/mcp/tools/codex-tools.cjs +250 -0
- package/lib/mcp/tools/codex-tools.d.ts +5 -0
- package/lib/mcp/tools/generateMcpModule.cjs +139 -0
- package/lib/mcp/tools/generateMcpModule.d.ts +9 -0
- package/lib/mcp/tools/index.cjs +46 -0
- package/lib/mcp/tools/index.d.ts +321 -0
- package/lib/mcp/tools/tools.cjs +282 -0
- package/lib/mcp/tools/tools.d.ts +10 -0
- package/lib/mcp/types.cjs +3 -0
- package/lib/mcp/types.d.ts +66 -0
- package/lib/mcp/utils.cjs +54 -0
- package/lib/mcp/utils.d.ts +4 -0
- package/lib/mcp/validation/index.cjs +123 -0
- package/lib/mcp/validation/index.d.ts +13 -0
- package/lib/mcp/validation/scaffoldModule.cjs +94 -0
- package/lib/mcp/validation/scaffoldModule.d.ts +9 -0
- package/lib/mcp/workspace.cjs +119 -0
- package/lib/mcp/workspace.d.ts +9 -0
- package/lib/metadata.cjs +1 -1
- package/lib/metadata.d.ts +1 -1
- package/lib/modules/_template/index.cjs +23 -0
- package/lib/modules/_template/index.d.ts +32 -0
- package/lib/modules/_template/prompts/index.cjs +12 -0
- package/lib/modules/_template/prompts/index.d.ts +6 -0
- package/lib/modules/_template/resources/index.cjs +12 -0
- package/lib/modules/_template/resources/index.d.ts +6 -0
- package/lib/modules/_template/templates/index.cjs +13 -0
- package/lib/modules/_template/templates/index.d.ts +7 -0
- package/lib/modules/_template/tools/index.cjs +18 -0
- package/lib/modules/_template/tools/index.d.ts +6 -0
- package/lib/modules/decoration/index.cjs +16 -1
- package/lib/modules/decoration/index.d.ts +46 -0
- package/lib/modules/decoration/prompts/index.cjs +5 -0
- package/lib/modules/decoration/prompts/index.d.ts +1 -0
- package/lib/modules/decoration/resources/index.cjs +13 -0
- package/lib/modules/decoration/resources/index.d.ts +7 -0
- package/lib/modules/decoration/templates/index.cjs +12 -0
- package/lib/modules/decoration/templates/index.d.ts +6 -0
- package/lib/modules/decoration/tools/index.cjs +10 -0
- package/lib/modules/decoration/tools/index.d.ts +26 -0
- package/lib/modules/index.cjs +13 -0
- package/lib/modules/index.d.ts +2 -0
- package/lib/modules/mcp/decoration-assist.cjs +8 -355
- package/lib/modules/mcp/decoration-assist.d.ts +3 -38
- package/lib/modules/mcp/index.cjs +21 -22
- package/lib/modules/mcp/index.d.ts +6 -2
- package/lib/modules/mcp/prompts/index.cjs +12 -0
- package/lib/modules/mcp/prompts/index.d.ts +2 -0
- package/lib/modules/mcp/resources/index.cjs +27 -0
- package/lib/modules/mcp/resources/index.d.ts +2 -0
- package/lib/modules/mcp/templates/index.cjs +31 -0
- package/lib/modules/mcp/templates/index.d.ts +2 -0
- package/lib/modules/mcp/tools/index.cjs +18 -0
- package/lib/modules/mcp/tools/index.d.ts +6 -0
- package/lib/types.cjs +1 -1
- package/lib/types.d.ts +41 -1
- package/lib/utils/modulePaths.cjs +43 -0
- package/lib/utils/modulePaths.d.ts +6 -0
- package/lib/utils/moduleValidator.cjs +184 -0
- package/lib/utils/moduleValidator.d.ts +14 -0
- package/lib/utils.cjs +5 -1
- package/lib/utils.d.ts +1 -0
- package/package.json +14 -13
- package/lib/esm/modules/mcp/decorator-tools.js +0 -237
- package/lib/esm/modules/mcp/mcp-module.d.ts +0 -230
- package/lib/esm/modules/mcp/mcp-module.js +0 -406
- package/lib/modules/mcp/decorator-tools.cjs +0 -243
- package/lib/modules/mcp/mcp-module.cjs +0 -452
- package/lib/modules/mcp/mcp-module.d.ts +0 -230
- /package/lib/esm/{modules/mcp → mcp}/decorator-tools.d.ts +0 -0
- /package/lib/{modules/mcp → mcp}/decorator-tools.d.ts +0 -0
|
@@ -0,0 +1,185 @@
|
|
|
1
|
+
// Aggregator: import module index files and merge exported arrays with provenance + duplicate detection
|
|
2
|
+
import path from "path";
|
|
3
|
+
import fs from "fs";
|
|
4
|
+
import { pathToFileURL } from "url";
|
|
5
|
+
import { findModuleDirs } from "./validation";
|
|
6
|
+
const SUBFOLDERS = ["prompts", "resources", "templates", "tools"];
|
|
7
|
+
const INDEX_CANDIDATES = [
|
|
8
|
+
"index.ts",
|
|
9
|
+
"index.tsx",
|
|
10
|
+
"index.js",
|
|
11
|
+
"index.cjs",
|
|
12
|
+
"index.mjs",
|
|
13
|
+
];
|
|
14
|
+
function findIndexFile(folder) {
|
|
15
|
+
for (const c of INDEX_CANDIDATES) {
|
|
16
|
+
const f = path.join(folder, c);
|
|
17
|
+
if (fs.existsSync(f))
|
|
18
|
+
return f;
|
|
19
|
+
}
|
|
20
|
+
return undefined;
|
|
21
|
+
}
|
|
22
|
+
function makeKeyForItem(item) {
|
|
23
|
+
if (!item)
|
|
24
|
+
return JSON.stringify(item);
|
|
25
|
+
if (typeof item === "string")
|
|
26
|
+
return `str:${item}`;
|
|
27
|
+
if (typeof item === "number")
|
|
28
|
+
return `num:${item}`;
|
|
29
|
+
if (item.id)
|
|
30
|
+
return `id:${item.id}`;
|
|
31
|
+
if (item.name)
|
|
32
|
+
return `name:${item.name}`;
|
|
33
|
+
// fallback to stable string
|
|
34
|
+
try {
|
|
35
|
+
return `obj:${JSON.stringify(item)}`;
|
|
36
|
+
}
|
|
37
|
+
catch (e) {
|
|
38
|
+
return `obj:${String(item)}`;
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
async function loadArrayFromIndex(filePath) {
|
|
42
|
+
// Prefer a fast, static parse of the first array literal found in the file.
|
|
43
|
+
try {
|
|
44
|
+
const content = fs.readFileSync(filePath, "utf8");
|
|
45
|
+
const start = content.indexOf("[");
|
|
46
|
+
if (start !== -1) {
|
|
47
|
+
let depth = 0;
|
|
48
|
+
let end = -1;
|
|
49
|
+
for (let i = start; i < content.length; i++) {
|
|
50
|
+
const ch = content[i];
|
|
51
|
+
if (ch === "[")
|
|
52
|
+
depth++;
|
|
53
|
+
else if (ch === "]") {
|
|
54
|
+
depth--;
|
|
55
|
+
if (depth === 0) {
|
|
56
|
+
end = i;
|
|
57
|
+
break;
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
if (end !== -1) {
|
|
62
|
+
const arrText = content.slice(start, end + 1);
|
|
63
|
+
try {
|
|
64
|
+
return JSON.parse(arrText);
|
|
65
|
+
}
|
|
66
|
+
catch (e) {
|
|
67
|
+
// Normalize TS object literals to JSON:
|
|
68
|
+
// - convert single quotes to double quotes
|
|
69
|
+
// - quote unquoted object keys
|
|
70
|
+
// - strip trailing commas
|
|
71
|
+
const normalized = arrText
|
|
72
|
+
// unify quotes in string literals
|
|
73
|
+
.replace(/'(?:\\'|[^'])*'/g, (m) => m.replace(/'/g, '"'))
|
|
74
|
+
// quote unquoted keys after { or ,
|
|
75
|
+
.replace(/([\{,]\s*)([A-Za-z_$][\w$]*)(\s*:)/g, '$1"$2"$3')
|
|
76
|
+
// remove trailing commas before ] or }
|
|
77
|
+
.replace(/,(\s*[\}\]])/g, '$1');
|
|
78
|
+
try {
|
|
79
|
+
return JSON.parse(normalized);
|
|
80
|
+
}
|
|
81
|
+
catch (e2) {
|
|
82
|
+
// fallthrough to import attempt below
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
catch (e) {
|
|
89
|
+
// ignore static parse errors and fall back to import
|
|
90
|
+
}
|
|
91
|
+
try {
|
|
92
|
+
const fileUrl = pathToFileURL(filePath).href;
|
|
93
|
+
const mod = await import(fileUrl);
|
|
94
|
+
// Find first export that is an array
|
|
95
|
+
for (const key of Object.keys(mod)) {
|
|
96
|
+
const val = mod[key];
|
|
97
|
+
if (Array.isArray(val))
|
|
98
|
+
return val;
|
|
99
|
+
}
|
|
100
|
+
// default export check
|
|
101
|
+
if (Array.isArray(mod.default))
|
|
102
|
+
return mod.default;
|
|
103
|
+
return undefined;
|
|
104
|
+
}
|
|
105
|
+
catch (err) {
|
|
106
|
+
// fallback: if import fails, try to static-parse again (already attempted) and finally return undefined
|
|
107
|
+
return undefined;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
export async function aggregateModules(repoRoot) {
|
|
111
|
+
const dirs = findModuleDirs(repoRoot);
|
|
112
|
+
const master = {
|
|
113
|
+
prompts: [],
|
|
114
|
+
resources: [],
|
|
115
|
+
templates: [],
|
|
116
|
+
tools: [],
|
|
117
|
+
conflicts: [],
|
|
118
|
+
};
|
|
119
|
+
// maps to detect duplicates per type
|
|
120
|
+
const maps = {
|
|
121
|
+
prompts: new Map(),
|
|
122
|
+
resources: new Map(),
|
|
123
|
+
templates: new Map(),
|
|
124
|
+
tools: new Map(),
|
|
125
|
+
};
|
|
126
|
+
for (const moduleDir of dirs) {
|
|
127
|
+
const moduleName = path.basename(moduleDir);
|
|
128
|
+
for (const sub of SUBFOLDERS) {
|
|
129
|
+
const folder = path.join(moduleDir, sub);
|
|
130
|
+
const indexFile = findIndexFile(folder);
|
|
131
|
+
if (!indexFile)
|
|
132
|
+
continue;
|
|
133
|
+
let arr;
|
|
134
|
+
try {
|
|
135
|
+
arr = await loadArrayFromIndex(indexFile);
|
|
136
|
+
}
|
|
137
|
+
catch (err) {
|
|
138
|
+
// skip module on import error but record as conflict-like issue
|
|
139
|
+
master.conflicts.push({
|
|
140
|
+
key: `import-error:${moduleName}:${sub}`,
|
|
141
|
+
existing: { moduleName, modulePath: moduleDir },
|
|
142
|
+
incoming: { moduleName, modulePath: moduleDir },
|
|
143
|
+
});
|
|
144
|
+
continue;
|
|
145
|
+
}
|
|
146
|
+
if (!arr || !Array.isArray(arr))
|
|
147
|
+
continue;
|
|
148
|
+
for (const item of arr) {
|
|
149
|
+
const key = makeKeyForItem(item);
|
|
150
|
+
const provenance = { moduleName, modulePath: moduleDir };
|
|
151
|
+
const map = maps[sub];
|
|
152
|
+
if (map.has(key)) {
|
|
153
|
+
// record conflict deterministically (existing vs incoming)
|
|
154
|
+
const existing = map.get(key);
|
|
155
|
+
master.conflicts.push({ key, existing, incoming: provenance });
|
|
156
|
+
// skip adding duplicate
|
|
157
|
+
continue;
|
|
158
|
+
}
|
|
159
|
+
map.set(key, provenance);
|
|
160
|
+
master[sub].push({ ...item, provenance });
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
return master;
|
|
165
|
+
}
|
|
166
|
+
// For compatibility with CommonJS call sites (not exported by ESM), provide a sync wrapper
|
|
167
|
+
export function aggregateModulesSync(repoRoot) {
|
|
168
|
+
// synchronous wrapper that runs the async function and blocks — suitable for small module sets
|
|
169
|
+
const p = aggregateModules(repoRoot);
|
|
170
|
+
let result;
|
|
171
|
+
let done = false;
|
|
172
|
+
p.then((r) => {
|
|
173
|
+
result = r;
|
|
174
|
+
done = true;
|
|
175
|
+
}).catch((e) => {
|
|
176
|
+
throw e;
|
|
177
|
+
});
|
|
178
|
+
// spin-wait (acceptable in small dev scripts)
|
|
179
|
+
const until = Date.now() + 5000;
|
|
180
|
+
while (!done && Date.now() < until) { }
|
|
181
|
+
if (!done)
|
|
182
|
+
throw new Error("aggregateModulesSync: timeout waiting for async aggregation");
|
|
183
|
+
return result;
|
|
184
|
+
}
|
|
185
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aggregateModules.js","sourceRoot":"","sources":["../../../src/mcp/aggregateModules.ts"],"names":[],"mappings":"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","sourcesContent":["// Aggregator: import module index files and merge exported arrays with provenance + duplicate detection\nimport path from \"path\";\nimport fs from \"fs\";\nimport { pathToFileURL } from \"url\";\nimport { findModuleDirs } from \"./validation\";\n\nexport type Provenance = { moduleName: string; modulePath: string };\nexport type AggregationConflict = {\n  key: string;\n  existing: Provenance;\n  incoming: Provenance;\n};\n\nexport type AggregationResult<T = any> = {\n  prompts: Array<T & { provenance: Provenance }>;\n  resources: Array<T & { provenance: Provenance }>;\n  templates: Array<T & { provenance: Provenance }>;\n  tools: Array<T & { provenance: Provenance }>;\n  conflicts: AggregationConflict[];\n};\n\nconst SUBFOLDERS = [\"prompts\", \"resources\", \"templates\", \"tools\"];\nconst INDEX_CANDIDATES = [\n  \"index.ts\",\n  \"index.tsx\",\n  \"index.js\",\n  \"index.cjs\",\n  \"index.mjs\",\n];\n\nfunction findIndexFile(folder: string): string | undefined {\n  for (const c of INDEX_CANDIDATES) {\n    const f = path.join(folder, c);\n    if (fs.existsSync(f)) return f;\n  }\n  return undefined;\n}\n\nfunction makeKeyForItem(item: any): string {\n  if (!item) return JSON.stringify(item);\n  if (typeof item === \"string\") return `str:${item}`;\n  if (typeof item === \"number\") return `num:${item}`;\n  if (item.id) return `id:${item.id}`;\n  if (item.name) return `name:${item.name}`;\n  // fallback to stable string\n  try {\n    return `obj:${JSON.stringify(item)}`;\n  } catch (e) {\n    return `obj:${String(item)}`;\n  }\n}\n\nasync function loadArrayFromIndex(\n  filePath: string\n): Promise<any[] | undefined> {\n  // Prefer a fast, static parse of the first array literal found in the file.\n  try {\n    const content = fs.readFileSync(filePath, \"utf8\");\n    const start = content.indexOf(\"[\");\n    if (start !== -1) {\n      let depth = 0;\n      let end = -1;\n      for (let i = start; i < content.length; i++) {\n        const ch = content[i];\n        if (ch === \"[\") depth++;\n        else if (ch === \"]\") {\n          depth--;\n          if (depth === 0) {\n            end = i;\n            break;\n          }\n        }\n      }\n      if (end !== -1) {\n        const arrText = content.slice(start, end + 1);\n        try {\n          return JSON.parse(arrText);\n        } catch (e) {\n          // Normalize TS object literals to JSON:\n          // - convert single quotes to double quotes\n          // - quote unquoted object keys\n          // - strip trailing commas\n          const normalized = arrText\n            // unify quotes in string literals\n            .replace(/'(?:\\\\'|[^'])*'/g, (m) => m.replace(/'/g, '\"'))\n            // quote unquoted keys after { or ,\n            .replace(/([\\{,]\\s*)([A-Za-z_$][\\w$]*)(\\s*:)/g, '$1\"$2\"$3')\n            // remove trailing commas before ] or }\n            .replace(/,(\\s*[\\}\\]])/g, '$1');\n          try {\n            return JSON.parse(normalized);\n          } catch (e2) {\n            // fallthrough to import attempt below\n          }\n        }\n      }\n    }\n  } catch (e) {\n    // ignore static parse errors and fall back to import\n  }\n\n  try {\n    const fileUrl = pathToFileURL(filePath).href;\n    const mod = await import(fileUrl);\n    // Find first export that is an array\n    for (const key of Object.keys(mod)) {\n      const val = (mod as any)[key];\n      if (Array.isArray(val)) return val;\n    }\n    // default export check\n    if (Array.isArray((mod as any).default)) return (mod as any).default;\n    return undefined;\n  } catch (err) {\n    // fallback: if import fails, try to static-parse again (already attempted) and finally return undefined\n    return undefined;\n  }\n}\n\nexport async function aggregateModules(\n  repoRoot: string\n): Promise<AggregationResult> {\n  const dirs = findModuleDirs(repoRoot);\n  const master = {\n    prompts: [] as any[],\n    resources: [] as any[],\n    templates: [] as any[],\n    tools: [] as any[],\n    conflicts: [] as AggregationConflict[],\n  };\n\n  // maps to detect duplicates per type\n  const maps: Record<string, Map<string, Provenance>> = {\n    prompts: new Map(),\n    resources: new Map(),\n    templates: new Map(),\n    tools: new Map(),\n  };\n\n  for (const moduleDir of dirs) {\n    const moduleName = path.basename(moduleDir);\n    for (const sub of SUBFOLDERS) {\n      const folder = path.join(moduleDir, sub);\n      const indexFile = findIndexFile(folder);\n      if (!indexFile) continue;\n      let arr: any[] | undefined;\n      try {\n        arr = await loadArrayFromIndex(indexFile);\n      } catch (err: any) {\n        // skip module on import error but record as conflict-like issue\n        master.conflicts.push({\n          key: `import-error:${moduleName}:${sub}`,\n          existing: { moduleName, modulePath: moduleDir },\n          incoming: { moduleName, modulePath: moduleDir },\n        });\n        continue;\n      }\n      if (!arr || !Array.isArray(arr)) continue;\n\n      for (const item of arr) {\n        const key = makeKeyForItem(item);\n        const provenance = { moduleName, modulePath: moduleDir };\n        const map = maps[sub];\n        if (map.has(key)) {\n          // record conflict deterministically (existing vs incoming)\n          const existing = map.get(key)!;\n          master.conflicts.push({ key, existing, incoming: provenance });\n          // skip adding duplicate\n          continue;\n        }\n        map.set(key, provenance);\n        (master as any)[sub].push({ ...item, provenance });\n      }\n    }\n  }\n\n  return master;\n}\n\n// For compatibility with CommonJS call sites (not exported by ESM), provide a sync wrapper\nexport function aggregateModulesSync(repoRoot: string) {\n  // synchronous wrapper that runs the async function and blocks — suitable for small module sets\n  const p = aggregateModules(repoRoot);\n  let result: any;\n  let done = false;\n  p.then((r) => {\n    result = r;\n    done = true;\n  }).catch((e) => {\n    throw e;\n  });\n  // spin-wait (acceptable in small dev scripts)\n  const until = Date.now() + 5000;\n  while (!done && Date.now() < until) {}\n  if (!done)\n    throw new Error(\n      \"aggregateModulesSync: timeout waiting for async aggregation\"\n    );\n  return result as AggregationResult;\n}\n"]}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
export declare function isSourceFile(p: string): boolean;
|
|
2
|
+
export declare function isTestFile(p: string): boolean;
|
|
3
|
+
export declare function extractExports(fileContent: string): string[];
|
|
4
|
+
export declare function extractDecorators(fileContent: string): string[];
|
|
5
|
+
export declare function summarizeReadme(readme?: string): {
|
|
6
|
+
title: string;
|
|
7
|
+
bullets: string[];
|
|
8
|
+
} | undefined;
|
|
9
|
+
export declare function analyzeRepo(root: string): {
|
|
10
|
+
files: string[];
|
|
11
|
+
testFiles: string[];
|
|
12
|
+
api: Record<string, {
|
|
13
|
+
exports: string[];
|
|
14
|
+
decorators: string[];
|
|
15
|
+
}>;
|
|
16
|
+
tests: Record<string, {
|
|
17
|
+
mentions: string[];
|
|
18
|
+
}>;
|
|
19
|
+
readme: {
|
|
20
|
+
title: string;
|
|
21
|
+
bullets: string[];
|
|
22
|
+
} | undefined;
|
|
23
|
+
};
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
// Analysis helpers (minimal yet effective, text-based to avoid heavy AST deps)
|
|
2
|
+
import path from "path";
|
|
3
|
+
import fs from "fs";
|
|
4
|
+
import { listFilesRecursive, readFileSafe } from "./utils";
|
|
5
|
+
export function isSourceFile(p) {
|
|
6
|
+
return /\.(ts|tsx|js|jsx)$/.test(p) && !p.endsWith(".d.ts");
|
|
7
|
+
}
|
|
8
|
+
export function isTestFile(p) {
|
|
9
|
+
return /(\.test\.|\.spec\.)/.test(p);
|
|
10
|
+
}
|
|
11
|
+
export function extractExports(fileContent) {
|
|
12
|
+
const names = new Set();
|
|
13
|
+
const exportRe = /(export\s+(?:default\s+)?(?:class|function|const|let|var|interface|type|enum)\s+)([A-Za-z0-9_]+)/g;
|
|
14
|
+
const namedRe = /export\s*\{([^}]+)\}/g;
|
|
15
|
+
let m;
|
|
16
|
+
while ((m = exportRe.exec(fileContent)))
|
|
17
|
+
names.add(m[2]);
|
|
18
|
+
while ((m = namedRe.exec(fileContent))) {
|
|
19
|
+
m[1]
|
|
20
|
+
.split(",")
|
|
21
|
+
.map((s) => s.trim().split(" as ")[0].trim())
|
|
22
|
+
.forEach((n) => {
|
|
23
|
+
if (n)
|
|
24
|
+
names.add(n);
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
return [...names].sort();
|
|
28
|
+
}
|
|
29
|
+
export function extractDecorators(fileContent) {
|
|
30
|
+
const decs = new Set();
|
|
31
|
+
const decRe = /@([A-Za-z_][A-Za-z0-9_]*)/g;
|
|
32
|
+
let m;
|
|
33
|
+
while ((m = decRe.exec(fileContent)))
|
|
34
|
+
decs.add(m[1]);
|
|
35
|
+
return [...decs].sort();
|
|
36
|
+
}
|
|
37
|
+
export function summarizeReadme(readme) {
|
|
38
|
+
if (!readme)
|
|
39
|
+
return undefined;
|
|
40
|
+
const lines = readme.split(/\r?\n/).filter(Boolean);
|
|
41
|
+
const title = lines.find((l) => /^#\s+/.test(l))?.replace(/^#\s+/, "") || "README";
|
|
42
|
+
const bullets = lines.filter((l) => /^[-*]\s+/.test(l)).slice(0, 20);
|
|
43
|
+
return { title, bullets };
|
|
44
|
+
}
|
|
45
|
+
export function analyzeRepo(root) {
|
|
46
|
+
const src = path.join(root, "src");
|
|
47
|
+
const testDir = path.join(root, "tests");
|
|
48
|
+
const readmePath = path.join(root, "README.md");
|
|
49
|
+
const readme = readFileSafe(readmePath);
|
|
50
|
+
const files = fs.existsSync(src) ? listFilesRecursive(src, isSourceFile) : [];
|
|
51
|
+
const testFiles = fs.existsSync(testDir)
|
|
52
|
+
? listFilesRecursive(testDir, (f) => isSourceFile(f) && isTestFile(f))
|
|
53
|
+
: [];
|
|
54
|
+
const api = {};
|
|
55
|
+
for (const f of files) {
|
|
56
|
+
const content = readFileSafe(f) || "";
|
|
57
|
+
api[path.relative(root, f)] = {
|
|
58
|
+
exports: extractExports(content),
|
|
59
|
+
decorators: extractDecorators(content),
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
const tests = {};
|
|
63
|
+
for (const f of testFiles) {
|
|
64
|
+
const content = readFileSafe(f) || "";
|
|
65
|
+
const mentions = Array.from(new Set([...extractExports(content), ...extractDecorators(content)])).sort();
|
|
66
|
+
tests[path.relative(root, f)] = { mentions };
|
|
67
|
+
}
|
|
68
|
+
return { files, testFiles, api, tests, readme: summarizeReadme(readme) };
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"code.js","sourceRoot":"","sources":["../../../src/mcp/code.ts"],"names":[],"mappings":"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","sourcesContent":["// Analysis helpers (minimal yet effective, text-based to avoid heavy AST deps)\nimport path from \"path\";\nimport fs from \"fs\";\nimport { listFilesRecursive, readFileSafe } from \"./utils\";\n\nexport function isSourceFile(p: string) {\n  return /\\.(ts|tsx|js|jsx)$/.test(p) && !p.endsWith(\".d.ts\");\n}\nexport function isTestFile(p: string) {\n  return /(\\.test\\.|\\.spec\\.)/.test(p);\n}\n\nexport function 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\nexport function 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\nexport function 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\nexport function 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"]}
|
|
@@ -0,0 +1,237 @@
|
|
|
1
|
+
import fs from "fs";
|
|
2
|
+
import path from "path";
|
|
3
|
+
function escapeRegExp(value) {
|
|
4
|
+
return value.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
|
5
|
+
}
|
|
6
|
+
function formatDecorator(spec) {
|
|
7
|
+
const args = (spec.args || []).map((arg) => JSON.stringify(arg)).join(", ");
|
|
8
|
+
return `@${spec.name}(${args})`;
|
|
9
|
+
}
|
|
10
|
+
function ensureDirectory(filePath) {
|
|
11
|
+
fs.mkdirSync(path.dirname(filePath), { recursive: true });
|
|
12
|
+
}
|
|
13
|
+
function collectDecoratorNames(classDecorators, properties) {
|
|
14
|
+
const names = new Set();
|
|
15
|
+
names.add("model");
|
|
16
|
+
for (const decorator of classDecorators || []) {
|
|
17
|
+
names.add(decorator.name);
|
|
18
|
+
}
|
|
19
|
+
for (const property of properties || []) {
|
|
20
|
+
for (const decorator of property.decorators || []) {
|
|
21
|
+
names.add(decorator.name);
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
return names;
|
|
25
|
+
}
|
|
26
|
+
function ensureImport(content, importsFrom, decorators) {
|
|
27
|
+
/* istanbul ignore next */
|
|
28
|
+
if (!decorators.size)
|
|
29
|
+
return content;
|
|
30
|
+
const importRegex = new RegExp(`import\\s+\\{([^}]+)\\}\\s+from\\s+["']${escapeRegExp(importsFrom)}["'];`);
|
|
31
|
+
const match = content.match(importRegex);
|
|
32
|
+
const sorted = Array.from(decorators).sort();
|
|
33
|
+
if (match) {
|
|
34
|
+
const existing = match[1]
|
|
35
|
+
.split(",")
|
|
36
|
+
.map((name) => name.trim())
|
|
37
|
+
.filter(Boolean);
|
|
38
|
+
const merged = Array.from(new Set([...existing, ...sorted])).sort();
|
|
39
|
+
return content.replace(importRegex, `import { ${merged.join(", ")} } from "${importsFrom}";`);
|
|
40
|
+
}
|
|
41
|
+
const importLine = `import { ${sorted.join(", ")} } from "${importsFrom}";`;
|
|
42
|
+
return `${importLine}\n\n${content}`;
|
|
43
|
+
}
|
|
44
|
+
function addPropertyBlock(property) {
|
|
45
|
+
const decorators = (property.decorators || [])
|
|
46
|
+
.map(formatDecorator)
|
|
47
|
+
.join("\n ");
|
|
48
|
+
const decoratorBlock = decorators ? ` ${decorators}\n` : "";
|
|
49
|
+
return `${decoratorBlock} ${property.name}: ${property.type};`;
|
|
50
|
+
}
|
|
51
|
+
function removePropertyBlock(content, propertyName) {
|
|
52
|
+
const lines = content.split(/\r?\n/);
|
|
53
|
+
const result = [];
|
|
54
|
+
for (let i = 0; i < lines.length; i++) {
|
|
55
|
+
const line = lines[i];
|
|
56
|
+
if (line.trim().startsWith(`@`) &&
|
|
57
|
+
lines[i + 1]?.includes(`${propertyName}:`)) {
|
|
58
|
+
continue;
|
|
59
|
+
}
|
|
60
|
+
if (line.includes(`${propertyName}:`)) {
|
|
61
|
+
// skip this line and any trailing blank line
|
|
62
|
+
continue;
|
|
63
|
+
}
|
|
64
|
+
result.push(line);
|
|
65
|
+
}
|
|
66
|
+
return result.join("\n");
|
|
67
|
+
}
|
|
68
|
+
function insertDecorator(content, decorator, target) {
|
|
69
|
+
const decoratorLine = formatDecorator(decorator);
|
|
70
|
+
if (target.kind === "class") {
|
|
71
|
+
const classRegex = /(export\s+class\s+[^\s{]+\s*\{)/;
|
|
72
|
+
if (content.includes(decoratorLine))
|
|
73
|
+
return content;
|
|
74
|
+
return content.replace(classRegex, `${decoratorLine}\n$1`);
|
|
75
|
+
}
|
|
76
|
+
if (!target.name)
|
|
77
|
+
return content;
|
|
78
|
+
const propertyRegex = new RegExp(`(^\\s*)(?:@.*\\n\\1)*${escapeRegExp(target.name)}:`, "m");
|
|
79
|
+
const match = propertyRegex.exec(content);
|
|
80
|
+
if (!match)
|
|
81
|
+
return content;
|
|
82
|
+
const indent = match[1] || " ";
|
|
83
|
+
if (content.includes(`${indent}${decoratorLine}`))
|
|
84
|
+
return content;
|
|
85
|
+
return (content.slice(0, match.index) +
|
|
86
|
+
`${indent}${decoratorLine}\n` +
|
|
87
|
+
content.slice(match.index));
|
|
88
|
+
}
|
|
89
|
+
function removeDecorator(content, decoratorName, target) {
|
|
90
|
+
const decoratorRegex = new RegExp(`^\\s*@${escapeRegExp(decoratorName)}\\([^)]*\\)`, "m");
|
|
91
|
+
if (target.kind === "class") {
|
|
92
|
+
return content.replace(decoratorRegex, "");
|
|
93
|
+
}
|
|
94
|
+
if (target.name) {
|
|
95
|
+
const pattern = new RegExp(`(^\\s*@${escapeRegExp(decoratorName)}\\([^)]*\\)\\s*$\\n)(?=\\s*${escapeRegExp(target.name)}:)`, "m");
|
|
96
|
+
return content.replace(pattern, "");
|
|
97
|
+
}
|
|
98
|
+
return content;
|
|
99
|
+
}
|
|
100
|
+
function writeIfChanged(filePath, content) {
|
|
101
|
+
ensureDirectory(filePath);
|
|
102
|
+
fs.writeFileSync(filePath, content);
|
|
103
|
+
}
|
|
104
|
+
export const decoratorTools = {
|
|
105
|
+
createOrUpdateModelTool: {
|
|
106
|
+
name: "create-or-update-model",
|
|
107
|
+
description: "Create or update a validation-ready model class",
|
|
108
|
+
execute: async (args) => {
|
|
109
|
+
if (!args.overwrite && fs.existsSync(args.filePath)) {
|
|
110
|
+
throw new Error(`File already exists at ${args.filePath}`);
|
|
111
|
+
}
|
|
112
|
+
const decorators = collectDecoratorNames(args.classDecorators, args.properties);
|
|
113
|
+
let content = `@model()`;
|
|
114
|
+
for (const decorator of args.classDecorators || []) {
|
|
115
|
+
content += `\n${formatDecorator(decorator)}`;
|
|
116
|
+
}
|
|
117
|
+
const properties = (args.properties || [])
|
|
118
|
+
.map(addPropertyBlock)
|
|
119
|
+
.join("\n\n");
|
|
120
|
+
content += `\nexport class ${args.className} {\n${properties ? `${properties}\n` : ""}}\n`;
|
|
121
|
+
content = ensureImport(content, args.importsFrom, decorators);
|
|
122
|
+
writeIfChanged(args.filePath, content);
|
|
123
|
+
return { filePath: args.filePath };
|
|
124
|
+
},
|
|
125
|
+
},
|
|
126
|
+
addAttributeTool: {
|
|
127
|
+
name: "add-attribute",
|
|
128
|
+
description: "Add a decorated attribute to an existing model",
|
|
129
|
+
execute: async (args) => {
|
|
130
|
+
if (!fs.existsSync(args.filePath)) {
|
|
131
|
+
throw new Error(`Model file not found at ${args.filePath}`);
|
|
132
|
+
}
|
|
133
|
+
let content = fs.readFileSync(args.filePath, "utf8");
|
|
134
|
+
if (content.includes(`${args.attribute.name}:`)) {
|
|
135
|
+
return { filePath: args.filePath };
|
|
136
|
+
}
|
|
137
|
+
const decorators = collectDecoratorNames(undefined, [args.attribute]);
|
|
138
|
+
content = ensureImport(content, args.importsFrom, decorators);
|
|
139
|
+
const insertionPoint = content.lastIndexOf("}");
|
|
140
|
+
const block = addPropertyBlock(args.attribute);
|
|
141
|
+
const before = content.slice(0, insertionPoint).replace(/\s*$/, "");
|
|
142
|
+
const after = content.slice(insertionPoint);
|
|
143
|
+
const updated = `${before}\n${block}\n${after}`;
|
|
144
|
+
writeIfChanged(args.filePath, updated);
|
|
145
|
+
return { filePath: args.filePath };
|
|
146
|
+
},
|
|
147
|
+
},
|
|
148
|
+
removeAttributeTool: {
|
|
149
|
+
name: "remove-attribute",
|
|
150
|
+
description: "Remove an attribute from a model class",
|
|
151
|
+
execute: async (args) => {
|
|
152
|
+
if (!fs.existsSync(args.filePath))
|
|
153
|
+
return { filePath: args.filePath };
|
|
154
|
+
const content = fs.readFileSync(args.filePath, "utf8");
|
|
155
|
+
const updated = removePropertyBlock(content, args.attributeName);
|
|
156
|
+
writeIfChanged(args.filePath, updated);
|
|
157
|
+
return { filePath: args.filePath };
|
|
158
|
+
},
|
|
159
|
+
},
|
|
160
|
+
applyDecoratorTool: {
|
|
161
|
+
name: "apply-decorator",
|
|
162
|
+
description: "Apply a decorator to a class or property",
|
|
163
|
+
execute: async (args) => {
|
|
164
|
+
if (!fs.existsSync(args.filePath)) {
|
|
165
|
+
throw new Error(`Model file not found at ${args.filePath}`);
|
|
166
|
+
}
|
|
167
|
+
let content = fs.readFileSync(args.filePath, "utf8");
|
|
168
|
+
const decorators = new Set([args.decorator.name]);
|
|
169
|
+
content = ensureImport(content, args.importsFrom, decorators);
|
|
170
|
+
content = insertDecorator(content, args.decorator, args.target);
|
|
171
|
+
writeIfChanged(args.filePath, content);
|
|
172
|
+
return { filePath: args.filePath };
|
|
173
|
+
},
|
|
174
|
+
},
|
|
175
|
+
removeDecoratorTool: {
|
|
176
|
+
name: "remove-decorator",
|
|
177
|
+
description: "Remove a decorator from a class or property",
|
|
178
|
+
execute: async (args) => {
|
|
179
|
+
if (!fs.existsSync(args.filePath))
|
|
180
|
+
return { filePath: args.filePath };
|
|
181
|
+
let content = fs.readFileSync(args.filePath, "utf8");
|
|
182
|
+
content = removeDecorator(content, args.decoratorName, args.target);
|
|
183
|
+
writeIfChanged(args.filePath, content);
|
|
184
|
+
return { filePath: args.filePath };
|
|
185
|
+
},
|
|
186
|
+
},
|
|
187
|
+
scaffoldValidatorTool: {
|
|
188
|
+
name: "scaffold-validator",
|
|
189
|
+
description: "Scaffold a validator class and optional decorator",
|
|
190
|
+
execute: async (args) => {
|
|
191
|
+
const classFile = path.join(args.validatorsDir, `${args.name}.ts`);
|
|
192
|
+
ensureDirectory(classFile);
|
|
193
|
+
const classContent = `export class ${args.name} {\n validate(value: unknown): boolean {\n return value !== undefined;\n }\n}\n`;
|
|
194
|
+
fs.writeFileSync(classFile, classContent);
|
|
195
|
+
let decoratorFile;
|
|
196
|
+
if (args.decoratorDir) {
|
|
197
|
+
decoratorFile = path.join(args.decoratorDir, `${args.name}Decorator.ts`);
|
|
198
|
+
ensureDirectory(decoratorFile);
|
|
199
|
+
fs.writeFileSync(decoratorFile, `export function ${args.name}Decorator() {\n return () => void 0;\n}\n`);
|
|
200
|
+
}
|
|
201
|
+
return { classFile, decoratorFile };
|
|
202
|
+
},
|
|
203
|
+
},
|
|
204
|
+
scaffoldSerializerTool: {
|
|
205
|
+
name: "scaffold-serializer",
|
|
206
|
+
description: "Scaffold a serializer class and optional registry",
|
|
207
|
+
execute: async (args) => {
|
|
208
|
+
const classFile = path.join(args.dir, `${args.name}.ts`);
|
|
209
|
+
ensureDirectory(classFile);
|
|
210
|
+
fs.writeFileSync(classFile, `export class ${args.name} {\n serialize(value: unknown): string {\n return JSON.stringify(value);\n }\n}\n`);
|
|
211
|
+
let registerFile;
|
|
212
|
+
if (args.registerDir) {
|
|
213
|
+
registerFile = path.join(args.registerDir, `${args.name}Register.ts`);
|
|
214
|
+
ensureDirectory(registerFile);
|
|
215
|
+
fs.writeFileSync(registerFile, `export function register${args.name}() {\n return ${args.setDefault ? "'default'" : "'optional'"};\n}\n`);
|
|
216
|
+
}
|
|
217
|
+
return { classFile, registerFile };
|
|
218
|
+
},
|
|
219
|
+
},
|
|
220
|
+
scaffoldHashingTool: {
|
|
221
|
+
name: "scaffold-hashing",
|
|
222
|
+
description: "Scaffold a hashing function and optional registry",
|
|
223
|
+
execute: async (args) => {
|
|
224
|
+
const functionFile = path.join(args.dir, `${args.name}.ts`);
|
|
225
|
+
ensureDirectory(functionFile);
|
|
226
|
+
fs.writeFileSync(functionFile, `export function ${args.name}(value: string): string {\n return value.split('').reverse().join('');\n}\n`);
|
|
227
|
+
let registerFile;
|
|
228
|
+
if (args.registerDir) {
|
|
229
|
+
registerFile = path.join(args.registerDir, `${args.name}Register.ts`);
|
|
230
|
+
ensureDirectory(registerFile);
|
|
231
|
+
fs.writeFileSync(registerFile, `export function register${args.name}() {\n return ${args.setDefault ? "'default'" : "'optional'"};\n}\n`);
|
|
232
|
+
}
|
|
233
|
+
return { functionFile, registerFile };
|
|
234
|
+
},
|
|
235
|
+
},
|
|
236
|
+
};
|
|
237
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorator-tools.js","sourceRoot":"","sources":["../../../src/mcp/decorator-tools.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from \"fs\";\nimport path from \"path\";\n\nexport type DecoratorSpec = {\n  name: string;\n  args?: unknown[];\n};\n\nexport type AttributeSpec = {\n  name: string;\n  type: string;\n  decorators?: DecoratorSpec[];\n};\n\nfunction escapeRegExp(value: string) {\n  return value.replace(/[.*+?^${}()|[\\]\\\\]/g, \"\\\\$&\");\n}\n\nfunction formatDecorator(spec: DecoratorSpec): string {\n  const args = (spec.args || []).map((arg) => JSON.stringify(arg)).join(\", \");\n  return `@${spec.name}(${args})`;\n}\n\nfunction ensureDirectory(filePath: string) {\n  fs.mkdirSync(path.dirname(filePath), { recursive: true });\n}\n\nfunction collectDecoratorNames(\n  classDecorators: DecoratorSpec[] | undefined,\n  properties: AttributeSpec[] | undefined\n) {\n  const names = new Set<string>();\n  names.add(\"model\");\n  for (const decorator of classDecorators || []) {\n    names.add(decorator.name);\n  }\n  for (const property of properties || []) {\n    for (const decorator of property.decorators || []) {\n      names.add(decorator.name);\n    }\n  }\n  return names;\n}\n\nfunction ensureImport(\n  content: string,\n  importsFrom: string,\n  decorators: Set<string>\n) {\n  /* istanbul ignore next */\n  if (!decorators.size) return content;\n  const importRegex = new RegExp(\n    `import\\\\s+\\\\{([^}]+)\\\\}\\\\s+from\\\\s+[\"']${escapeRegExp(importsFrom)}[\"'];`\n  );\n  const match = content.match(importRegex);\n  const sorted = Array.from(decorators).sort();\n\n  if (match) {\n    const existing = match[1]\n      .split(\",\")\n      .map((name) => name.trim())\n      .filter(Boolean);\n    const merged = Array.from(new Set([...existing, ...sorted])).sort();\n    return content.replace(\n      importRegex,\n      `import { ${merged.join(\", \")} } from \"${importsFrom}\";`\n    );\n  }\n\n  const importLine = `import { ${sorted.join(\", \")} } from \"${importsFrom}\";`;\n  return `${importLine}\\n\\n${content}`;\n}\n\nfunction addPropertyBlock(property: AttributeSpec) {\n  const decorators = (property.decorators || [])\n    .map(formatDecorator)\n    .join(\"\\n  \");\n  const decoratorBlock = decorators ? `  ${decorators}\\n` : \"\";\n  return `${decoratorBlock}  ${property.name}: ${property.type};`;\n}\n\nfunction removePropertyBlock(content: string, propertyName: string) {\n  const lines = content.split(/\\r?\\n/);\n  const result: string[] = [];\n  for (let i = 0; i < lines.length; i++) {\n    const line = lines[i];\n    if (\n      line.trim().startsWith(`@`) &&\n      lines[i + 1]?.includes(`${propertyName}:`)\n    ) {\n      continue;\n    }\n    if (line.includes(`${propertyName}:`)) {\n      // skip this line and any trailing blank line\n      continue;\n    }\n    result.push(line);\n  }\n  return result.join(\"\\n\");\n}\n\nfunction insertDecorator(\n  content: string,\n  decorator: DecoratorSpec,\n  target: {\n    kind: \"class\" | \"property\";\n    name?: string;\n  }\n) {\n  const decoratorLine = formatDecorator(decorator);\n  if (target.kind === \"class\") {\n    const classRegex = /(export\\s+class\\s+[^\\s{]+\\s*\\{)/;\n    if (content.includes(decoratorLine)) return content;\n    return content.replace(classRegex, `${decoratorLine}\\n$1`);\n  }\n  if (!target.name) return content;\n  const propertyRegex = new RegExp(\n    `(^\\\\s*)(?:@.*\\\\n\\\\1)*${escapeRegExp(target.name)}:`,\n    \"m\"\n  );\n  const match = propertyRegex.exec(content);\n  if (!match) return content;\n  const indent = match[1] || \"  \";\n  if (content.includes(`${indent}${decoratorLine}`)) return content;\n  return (\n    content.slice(0, match.index) +\n    `${indent}${decoratorLine}\\n` +\n    content.slice(match.index)\n  );\n}\n\nfunction removeDecorator(\n  content: string,\n  decoratorName: string,\n  target: {\n    kind: \"class\" | \"property\";\n    name?: string;\n  }\n) {\n  const decoratorRegex = new RegExp(\n    `^\\\\s*@${escapeRegExp(decoratorName)}\\\\([^)]*\\\\)`,\n    \"m\"\n  );\n  if (target.kind === \"class\") {\n    return content.replace(decoratorRegex, \"\");\n  }\n  if (target.name) {\n    const pattern = new RegExp(\n      `(^\\\\s*@${escapeRegExp(decoratorName)}\\\\([^)]*\\\\)\\\\s*$\\\\n)(?=\\\\s*${escapeRegExp(target.name)}:)`,\n      \"m\"\n    );\n    return content.replace(pattern, \"\");\n  }\n  return content;\n}\n\nfunction writeIfChanged(filePath: string, content: string) {\n  ensureDirectory(filePath);\n  fs.writeFileSync(filePath, content);\n}\n\nexport const decoratorTools = {\n  createOrUpdateModelTool: {\n    name: \"create-or-update-model\",\n    description: \"Create or update a validation-ready model class\",\n    execute: async (args: {\n      filePath: string;\n      className: string;\n      classDecorators?: DecoratorSpec[];\n      properties: AttributeSpec[];\n      importsFrom: string;\n      overwrite?: boolean;\n    }) => {\n      if (!args.overwrite && fs.existsSync(args.filePath)) {\n        throw new Error(`File already exists at ${args.filePath}`);\n      }\n      const decorators = collectDecoratorNames(\n        args.classDecorators,\n        args.properties\n      );\n      let content = `@model()`;\n      for (const decorator of args.classDecorators || []) {\n        content += `\\n${formatDecorator(decorator)}`;\n      }\n      const properties = (args.properties || [])\n        .map(addPropertyBlock)\n        .join(\"\\n\\n\");\n      content += `\\nexport class ${args.className} {\\n${properties ? `${properties}\\n` : \"\"}}\\n`;\n      content = ensureImport(content, args.importsFrom, decorators);\n      writeIfChanged(args.filePath, content);\n      return { filePath: args.filePath };\n    },\n  },\n  addAttributeTool: {\n    name: \"add-attribute\",\n    description: \"Add a decorated attribute to an existing model\",\n    execute: async (args: {\n      filePath: string;\n      className: string;\n      attribute: AttributeSpec;\n      importsFrom: string;\n    }) => {\n      if (!fs.existsSync(args.filePath)) {\n        throw new Error(`Model file not found at ${args.filePath}`);\n      }\n      let content = fs.readFileSync(args.filePath, \"utf8\");\n      if (content.includes(`${args.attribute.name}:`)) {\n        return { filePath: args.filePath };\n      }\n      const decorators = collectDecoratorNames(undefined, [args.attribute]);\n      content = ensureImport(content, args.importsFrom, decorators);\n      const insertionPoint = content.lastIndexOf(\"}\");\n      const block = addPropertyBlock(args.attribute);\n      const before = content.slice(0, insertionPoint).replace(/\\s*$/, \"\");\n      const after = content.slice(insertionPoint);\n      const updated = `${before}\\n${block}\\n${after}`;\n      writeIfChanged(args.filePath, updated);\n      return { filePath: args.filePath };\n    },\n  },\n  removeAttributeTool: {\n    name: \"remove-attribute\",\n    description: \"Remove an attribute from a model class\",\n    execute: async (args: {\n      filePath: string;\n      className: string;\n      attributeName: string;\n    }) => {\n      if (!fs.existsSync(args.filePath)) return { filePath: args.filePath };\n      const content = fs.readFileSync(args.filePath, \"utf8\");\n      const updated = removePropertyBlock(content, args.attributeName);\n      writeIfChanged(args.filePath, updated);\n      return { filePath: args.filePath };\n    },\n  },\n  applyDecoratorTool: {\n    name: \"apply-decorator\",\n    description: \"Apply a decorator to a class or property\",\n    execute: async (args: {\n      filePath: string;\n      className: string;\n      target: { kind: \"class\" | \"property\"; name?: string };\n      decorator: DecoratorSpec;\n      importsFrom: string;\n    }) => {\n      if (!fs.existsSync(args.filePath)) {\n        throw new Error(`Model file not found at ${args.filePath}`);\n      }\n      let content = fs.readFileSync(args.filePath, \"utf8\");\n      const decorators = new Set<string>([args.decorator.name]);\n      content = ensureImport(content, args.importsFrom, decorators);\n      content = insertDecorator(content, args.decorator, args.target);\n      writeIfChanged(args.filePath, content);\n      return { filePath: args.filePath };\n    },\n  },\n  removeDecoratorTool: {\n    name: \"remove-decorator\",\n    description: \"Remove a decorator from a class or property\",\n    execute: async (args: {\n      filePath: string;\n      className: string;\n      target: { kind: \"class\" | \"property\"; name?: string };\n      decoratorName: string;\n    }) => {\n      if (!fs.existsSync(args.filePath)) return { filePath: args.filePath };\n      let content = fs.readFileSync(args.filePath, \"utf8\");\n      content = removeDecorator(content, args.decoratorName, args.target);\n      writeIfChanged(args.filePath, content);\n      return { filePath: args.filePath };\n    },\n  },\n  scaffoldValidatorTool: {\n    name: \"scaffold-validator\",\n    description: \"Scaffold a validator class and optional decorator\",\n    execute: async (args: {\n      validatorsDir: string;\n      decoratorDir?: string;\n      name: string;\n    }) => {\n      const classFile = path.join(args.validatorsDir, `${args.name}.ts`);\n      ensureDirectory(classFile);\n      const classContent = `export class ${args.name} {\\n  validate(value: unknown): boolean {\\n    return value !== undefined;\\n  }\\n}\\n`;\n      fs.writeFileSync(classFile, classContent);\n      let decoratorFile: string | undefined;\n      if (args.decoratorDir) {\n        decoratorFile = path.join(\n          args.decoratorDir,\n          `${args.name}Decorator.ts`\n        );\n        ensureDirectory(decoratorFile);\n        fs.writeFileSync(\n          decoratorFile,\n          `export function ${args.name}Decorator() {\\n  return () => void 0;\\n}\\n`\n        );\n      }\n      return { classFile, decoratorFile };\n    },\n  },\n  scaffoldSerializerTool: {\n    name: \"scaffold-serializer\",\n    description: \"Scaffold a serializer class and optional registry\",\n    execute: async (args: {\n      dir: string;\n      name: string;\n      registerDir?: string;\n      setDefault?: boolean;\n    }) => {\n      const classFile = path.join(args.dir, `${args.name}.ts`);\n      ensureDirectory(classFile);\n      fs.writeFileSync(\n        classFile,\n        `export class ${args.name} {\\n  serialize(value: unknown): string {\\n    return JSON.stringify(value);\\n  }\\n}\\n`\n      );\n      let registerFile: string | undefined;\n      if (args.registerDir) {\n        registerFile = path.join(args.registerDir, `${args.name}Register.ts`);\n        ensureDirectory(registerFile);\n        fs.writeFileSync(\n          registerFile,\n          `export function register${args.name}() {\\n  return ${args.setDefault ? \"'default'\" : \"'optional'\"};\\n}\\n`\n        );\n      }\n      return { classFile, registerFile };\n    },\n  },\n  scaffoldHashingTool: {\n    name: \"scaffold-hashing\",\n    description: \"Scaffold a hashing function and optional registry\",\n    execute: async (args: {\n      dir: string;\n      name: string;\n      registerDir?: string;\n      setDefault?: boolean;\n    }) => {\n      const functionFile = path.join(args.dir, `${args.name}.ts`);\n      ensureDirectory(functionFile);\n      fs.writeFileSync(\n        functionFile,\n        `export function ${args.name}(value: string): string {\\n  return value.split('').reverse().join('');\\n}\\n`\n      );\n      let registerFile: string | undefined;\n      if (args.registerDir) {\n        registerFile = path.join(args.registerDir, `${args.name}Register.ts`);\n        ensureDirectory(registerFile);\n        fs.writeFileSync(\n          registerFile,\n          `export function register${args.name}() {\\n  return ${args.setDefault ? \"'default'\" : \"'optional'\"};\\n}\\n`\n        );\n      }\n      return { functionFile, registerFile };\n    },\n  },\n} as const;\n\nexport type DecoratorTools = typeof decoratorTools;\n"]}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
export type FastMCPLike = {
|
|
2
|
+
addPrompt: (p: any) => void;
|
|
3
|
+
addTool: (t: any) => void;
|
|
4
|
+
addResource: (r: any) => void;
|
|
5
|
+
addResourceTemplate: (t: any) => void;
|
|
6
|
+
};
|
|
7
|
+
/**
|
|
8
|
+
* Aggregate module assets and register them on the provided FastMCP-like server.
|
|
9
|
+
* Falls back to built-in lists if aggregation yields none.
|
|
10
|
+
*/
|
|
11
|
+
export declare function EnrichCoreWithAggregation(server: FastMCPLike, repoRoot?: string): Promise<{
|
|
12
|
+
modulesChecked: number;
|
|
13
|
+
conflicts: import("./aggregateModules").AggregationConflict[];
|
|
14
|
+
}>;
|