@decaf-ts/mcp-server 0.0.2 → 0.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE.md +86 -21
- package/README.md +13 -13
- package/dist/mcp-server.cjs +893 -187
- package/dist/mcp-server.esm.cjs +888 -180
- package/lib/McpWrapper.cjs +189 -0
- package/lib/McpWrapper.d.ts +101 -0
- package/lib/bin/cli.cjs +30 -54
- package/lib/bin/cli.d.ts +18 -44
- package/lib/constants.cjs +12 -0
- package/lib/constants.d.ts +8 -0
- package/lib/esm/McpWrapper.d.ts +101 -0
- package/lib/esm/McpWrapper.js +182 -0
- package/lib/esm/bin/cli.d.ts +18 -44
- package/lib/esm/bin/cli.js +28 -55
- package/lib/esm/constants.d.ts +8 -0
- package/lib/esm/constants.js +9 -0
- package/lib/esm/index.d.ts +5 -26
- package/lib/esm/index.js +6 -27
- package/lib/esm/mcp/index.d.ts +1 -0
- package/lib/esm/mcp/index.js +2 -0
- package/lib/esm/metadata.d.ts +9 -0
- package/lib/esm/metadata.js +22 -0
- package/lib/esm/modules/decoration/index.d.ts +0 -0
- package/lib/esm/modules/decoration/index.js +2 -0
- package/lib/esm/modules/mcp/decoration-assist.d.ts +39 -0
- package/lib/esm/modules/mcp/decoration-assist.js +353 -0
- package/lib/esm/modules/mcp/decorator-tools.d.ts +118 -0
- package/lib/esm/modules/mcp/decorator-tools.js +237 -0
- package/lib/esm/modules/mcp/index.d.ts +2 -0
- package/lib/esm/modules/mcp/index.js +3 -0
- package/lib/esm/modules/mcp/mcp-module.d.ts +230 -0
- package/lib/esm/modules/mcp/mcp-module.js +406 -0
- package/lib/esm/types.d.ts +15 -0
- package/lib/esm/types.js +2 -0
- package/lib/esm/utils.d.ts +54 -13
- package/lib/esm/utils.js +78 -15
- package/lib/index.cjs +6 -28
- package/lib/index.d.ts +5 -26
- package/lib/mcp/index.cjs +17 -0
- package/lib/mcp/index.d.ts +1 -0
- package/lib/metadata.cjs +25 -0
- package/lib/metadata.d.ts +9 -0
- package/lib/modules/decoration/index.cjs +2 -0
- package/lib/modules/decoration/index.d.ts +0 -0
- package/lib/modules/mcp/decoration-assist.cjs +360 -0
- package/lib/modules/mcp/decoration-assist.d.ts +39 -0
- package/lib/modules/mcp/decorator-tools.cjs +243 -0
- package/lib/modules/mcp/decorator-tools.d.ts +118 -0
- package/lib/modules/mcp/index.cjs +24 -0
- package/lib/modules/mcp/index.d.ts +2 -0
- package/lib/modules/mcp/mcp-module.cjs +452 -0
- package/lib/modules/mcp/mcp-module.d.ts +230 -0
- package/lib/types.cjs +3 -0
- package/lib/types.d.ts +15 -0
- package/lib/utils.cjs +116 -16
- package/lib/utils.d.ts +54 -13
- package/package.json +35 -7
- package/lib/esm/namespace/Class.d.ts +0 -74
- package/lib/esm/namespace/Class.js +0 -73
- package/lib/esm/namespace/Interface.d.ts +0 -17
- package/lib/esm/namespace/Interface.js +0 -2
- package/lib/esm/namespace/children/ChildClass.d.ts +0 -44
- package/lib/esm/namespace/children/ChildClass.js +0 -43
- package/lib/esm/namespace/children/ChildInterface.d.ts +0 -22
- package/lib/esm/namespace/children/ChildInterface.js +0 -2
- package/lib/esm/namespace/children/Enum.d.ts +0 -14
- package/lib/esm/namespace/children/Enum.js +0 -16
- package/lib/esm/namespace/children/function.d.ts +0 -31
- package/lib/esm/namespace/children/function.js +0 -33
- package/lib/esm/namespace/children/index.d.ts +0 -25
- package/lib/esm/namespace/children/index.js +0 -26
- package/lib/esm/namespace/index.d.ts +0 -18
- package/lib/esm/namespace/index.js +0 -19
- package/lib/esm/namespace/type.d.ts +0 -28
- package/lib/esm/namespace/type.js +0 -2
- package/lib/namespace/Class.cjs +0 -77
- package/lib/namespace/Class.d.ts +0 -74
- package/lib/namespace/Interface.cjs +0 -3
- package/lib/namespace/Interface.d.ts +0 -17
- package/lib/namespace/children/ChildClass.cjs +0 -47
- package/lib/namespace/children/ChildClass.d.ts +0 -44
- package/lib/namespace/children/ChildInterface.cjs +0 -3
- package/lib/namespace/children/ChildInterface.d.ts +0 -22
- package/lib/namespace/children/Enum.cjs +0 -19
- package/lib/namespace/children/Enum.d.ts +0 -14
- package/lib/namespace/children/function.cjs +0 -36
- package/lib/namespace/children/function.d.ts +0 -31
- package/lib/namespace/children/index.cjs +0 -42
- package/lib/namespace/children/index.d.ts +0 -25
- package/lib/namespace/index.cjs +0 -35
- package/lib/namespace/index.d.ts +0 -18
- package/lib/namespace/type.cjs +0 -3
- package/lib/namespace/type.d.ts +0 -28
|
@@ -0,0 +1,243 @@
|
|
|
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.decoratorTools = void 0;
|
|
7
|
+
const fs_1 = __importDefault(require("fs"));
|
|
8
|
+
const path_1 = __importDefault(require("path"));
|
|
9
|
+
function escapeRegExp(value) {
|
|
10
|
+
return value.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
|
11
|
+
}
|
|
12
|
+
function formatDecorator(spec) {
|
|
13
|
+
const args = (spec.args || []).map((arg) => JSON.stringify(arg)).join(", ");
|
|
14
|
+
return `@${spec.name}(${args})`;
|
|
15
|
+
}
|
|
16
|
+
function ensureDirectory(filePath) {
|
|
17
|
+
fs_1.default.mkdirSync(path_1.default.dirname(filePath), { recursive: true });
|
|
18
|
+
}
|
|
19
|
+
function collectDecoratorNames(classDecorators, properties) {
|
|
20
|
+
const names = new Set();
|
|
21
|
+
names.add("model");
|
|
22
|
+
for (const decorator of classDecorators || []) {
|
|
23
|
+
names.add(decorator.name);
|
|
24
|
+
}
|
|
25
|
+
for (const property of properties || []) {
|
|
26
|
+
for (const decorator of property.decorators || []) {
|
|
27
|
+
names.add(decorator.name);
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
return names;
|
|
31
|
+
}
|
|
32
|
+
function ensureImport(content, importsFrom, decorators) {
|
|
33
|
+
/* istanbul ignore next */
|
|
34
|
+
if (!decorators.size)
|
|
35
|
+
return content;
|
|
36
|
+
const importRegex = new RegExp(`import\\s+\\{([^}]+)\\}\\s+from\\s+["']${escapeRegExp(importsFrom)}["'];`);
|
|
37
|
+
const match = content.match(importRegex);
|
|
38
|
+
const sorted = Array.from(decorators).sort();
|
|
39
|
+
if (match) {
|
|
40
|
+
const existing = match[1]
|
|
41
|
+
.split(",")
|
|
42
|
+
.map((name) => name.trim())
|
|
43
|
+
.filter(Boolean);
|
|
44
|
+
const merged = Array.from(new Set([...existing, ...sorted])).sort();
|
|
45
|
+
return content.replace(importRegex, `import { ${merged.join(", ")} } from "${importsFrom}";`);
|
|
46
|
+
}
|
|
47
|
+
const importLine = `import { ${sorted.join(", ")} } from "${importsFrom}";`;
|
|
48
|
+
return `${importLine}\n\n${content}`;
|
|
49
|
+
}
|
|
50
|
+
function addPropertyBlock(property) {
|
|
51
|
+
const decorators = (property.decorators || [])
|
|
52
|
+
.map(formatDecorator)
|
|
53
|
+
.join("\n ");
|
|
54
|
+
const decoratorBlock = decorators ? ` ${decorators}\n` : "";
|
|
55
|
+
return `${decoratorBlock} ${property.name}: ${property.type};`;
|
|
56
|
+
}
|
|
57
|
+
function removePropertyBlock(content, propertyName) {
|
|
58
|
+
const lines = content.split(/\r?\n/);
|
|
59
|
+
const result = [];
|
|
60
|
+
for (let i = 0; i < lines.length; i++) {
|
|
61
|
+
const line = lines[i];
|
|
62
|
+
if (line.trim().startsWith(`@`) &&
|
|
63
|
+
lines[i + 1]?.includes(`${propertyName}:`)) {
|
|
64
|
+
continue;
|
|
65
|
+
}
|
|
66
|
+
if (line.includes(`${propertyName}:`)) {
|
|
67
|
+
// skip this line and any trailing blank line
|
|
68
|
+
continue;
|
|
69
|
+
}
|
|
70
|
+
result.push(line);
|
|
71
|
+
}
|
|
72
|
+
return result.join("\n");
|
|
73
|
+
}
|
|
74
|
+
function insertDecorator(content, decorator, target) {
|
|
75
|
+
const decoratorLine = formatDecorator(decorator);
|
|
76
|
+
if (target.kind === "class") {
|
|
77
|
+
const classRegex = /(export\s+class\s+[^\s{]+\s*\{)/;
|
|
78
|
+
if (content.includes(decoratorLine))
|
|
79
|
+
return content;
|
|
80
|
+
return content.replace(classRegex, `${decoratorLine}\n$1`);
|
|
81
|
+
}
|
|
82
|
+
if (!target.name)
|
|
83
|
+
return content;
|
|
84
|
+
const propertyRegex = new RegExp(`(^\\s*)(?:@.*\\n\\1)*${escapeRegExp(target.name)}:`, "m");
|
|
85
|
+
const match = propertyRegex.exec(content);
|
|
86
|
+
if (!match)
|
|
87
|
+
return content;
|
|
88
|
+
const indent = match[1] || " ";
|
|
89
|
+
if (content.includes(`${indent}${decoratorLine}`))
|
|
90
|
+
return content;
|
|
91
|
+
return (content.slice(0, match.index) +
|
|
92
|
+
`${indent}${decoratorLine}\n` +
|
|
93
|
+
content.slice(match.index));
|
|
94
|
+
}
|
|
95
|
+
function removeDecorator(content, decoratorName, target) {
|
|
96
|
+
const decoratorRegex = new RegExp(`^\\s*@${escapeRegExp(decoratorName)}\\([^)]*\\)`, "m");
|
|
97
|
+
if (target.kind === "class") {
|
|
98
|
+
return content.replace(decoratorRegex, "");
|
|
99
|
+
}
|
|
100
|
+
if (target.name) {
|
|
101
|
+
const pattern = new RegExp(`(^\\s*@${escapeRegExp(decoratorName)}\\([^)]*\\)\\s*$\\n)(?=\\s*${escapeRegExp(target.name)}:)`, "m");
|
|
102
|
+
return content.replace(pattern, "");
|
|
103
|
+
}
|
|
104
|
+
return content;
|
|
105
|
+
}
|
|
106
|
+
function writeIfChanged(filePath, content) {
|
|
107
|
+
ensureDirectory(filePath);
|
|
108
|
+
fs_1.default.writeFileSync(filePath, content);
|
|
109
|
+
}
|
|
110
|
+
exports.decoratorTools = {
|
|
111
|
+
createOrUpdateModelTool: {
|
|
112
|
+
name: "create-or-update-model",
|
|
113
|
+
description: "Create or update a validation-ready model class",
|
|
114
|
+
execute: async (args) => {
|
|
115
|
+
if (!args.overwrite && fs_1.default.existsSync(args.filePath)) {
|
|
116
|
+
throw new Error(`File already exists at ${args.filePath}`);
|
|
117
|
+
}
|
|
118
|
+
const decorators = collectDecoratorNames(args.classDecorators, args.properties);
|
|
119
|
+
let content = `@model()`;
|
|
120
|
+
for (const decorator of args.classDecorators || []) {
|
|
121
|
+
content += `\n${formatDecorator(decorator)}`;
|
|
122
|
+
}
|
|
123
|
+
const properties = (args.properties || [])
|
|
124
|
+
.map(addPropertyBlock)
|
|
125
|
+
.join("\n\n");
|
|
126
|
+
content += `\nexport class ${args.className} {\n${properties ? `${properties}\n` : ""}}\n`;
|
|
127
|
+
content = ensureImport(content, args.importsFrom, decorators);
|
|
128
|
+
writeIfChanged(args.filePath, content);
|
|
129
|
+
return { filePath: args.filePath };
|
|
130
|
+
},
|
|
131
|
+
},
|
|
132
|
+
addAttributeTool: {
|
|
133
|
+
name: "add-attribute",
|
|
134
|
+
description: "Add a decorated attribute to an existing model",
|
|
135
|
+
execute: async (args) => {
|
|
136
|
+
if (!fs_1.default.existsSync(args.filePath)) {
|
|
137
|
+
throw new Error(`Model file not found at ${args.filePath}`);
|
|
138
|
+
}
|
|
139
|
+
let content = fs_1.default.readFileSync(args.filePath, "utf8");
|
|
140
|
+
if (content.includes(`${args.attribute.name}:`)) {
|
|
141
|
+
return { filePath: args.filePath };
|
|
142
|
+
}
|
|
143
|
+
const decorators = collectDecoratorNames(undefined, [args.attribute]);
|
|
144
|
+
content = ensureImport(content, args.importsFrom, decorators);
|
|
145
|
+
const insertionPoint = content.lastIndexOf("}");
|
|
146
|
+
const block = addPropertyBlock(args.attribute);
|
|
147
|
+
const before = content.slice(0, insertionPoint).replace(/\s*$/, "");
|
|
148
|
+
const after = content.slice(insertionPoint);
|
|
149
|
+
const updated = `${before}\n${block}\n${after}`;
|
|
150
|
+
writeIfChanged(args.filePath, updated);
|
|
151
|
+
return { filePath: args.filePath };
|
|
152
|
+
},
|
|
153
|
+
},
|
|
154
|
+
removeAttributeTool: {
|
|
155
|
+
name: "remove-attribute",
|
|
156
|
+
description: "Remove an attribute from a model class",
|
|
157
|
+
execute: async (args) => {
|
|
158
|
+
if (!fs_1.default.existsSync(args.filePath))
|
|
159
|
+
return { filePath: args.filePath };
|
|
160
|
+
const content = fs_1.default.readFileSync(args.filePath, "utf8");
|
|
161
|
+
const updated = removePropertyBlock(content, args.attributeName);
|
|
162
|
+
writeIfChanged(args.filePath, updated);
|
|
163
|
+
return { filePath: args.filePath };
|
|
164
|
+
},
|
|
165
|
+
},
|
|
166
|
+
applyDecoratorTool: {
|
|
167
|
+
name: "apply-decorator",
|
|
168
|
+
description: "Apply a decorator to a class or property",
|
|
169
|
+
execute: async (args) => {
|
|
170
|
+
if (!fs_1.default.existsSync(args.filePath)) {
|
|
171
|
+
throw new Error(`Model file not found at ${args.filePath}`);
|
|
172
|
+
}
|
|
173
|
+
let content = fs_1.default.readFileSync(args.filePath, "utf8");
|
|
174
|
+
const decorators = new Set([args.decorator.name]);
|
|
175
|
+
content = ensureImport(content, args.importsFrom, decorators);
|
|
176
|
+
content = insertDecorator(content, args.decorator, args.target);
|
|
177
|
+
writeIfChanged(args.filePath, content);
|
|
178
|
+
return { filePath: args.filePath };
|
|
179
|
+
},
|
|
180
|
+
},
|
|
181
|
+
removeDecoratorTool: {
|
|
182
|
+
name: "remove-decorator",
|
|
183
|
+
description: "Remove a decorator from a class or property",
|
|
184
|
+
execute: async (args) => {
|
|
185
|
+
if (!fs_1.default.existsSync(args.filePath))
|
|
186
|
+
return { filePath: args.filePath };
|
|
187
|
+
let content = fs_1.default.readFileSync(args.filePath, "utf8");
|
|
188
|
+
content = removeDecorator(content, args.decoratorName, args.target);
|
|
189
|
+
writeIfChanged(args.filePath, content);
|
|
190
|
+
return { filePath: args.filePath };
|
|
191
|
+
},
|
|
192
|
+
},
|
|
193
|
+
scaffoldValidatorTool: {
|
|
194
|
+
name: "scaffold-validator",
|
|
195
|
+
description: "Scaffold a validator class and optional decorator",
|
|
196
|
+
execute: async (args) => {
|
|
197
|
+
const classFile = path_1.default.join(args.validatorsDir, `${args.name}.ts`);
|
|
198
|
+
ensureDirectory(classFile);
|
|
199
|
+
const classContent = `export class ${args.name} {\n validate(value: unknown): boolean {\n return value !== undefined;\n }\n}\n`;
|
|
200
|
+
fs_1.default.writeFileSync(classFile, classContent);
|
|
201
|
+
let decoratorFile;
|
|
202
|
+
if (args.decoratorDir) {
|
|
203
|
+
decoratorFile = path_1.default.join(args.decoratorDir, `${args.name}Decorator.ts`);
|
|
204
|
+
ensureDirectory(decoratorFile);
|
|
205
|
+
fs_1.default.writeFileSync(decoratorFile, `export function ${args.name}Decorator() {\n return () => void 0;\n}\n`);
|
|
206
|
+
}
|
|
207
|
+
return { classFile, decoratorFile };
|
|
208
|
+
},
|
|
209
|
+
},
|
|
210
|
+
scaffoldSerializerTool: {
|
|
211
|
+
name: "scaffold-serializer",
|
|
212
|
+
description: "Scaffold a serializer class and optional registry",
|
|
213
|
+
execute: async (args) => {
|
|
214
|
+
const classFile = path_1.default.join(args.dir, `${args.name}.ts`);
|
|
215
|
+
ensureDirectory(classFile);
|
|
216
|
+
fs_1.default.writeFileSync(classFile, `export class ${args.name} {\n serialize(value: unknown): string {\n return JSON.stringify(value);\n }\n}\n`);
|
|
217
|
+
let registerFile;
|
|
218
|
+
if (args.registerDir) {
|
|
219
|
+
registerFile = path_1.default.join(args.registerDir, `${args.name}Register.ts`);
|
|
220
|
+
ensureDirectory(registerFile);
|
|
221
|
+
fs_1.default.writeFileSync(registerFile, `export function register${args.name}() {\n return ${args.setDefault ? "'default'" : "'optional'"};\n}\n`);
|
|
222
|
+
}
|
|
223
|
+
return { classFile, registerFile };
|
|
224
|
+
},
|
|
225
|
+
},
|
|
226
|
+
scaffoldHashingTool: {
|
|
227
|
+
name: "scaffold-hashing",
|
|
228
|
+
description: "Scaffold a hashing function and optional registry",
|
|
229
|
+
execute: async (args) => {
|
|
230
|
+
const functionFile = path_1.default.join(args.dir, `${args.name}.ts`);
|
|
231
|
+
ensureDirectory(functionFile);
|
|
232
|
+
fs_1.default.writeFileSync(functionFile, `export function ${args.name}(value: string): string {\n return value.split('').reverse().join('');\n}\n`);
|
|
233
|
+
let registerFile;
|
|
234
|
+
if (args.registerDir) {
|
|
235
|
+
registerFile = path_1.default.join(args.registerDir, `${args.name}Register.ts`);
|
|
236
|
+
ensureDirectory(registerFile);
|
|
237
|
+
fs_1.default.writeFileSync(registerFile, `export function register${args.name}() {\n return ${args.setDefault ? "'default'" : "'optional'"};\n}\n`);
|
|
238
|
+
}
|
|
239
|
+
return { functionFile, registerFile };
|
|
240
|
+
},
|
|
241
|
+
},
|
|
242
|
+
};
|
|
243
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorator-tools.js","sourceRoot":"","sources":["../../../src/modules/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,118 @@
|
|
|
1
|
+
export type DecoratorSpec = {
|
|
2
|
+
name: string;
|
|
3
|
+
args?: unknown[];
|
|
4
|
+
};
|
|
5
|
+
export type AttributeSpec = {
|
|
6
|
+
name: string;
|
|
7
|
+
type: string;
|
|
8
|
+
decorators?: DecoratorSpec[];
|
|
9
|
+
};
|
|
10
|
+
export declare const decoratorTools: {
|
|
11
|
+
readonly createOrUpdateModelTool: {
|
|
12
|
+
readonly name: "create-or-update-model";
|
|
13
|
+
readonly description: "Create or update a validation-ready model class";
|
|
14
|
+
readonly execute: (args: {
|
|
15
|
+
filePath: string;
|
|
16
|
+
className: string;
|
|
17
|
+
classDecorators?: DecoratorSpec[];
|
|
18
|
+
properties: AttributeSpec[];
|
|
19
|
+
importsFrom: string;
|
|
20
|
+
overwrite?: boolean;
|
|
21
|
+
}) => Promise<{
|
|
22
|
+
filePath: string;
|
|
23
|
+
}>;
|
|
24
|
+
};
|
|
25
|
+
readonly addAttributeTool: {
|
|
26
|
+
readonly name: "add-attribute";
|
|
27
|
+
readonly description: "Add a decorated attribute to an existing model";
|
|
28
|
+
readonly execute: (args: {
|
|
29
|
+
filePath: string;
|
|
30
|
+
className: string;
|
|
31
|
+
attribute: AttributeSpec;
|
|
32
|
+
importsFrom: string;
|
|
33
|
+
}) => Promise<{
|
|
34
|
+
filePath: string;
|
|
35
|
+
}>;
|
|
36
|
+
};
|
|
37
|
+
readonly removeAttributeTool: {
|
|
38
|
+
readonly name: "remove-attribute";
|
|
39
|
+
readonly description: "Remove an attribute from a model class";
|
|
40
|
+
readonly execute: (args: {
|
|
41
|
+
filePath: string;
|
|
42
|
+
className: string;
|
|
43
|
+
attributeName: string;
|
|
44
|
+
}) => Promise<{
|
|
45
|
+
filePath: string;
|
|
46
|
+
}>;
|
|
47
|
+
};
|
|
48
|
+
readonly applyDecoratorTool: {
|
|
49
|
+
readonly name: "apply-decorator";
|
|
50
|
+
readonly description: "Apply a decorator to a class or property";
|
|
51
|
+
readonly execute: (args: {
|
|
52
|
+
filePath: string;
|
|
53
|
+
className: string;
|
|
54
|
+
target: {
|
|
55
|
+
kind: "class" | "property";
|
|
56
|
+
name?: string;
|
|
57
|
+
};
|
|
58
|
+
decorator: DecoratorSpec;
|
|
59
|
+
importsFrom: string;
|
|
60
|
+
}) => Promise<{
|
|
61
|
+
filePath: string;
|
|
62
|
+
}>;
|
|
63
|
+
};
|
|
64
|
+
readonly removeDecoratorTool: {
|
|
65
|
+
readonly name: "remove-decorator";
|
|
66
|
+
readonly description: "Remove a decorator from a class or property";
|
|
67
|
+
readonly execute: (args: {
|
|
68
|
+
filePath: string;
|
|
69
|
+
className: string;
|
|
70
|
+
target: {
|
|
71
|
+
kind: "class" | "property";
|
|
72
|
+
name?: string;
|
|
73
|
+
};
|
|
74
|
+
decoratorName: string;
|
|
75
|
+
}) => Promise<{
|
|
76
|
+
filePath: string;
|
|
77
|
+
}>;
|
|
78
|
+
};
|
|
79
|
+
readonly scaffoldValidatorTool: {
|
|
80
|
+
readonly name: "scaffold-validator";
|
|
81
|
+
readonly description: "Scaffold a validator class and optional decorator";
|
|
82
|
+
readonly execute: (args: {
|
|
83
|
+
validatorsDir: string;
|
|
84
|
+
decoratorDir?: string;
|
|
85
|
+
name: string;
|
|
86
|
+
}) => Promise<{
|
|
87
|
+
classFile: string;
|
|
88
|
+
decoratorFile: string | undefined;
|
|
89
|
+
}>;
|
|
90
|
+
};
|
|
91
|
+
readonly scaffoldSerializerTool: {
|
|
92
|
+
readonly name: "scaffold-serializer";
|
|
93
|
+
readonly description: "Scaffold a serializer class and optional registry";
|
|
94
|
+
readonly execute: (args: {
|
|
95
|
+
dir: string;
|
|
96
|
+
name: string;
|
|
97
|
+
registerDir?: string;
|
|
98
|
+
setDefault?: boolean;
|
|
99
|
+
}) => Promise<{
|
|
100
|
+
classFile: string;
|
|
101
|
+
registerFile: string | undefined;
|
|
102
|
+
}>;
|
|
103
|
+
};
|
|
104
|
+
readonly scaffoldHashingTool: {
|
|
105
|
+
readonly name: "scaffold-hashing";
|
|
106
|
+
readonly description: "Scaffold a hashing function and optional registry";
|
|
107
|
+
readonly execute: (args: {
|
|
108
|
+
dir: string;
|
|
109
|
+
name: string;
|
|
110
|
+
registerDir?: string;
|
|
111
|
+
setDefault?: boolean;
|
|
112
|
+
}) => Promise<{
|
|
113
|
+
functionFile: string;
|
|
114
|
+
registerFile: string | undefined;
|
|
115
|
+
}>;
|
|
116
|
+
};
|
|
117
|
+
};
|
|
118
|
+
export type DecoratorTools = typeof decoratorTools;
|
|
@@ -0,0 +1,24 @@
|
|
|
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
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
17
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
18
|
+
};
|
|
19
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
20
|
+
exports.default = void 0;
|
|
21
|
+
var mcp_module_1 = require("./mcp-module.cjs");
|
|
22
|
+
Object.defineProperty(exports, "default", { enumerable: true, get: function () { return __importDefault(mcp_module_1).default; } });
|
|
23
|
+
__exportStar(require("./mcp-module.cjs"), exports);
|
|
24
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbW9kdWxlcy9tY3AvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSwrQ0FBdUM7QUFBOUIsc0hBQUEsT0FBTyxPQUFBO0FBQ2hCLG1EQUE2QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IGRlZmF1bHQgfSBmcm9tIFwiLi9tY3AtbW9kdWxlXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9tY3AtbW9kdWxlXCI7XG4iXX0=
|