@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,406 @@
|
|
|
1
|
+
import fs from "fs";
|
|
2
|
+
import path from "path";
|
|
3
|
+
import { applyPatch, createTwoFilesPatch } from "diff";
|
|
4
|
+
import { z } from "zod";
|
|
5
|
+
import { PACKAGE_NAME as PKG, VERSION as V } from "../../metadata";
|
|
6
|
+
import { decoratorTools } from "./decorator-tools";
|
|
7
|
+
const WORKSPACE_ROOT_ENV = "MCP_WORKSPACE_ROOT";
|
|
8
|
+
const PROMPT_DIRECTORIES = [".code/prompts", ".codex/prompts"];
|
|
9
|
+
const DEFAULT_PROMPT_NAME = "doc";
|
|
10
|
+
const CLIENT_INTEGRATIONS = [
|
|
11
|
+
{
|
|
12
|
+
id: "vscode",
|
|
13
|
+
display: "Visual Studio Code",
|
|
14
|
+
instructions: "When interacting from Visual Studio Code, prefer the vscode://workspace/{path} resource template to fetch file contents and use the apply-code-change tool to commit edits with previewable diffs.",
|
|
15
|
+
},
|
|
16
|
+
{
|
|
17
|
+
id: "cursor",
|
|
18
|
+
display: "Cursor",
|
|
19
|
+
instructions: "Cursor clients can retrieve and update files through the cursor://workspace/{path} resource template. Always validate patches in dryRun mode before applying permanent changes.",
|
|
20
|
+
},
|
|
21
|
+
{
|
|
22
|
+
id: "copilot",
|
|
23
|
+
display: "GitHub Copilot",
|
|
24
|
+
instructions: "Use the copilot://workspace/{path} resource template to stream file content into Copilot chat sessions. Prefer returning unified diffs to maintain alignment with Copilot's diff visualization.",
|
|
25
|
+
},
|
|
26
|
+
];
|
|
27
|
+
const documentCodeSchema = z
|
|
28
|
+
.object({
|
|
29
|
+
filePath: z.string().min(1, "filePath is required"),
|
|
30
|
+
promptName: z.string().optional(),
|
|
31
|
+
includePrompt: z.boolean().default(true),
|
|
32
|
+
includeCode: z.boolean().default(true),
|
|
33
|
+
includeMetadata: z.boolean().default(true),
|
|
34
|
+
additionalContext: z.string().optional(),
|
|
35
|
+
encoding: z.string().default("utf8"),
|
|
36
|
+
})
|
|
37
|
+
.strict();
|
|
38
|
+
const codeChangeSchema = z
|
|
39
|
+
.object({
|
|
40
|
+
filePath: z.string().min(1, "filePath is required"),
|
|
41
|
+
patch: z.string().min(1, "patch is required"),
|
|
42
|
+
dryRun: z.boolean().default(false),
|
|
43
|
+
showDiff: z.boolean().default(true),
|
|
44
|
+
diffContext: z.number().int().min(0).max(100).default(3),
|
|
45
|
+
encoding: z.string().default("utf8"),
|
|
46
|
+
})
|
|
47
|
+
.strict();
|
|
48
|
+
let workspaceRoot = initializeWorkspaceRoot();
|
|
49
|
+
let userErrorCtor;
|
|
50
|
+
class WorkspaceError extends Error {
|
|
51
|
+
constructor(message) {
|
|
52
|
+
super(message);
|
|
53
|
+
this.name = "WorkspaceError";
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
async function getUserErrorCtor() {
|
|
57
|
+
if (!userErrorCtor) {
|
|
58
|
+
try {
|
|
59
|
+
const mod = await import("fastmcp");
|
|
60
|
+
userErrorCtor = mod
|
|
61
|
+
.UserError;
|
|
62
|
+
}
|
|
63
|
+
catch {
|
|
64
|
+
userErrorCtor = class MCPUserError extends Error {
|
|
65
|
+
constructor(message) {
|
|
66
|
+
super(message);
|
|
67
|
+
this.name = "MCPUserError";
|
|
68
|
+
}
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
return userErrorCtor;
|
|
73
|
+
}
|
|
74
|
+
async function throwUserError(message) {
|
|
75
|
+
const Ctor = await getUserErrorCtor();
|
|
76
|
+
throw new Ctor(message);
|
|
77
|
+
}
|
|
78
|
+
const documentCodeTool = {
|
|
79
|
+
annotations: {
|
|
80
|
+
idempotentHint: true,
|
|
81
|
+
openWorldHint: false,
|
|
82
|
+
readOnlyHint: true,
|
|
83
|
+
title: "Document Source File",
|
|
84
|
+
},
|
|
85
|
+
description: "Generate documentation guidance for a file by combining repository prompts with the target source code.",
|
|
86
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
87
|
+
execute: async (input, _context) => {
|
|
88
|
+
const args = documentCodeSchema.parse(input);
|
|
89
|
+
const root = getWorkspaceRoot();
|
|
90
|
+
let filePath;
|
|
91
|
+
try {
|
|
92
|
+
filePath = resolveInWorkspace(root, args.filePath);
|
|
93
|
+
}
|
|
94
|
+
catch (error) {
|
|
95
|
+
if (error instanceof WorkspaceError) {
|
|
96
|
+
return throwUserError(error.message);
|
|
97
|
+
}
|
|
98
|
+
/* istanbul ignore next */
|
|
99
|
+
throw error;
|
|
100
|
+
}
|
|
101
|
+
if (!fs.existsSync(filePath)) {
|
|
102
|
+
return throwUserError(`Cannot document missing file at ${args.filePath}`);
|
|
103
|
+
}
|
|
104
|
+
const fileContent = fs.readFileSync(filePath, {
|
|
105
|
+
encoding: args.encoding,
|
|
106
|
+
});
|
|
107
|
+
const prompts = discoverDocPrompts(root);
|
|
108
|
+
if (!prompts.length) {
|
|
109
|
+
return throwUserError("No documentation prompts found under .code/prompts or .codex/prompts");
|
|
110
|
+
}
|
|
111
|
+
const prompt = selectPrompt(prompts, args.promptName ?? DEFAULT_PROMPT_NAME);
|
|
112
|
+
return buildDocumentationPayload({
|
|
113
|
+
filePath: args.filePath,
|
|
114
|
+
fileContent,
|
|
115
|
+
prompt,
|
|
116
|
+
includeCode: args.includeCode,
|
|
117
|
+
includePrompt: args.includePrompt,
|
|
118
|
+
includeMetadata: args.includeMetadata,
|
|
119
|
+
additionalContext: args.additionalContext,
|
|
120
|
+
});
|
|
121
|
+
},
|
|
122
|
+
name: "document-code",
|
|
123
|
+
parameters: documentCodeSchema,
|
|
124
|
+
};
|
|
125
|
+
const applyCodeChangeTool = {
|
|
126
|
+
annotations: {
|
|
127
|
+
destructiveHint: true,
|
|
128
|
+
idempotentHint: false,
|
|
129
|
+
openWorldHint: false,
|
|
130
|
+
readOnlyHint: false,
|
|
131
|
+
title: "Apply Code Patch",
|
|
132
|
+
},
|
|
133
|
+
description: "Apply a unified diff patch to a workspace file with optional dry-run validation and diff preview.",
|
|
134
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
135
|
+
execute: async (input, _context) => {
|
|
136
|
+
const args = codeChangeSchema.parse(input);
|
|
137
|
+
const root = getWorkspaceRoot();
|
|
138
|
+
let filePath;
|
|
139
|
+
try {
|
|
140
|
+
filePath = resolveInWorkspace(root, args.filePath);
|
|
141
|
+
}
|
|
142
|
+
catch (error) {
|
|
143
|
+
if (error instanceof WorkspaceError) {
|
|
144
|
+
return throwUserError(error.message);
|
|
145
|
+
}
|
|
146
|
+
throw error;
|
|
147
|
+
}
|
|
148
|
+
const original = fs.existsSync(filePath)
|
|
149
|
+
? fs.readFileSync(filePath, args.encoding)
|
|
150
|
+
: "";
|
|
151
|
+
let patched;
|
|
152
|
+
try {
|
|
153
|
+
patched = applyPatch(original, args.patch);
|
|
154
|
+
}
|
|
155
|
+
catch (error) {
|
|
156
|
+
return throwUserError(`Failed to apply provided patch to ${args.filePath}: ${error instanceof Error ? error.message : error}`);
|
|
157
|
+
}
|
|
158
|
+
/* istanbul ignore next */
|
|
159
|
+
if (patched === false) {
|
|
160
|
+
return throwUserError(`Failed to apply provided patch to ${args.filePath}`);
|
|
161
|
+
}
|
|
162
|
+
if (!args.dryRun) {
|
|
163
|
+
fs.mkdirSync(path.dirname(filePath), { recursive: true });
|
|
164
|
+
fs.writeFileSync(filePath, patched, {
|
|
165
|
+
encoding: args.encoding,
|
|
166
|
+
});
|
|
167
|
+
}
|
|
168
|
+
if (!args.showDiff) {
|
|
169
|
+
return `Patch ${args.dryRun ? "validated" : "applied"} for ${args.filePath}`;
|
|
170
|
+
}
|
|
171
|
+
const preview = createTwoFilesPatch(args.filePath, args.filePath, original, patched, undefined, undefined, { context: args.diffContext });
|
|
172
|
+
return {
|
|
173
|
+
content: [
|
|
174
|
+
{
|
|
175
|
+
type: "text",
|
|
176
|
+
text: `Patch ${args.dryRun ? "validated" : "applied"} for ${args.filePath}`,
|
|
177
|
+
},
|
|
178
|
+
{
|
|
179
|
+
type: "text",
|
|
180
|
+
text: ["```diff", preview.trim(), "```"].join("\n"),
|
|
181
|
+
},
|
|
182
|
+
],
|
|
183
|
+
};
|
|
184
|
+
},
|
|
185
|
+
name: "apply-code-change",
|
|
186
|
+
parameters: codeChangeSchema,
|
|
187
|
+
};
|
|
188
|
+
export const tools = {
|
|
189
|
+
...decoratorTools,
|
|
190
|
+
documentCodeTool,
|
|
191
|
+
applyCodeChangeTool,
|
|
192
|
+
};
|
|
193
|
+
export function enrich(mcp) {
|
|
194
|
+
for (const prompt of buildDocPrompts()) {
|
|
195
|
+
mcp.addPrompt(prompt);
|
|
196
|
+
}
|
|
197
|
+
for (const tool of Object.values(tools)) {
|
|
198
|
+
mcp.addTool(tool);
|
|
199
|
+
}
|
|
200
|
+
for (const template of buildResourceTemplates()) {
|
|
201
|
+
mcp.addResourceTemplate(template);
|
|
202
|
+
}
|
|
203
|
+
return mcp;
|
|
204
|
+
}
|
|
205
|
+
export default enrich;
|
|
206
|
+
export const PACKAGE_NAME = PKG;
|
|
207
|
+
export const VERSION = V;
|
|
208
|
+
export function setWorkspaceRoot(root) {
|
|
209
|
+
workspaceRoot = path.resolve(root);
|
|
210
|
+
}
|
|
211
|
+
export function getWorkspaceRoot() {
|
|
212
|
+
return workspaceRoot;
|
|
213
|
+
}
|
|
214
|
+
export function buildResourceTemplates() {
|
|
215
|
+
const root = getWorkspaceRoot();
|
|
216
|
+
const sharedArguments = [
|
|
217
|
+
{
|
|
218
|
+
name: "path",
|
|
219
|
+
description: "Path relative to the workspace root",
|
|
220
|
+
required: true,
|
|
221
|
+
},
|
|
222
|
+
];
|
|
223
|
+
return [
|
|
224
|
+
{
|
|
225
|
+
name: "vscode-workspace-file",
|
|
226
|
+
description: "Expose workspace files to Visual Studio Code via vscode:// URIs",
|
|
227
|
+
uriTemplate: "vscode://workspace/{path}",
|
|
228
|
+
mimeType: "text/plain",
|
|
229
|
+
arguments: sharedArguments,
|
|
230
|
+
load: async (args) => {
|
|
231
|
+
const text = await readWorkspaceFile(root, args.path);
|
|
232
|
+
return { text };
|
|
233
|
+
},
|
|
234
|
+
},
|
|
235
|
+
{
|
|
236
|
+
name: "cursor-workspace-file",
|
|
237
|
+
description: "Expose workspace files to Cursor via cursor:// URIs",
|
|
238
|
+
uriTemplate: "cursor://workspace/{path}",
|
|
239
|
+
mimeType: "text/plain",
|
|
240
|
+
arguments: sharedArguments,
|
|
241
|
+
load: async (args) => {
|
|
242
|
+
const text = await readWorkspaceFile(root, args.path);
|
|
243
|
+
return { text };
|
|
244
|
+
},
|
|
245
|
+
},
|
|
246
|
+
{
|
|
247
|
+
name: "copilot-workspace-file",
|
|
248
|
+
description: "Expose workspace files to GitHub Copilot via copilot:// URIs",
|
|
249
|
+
uriTemplate: "copilot://workspace/{path}",
|
|
250
|
+
mimeType: "text/plain",
|
|
251
|
+
arguments: sharedArguments,
|
|
252
|
+
load: async (args) => {
|
|
253
|
+
const text = await readWorkspaceFile(root, args.path);
|
|
254
|
+
return { text };
|
|
255
|
+
},
|
|
256
|
+
},
|
|
257
|
+
];
|
|
258
|
+
}
|
|
259
|
+
function initializeWorkspaceRoot() {
|
|
260
|
+
const configured = process.env[WORKSPACE_ROOT_ENV];
|
|
261
|
+
if (configured && configured.trim().length > 0) {
|
|
262
|
+
return path.resolve(configured.trim());
|
|
263
|
+
}
|
|
264
|
+
return process.cwd();
|
|
265
|
+
}
|
|
266
|
+
export function buildDocPrompts() {
|
|
267
|
+
const root = getWorkspaceRoot();
|
|
268
|
+
const fileBasedPrompts = discoverDocPrompts(root).map((prompt) => ({
|
|
269
|
+
name: `doc/${prompt.name}`,
|
|
270
|
+
description: prompt.description,
|
|
271
|
+
load: async () => prompt.content,
|
|
272
|
+
}));
|
|
273
|
+
const integrationPrompts = CLIENT_INTEGRATIONS.map((integration) => ({
|
|
274
|
+
name: `integration/${integration.id}`,
|
|
275
|
+
description: `${integration.display} integration guidance`,
|
|
276
|
+
load: async () => `You are coordinating with ${integration.display}. ${integration.instructions}\n\nTools available:\n- document-code\n- apply-code-change\n\nEnsure responses include actionable steps for the client.`,
|
|
277
|
+
}));
|
|
278
|
+
return [...fileBasedPrompts, ...integrationPrompts];
|
|
279
|
+
}
|
|
280
|
+
function resolveInWorkspace(root, targetPath) {
|
|
281
|
+
const resolved = path.isAbsolute(targetPath)
|
|
282
|
+
? path.normalize(targetPath)
|
|
283
|
+
: path.resolve(root, targetPath);
|
|
284
|
+
const relative = path.relative(root, resolved);
|
|
285
|
+
if (relative.startsWith("..") || path.isAbsolute(relative)) {
|
|
286
|
+
throw new WorkspaceError(`Path ${targetPath} escapes the workspace root at ${root}`);
|
|
287
|
+
}
|
|
288
|
+
return resolved;
|
|
289
|
+
}
|
|
290
|
+
async function readWorkspaceFile(root, target) {
|
|
291
|
+
try {
|
|
292
|
+
const absolute = resolveInWorkspace(root, target);
|
|
293
|
+
return fs.readFileSync(absolute, "utf8");
|
|
294
|
+
}
|
|
295
|
+
catch (error) {
|
|
296
|
+
if (error instanceof WorkspaceError) {
|
|
297
|
+
await throwUserError(error.message);
|
|
298
|
+
}
|
|
299
|
+
/* istanbul ignore next */
|
|
300
|
+
throw error;
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
function discoverDocPrompts(root) {
|
|
304
|
+
const discovered = [];
|
|
305
|
+
for (const directory of PROMPT_DIRECTORIES) {
|
|
306
|
+
const promptDir = path.join(root, directory);
|
|
307
|
+
if (!fs.existsSync(promptDir) || !fs.statSync(promptDir).isDirectory()) {
|
|
308
|
+
continue;
|
|
309
|
+
}
|
|
310
|
+
for (const entry of fs.readdirSync(promptDir)) {
|
|
311
|
+
const fullPath = path.join(promptDir, entry);
|
|
312
|
+
if (!fs.statSync(fullPath).isFile())
|
|
313
|
+
continue;
|
|
314
|
+
const name = path.parse(entry).name;
|
|
315
|
+
const content = fs.readFileSync(fullPath, "utf8");
|
|
316
|
+
const title = toTitleCase(name.replace(/[-_]/g, " "));
|
|
317
|
+
const description = extractDescription(content, fullPath);
|
|
318
|
+
discovered.push({
|
|
319
|
+
name,
|
|
320
|
+
title,
|
|
321
|
+
description,
|
|
322
|
+
content,
|
|
323
|
+
absolutePath: fullPath,
|
|
324
|
+
});
|
|
325
|
+
}
|
|
326
|
+
}
|
|
327
|
+
const unique = new Map();
|
|
328
|
+
for (const prompt of discovered) {
|
|
329
|
+
if (!unique.has(prompt.name)) {
|
|
330
|
+
unique.set(prompt.name, prompt);
|
|
331
|
+
}
|
|
332
|
+
}
|
|
333
|
+
return Array.from(unique.values()).sort((a, b) => a.name.localeCompare(b.name));
|
|
334
|
+
}
|
|
335
|
+
function selectPrompt(prompts, requestedName) {
|
|
336
|
+
const direct = prompts.find((prompt) => prompt.name === requestedName);
|
|
337
|
+
if (direct)
|
|
338
|
+
return direct;
|
|
339
|
+
const fallback = prompts.find((prompt) => prompt.name === DEFAULT_PROMPT_NAME);
|
|
340
|
+
if (fallback)
|
|
341
|
+
return fallback;
|
|
342
|
+
if (!prompts.length) {
|
|
343
|
+
throw new WorkspaceError("No documentation prompts available");
|
|
344
|
+
}
|
|
345
|
+
return prompts[0];
|
|
346
|
+
}
|
|
347
|
+
function buildDocumentationPayload({ filePath, fileContent, prompt, includePrompt, includeCode, includeMetadata, additionalContext, }) {
|
|
348
|
+
const sections = [];
|
|
349
|
+
if (includeMetadata) {
|
|
350
|
+
sections.push(`# Documentation Request\n- prompt: ${prompt.name}\n- file: ${filePath}`);
|
|
351
|
+
}
|
|
352
|
+
if (includePrompt) {
|
|
353
|
+
sections.push(`## Prompt Guidance (${prompt.title})\n\n${prompt.content.trim()}`);
|
|
354
|
+
}
|
|
355
|
+
if (additionalContext?.trim()) {
|
|
356
|
+
sections.push(`## Additional Context\n\n${additionalContext.trim()}`);
|
|
357
|
+
}
|
|
358
|
+
if (includeCode) {
|
|
359
|
+
sections.push(`## Source\n\n\`\`\`${inferLanguageFromPath(filePath)}\n${fileContent}\n\`\`\``);
|
|
360
|
+
}
|
|
361
|
+
return {
|
|
362
|
+
content: [
|
|
363
|
+
{
|
|
364
|
+
type: "text",
|
|
365
|
+
text: sections.join("\n\n"),
|
|
366
|
+
},
|
|
367
|
+
],
|
|
368
|
+
};
|
|
369
|
+
}
|
|
370
|
+
function extractDescription(content, filePath) {
|
|
371
|
+
const firstLine = content
|
|
372
|
+
.split(/\r?\n/)
|
|
373
|
+
.map((line) => line.trim())
|
|
374
|
+
.find((line) => line.length > 0);
|
|
375
|
+
return (firstLine?.slice(0, 240) ??
|
|
376
|
+
`Documentation prompt loaded from ${path.basename(filePath)}`);
|
|
377
|
+
}
|
|
378
|
+
function toTitleCase(value) {
|
|
379
|
+
return value
|
|
380
|
+
.split(/\s+/)
|
|
381
|
+
.filter(Boolean)
|
|
382
|
+
.map((part) => part.charAt(0).toUpperCase() + part.slice(1).toLowerCase())
|
|
383
|
+
.join(" ");
|
|
384
|
+
}
|
|
385
|
+
function inferLanguageFromPath(filePath) {
|
|
386
|
+
const extension = path.extname(filePath).toLowerCase();
|
|
387
|
+
switch (extension) {
|
|
388
|
+
case ".ts":
|
|
389
|
+
case ".tsx":
|
|
390
|
+
return "ts";
|
|
391
|
+
case ".js":
|
|
392
|
+
case ".jsx":
|
|
393
|
+
return "js";
|
|
394
|
+
case ".json":
|
|
395
|
+
return "json";
|
|
396
|
+
case ".md":
|
|
397
|
+
return "md";
|
|
398
|
+
default:
|
|
399
|
+
return "text";
|
|
400
|
+
}
|
|
401
|
+
}
|
|
402
|
+
export function __resetWorkspaceRoot(root) {
|
|
403
|
+
setWorkspaceRoot(root);
|
|
404
|
+
}
|
|
405
|
+
export { documentCodeSchema, codeChangeSchema };
|
|
406
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mcp-module.js","sourceRoot":"","sources":["../../../../src/modules/mcp/mcp-module.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from \"fs\";\nimport path from \"path\";\nimport type { FastMCP, ContentResult, InputPrompt, Tool } from \"fastmcp\";\nimport { applyPatch, createTwoFilesPatch } from \"diff\";\nimport { z } from \"zod\";\nimport { PACKAGE_NAME as PKG, VERSION as V } from \"../../metadata\";\nimport { decoratorTools } from \"./decorator-tools\";\n\nconst WORKSPACE_ROOT_ENV = \"MCP_WORKSPACE_ROOT\";\nconst PROMPT_DIRECTORIES = [\".code/prompts\", \".codex/prompts\"];\nconst DEFAULT_PROMPT_NAME = \"doc\";\nconst CLIENT_INTEGRATIONS = [\n  {\n    id: \"vscode\",\n    display: \"Visual Studio Code\",\n    instructions:\n      \"When interacting from Visual Studio Code, prefer the vscode://workspace/{path} resource template to fetch file contents and use the apply-code-change tool to commit edits with previewable diffs.\",\n  },\n  {\n    id: \"cursor\",\n    display: \"Cursor\",\n    instructions:\n      \"Cursor clients can retrieve and update files through the cursor://workspace/{path} resource template. Always validate patches in dryRun mode before applying permanent changes.\",\n  },\n  {\n    id: \"copilot\",\n    display: \"GitHub Copilot\",\n    instructions:\n      \"Use the copilot://workspace/{path} resource template to stream file content into Copilot chat sessions. Prefer returning unified diffs to maintain alignment with Copilot's diff visualization.\",\n  },\n] as const;\n\nconst documentCodeSchema = z\n  .object({\n    filePath: z.string().min(1, \"filePath is required\"),\n    promptName: z.string().optional(),\n    includePrompt: z.boolean().default(true),\n    includeCode: z.boolean().default(true),\n    includeMetadata: z.boolean().default(true),\n    additionalContext: z.string().optional(),\n    encoding: z.string().default(\"utf8\"),\n  })\n  .strict();\n\ntype DocumentCodeArgs = z.infer<typeof documentCodeSchema>;\n\nconst codeChangeSchema = z\n  .object({\n    filePath: z.string().min(1, \"filePath is required\"),\n    patch: z.string().min(1, \"patch is required\"),\n    dryRun: z.boolean().default(false),\n    showDiff: z.boolean().default(true),\n    diffContext: z.number().int().min(0).max(100).default(3),\n    encoding: z.string().default(\"utf8\"),\n  })\n  .strict();\n\ntype ApplyCodeChangeArgs = z.infer<typeof codeChangeSchema>;\n\ntype DocPrompt = {\n  name: string;\n  title: string;\n  description: string;\n  content: string;\n  absolutePath: string;\n};\n\ntype WorkspaceResourceTemplate = {\n  name: string;\n  description: string;\n  uriTemplate: string;\n  mimeType: string;\n  arguments: ReadonlyArray<{\n    name: string;\n    description: string;\n    required: boolean;\n  }>;\n  load: (args: { path: string }) => Promise<{ text: string }>;\n};\n\nlet workspaceRoot = initializeWorkspaceRoot();\nlet userErrorCtor: (new (message: string) => Error) | undefined;\n\nclass WorkspaceError extends Error {\n  constructor(message: string) {\n    super(message);\n    this.name = \"WorkspaceError\";\n  }\n}\n\nasync function getUserErrorCtor(): Promise<new (message: string) => Error> {\n  if (!userErrorCtor) {\n    try {\n      const mod = await import(\"fastmcp\");\n      userErrorCtor = (mod as { UserError: new (message: string) => Error })\n        .UserError;\n    } catch {\n      userErrorCtor = class MCPUserError extends Error {\n        constructor(message: string) {\n          super(message);\n          this.name = \"MCPUserError\";\n        }\n      };\n    }\n  }\n  return userErrorCtor;\n}\n\nasync function throwUserError(message: string): Promise<never> {\n  const Ctor = await getUserErrorCtor();\n  throw new Ctor(message);\n}\n\nconst documentCodeTool: Tool<undefined, typeof documentCodeSchema> = {\n  annotations: {\n    idempotentHint: true,\n    openWorldHint: false,\n    readOnlyHint: true,\n    title: \"Document Source File\",\n  },\n  description:\n    \"Generate documentation guidance for a file by combining repository prompts with the target source code.\",\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  execute: async (input, _context): Promise<ContentResult> => {\n    const args = documentCodeSchema.parse(input as DocumentCodeArgs);\n    const root = getWorkspaceRoot();\n    let filePath: string;\n    try {\n      filePath = resolveInWorkspace(root, args.filePath);\n    } catch (error) {\n      if (error instanceof WorkspaceError) {\n        return throwUserError(error.message);\n      }\n      /* istanbul ignore next */\n      throw error;\n    }\n\n    if (!fs.existsSync(filePath)) {\n      return throwUserError(`Cannot document missing file at ${args.filePath}`);\n    }\n\n    const fileContent = fs.readFileSync(filePath, {\n      encoding: args.encoding as BufferEncoding,\n    });\n    const prompts = discoverDocPrompts(root);\n\n    if (!prompts.length) {\n      return throwUserError(\n        \"No documentation prompts found under .code/prompts or .codex/prompts\"\n      );\n    }\n\n    const prompt = selectPrompt(\n      prompts,\n      args.promptName ?? DEFAULT_PROMPT_NAME\n    );\n\n    return buildDocumentationPayload({\n      filePath: args.filePath,\n      fileContent,\n      prompt,\n      includeCode: args.includeCode,\n      includePrompt: args.includePrompt,\n      includeMetadata: args.includeMetadata,\n      additionalContext: args.additionalContext,\n    });\n  },\n  name: \"document-code\",\n  parameters: documentCodeSchema,\n};\n\nconst applyCodeChangeTool: Tool<undefined, typeof codeChangeSchema> = {\n  annotations: {\n    destructiveHint: true,\n    idempotentHint: false,\n    openWorldHint: false,\n    readOnlyHint: false,\n    title: \"Apply Code Patch\",\n  },\n  description:\n    \"Apply a unified diff patch to a workspace file with optional dry-run validation and diff preview.\",\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  execute: async (input, _context): Promise<string | ContentResult> => {\n    const args = codeChangeSchema.parse(input as ApplyCodeChangeArgs);\n    const root = getWorkspaceRoot();\n    let filePath: string;\n    try {\n      filePath = resolveInWorkspace(root, args.filePath);\n    } catch (error) {\n      if (error instanceof WorkspaceError) {\n        return throwUserError(error.message);\n      }\n      throw error;\n    }\n\n    const original = fs.existsSync(filePath)\n      ? fs.readFileSync(filePath, args.encoding as BufferEncoding)\n      : \"\";\n\n    let patched: string | false;\n    try {\n      patched = applyPatch(original, args.patch);\n    } catch (error) {\n      return throwUserError(\n        `Failed to apply provided patch to ${args.filePath}: ${error instanceof Error ? error.message : error}`\n      );\n    }\n    /* istanbul ignore next */\n    if (patched === false) {\n      return throwUserError(\n        `Failed to apply provided patch to ${args.filePath}`\n      );\n    }\n\n    if (!args.dryRun) {\n      fs.mkdirSync(path.dirname(filePath), { recursive: true });\n      fs.writeFileSync(filePath, patched, {\n        encoding: args.encoding as BufferEncoding,\n      });\n    }\n\n    if (!args.showDiff) {\n      return `Patch ${args.dryRun ? \"validated\" : \"applied\"} for ${args.filePath}`;\n    }\n\n    const preview = createTwoFilesPatch(\n      args.filePath,\n      args.filePath,\n      original,\n      patched,\n      undefined,\n      undefined,\n      { context: args.diffContext }\n    );\n\n    return {\n      content: [\n        {\n          type: \"text\",\n          text: `Patch ${args.dryRun ? \"validated\" : \"applied\"} for ${args.filePath}`,\n        },\n        {\n          type: \"text\",\n          text: [\"```diff\", preview.trim(), \"```\"].join(\"\\n\"),\n        },\n      ],\n    } satisfies ContentResult;\n  },\n  name: \"apply-code-change\",\n  parameters: codeChangeSchema,\n};\n\nexport const tools = {\n  ...decoratorTools,\n  documentCodeTool,\n  applyCodeChangeTool,\n} as const;\n\nexport function enrich(mcp: FastMCP): FastMCP {\n  for (const prompt of buildDocPrompts()) {\n    mcp.addPrompt(prompt as any);\n  }\n\n  for (const tool of Object.values(tools)) {\n    mcp.addTool(tool as any);\n  }\n\n  for (const template of buildResourceTemplates()) {\n    mcp.addResourceTemplate(template as any);\n  }\n\n  return mcp;\n}\n\nexport default enrich;\nexport const PACKAGE_NAME = PKG;\nexport const VERSION = V;\n\nexport function setWorkspaceRoot(root: string) {\n  workspaceRoot = path.resolve(root);\n}\n\nexport function getWorkspaceRoot(): string {\n  return workspaceRoot;\n}\n\nexport function buildResourceTemplates(): WorkspaceResourceTemplate[] {\n  const root = getWorkspaceRoot();\n  const sharedArguments = [\n    {\n      name: \"path\",\n      description: \"Path relative to the workspace root\",\n      required: true,\n    },\n  ] as const;\n\n  return [\n    {\n      name: \"vscode-workspace-file\",\n      description:\n        \"Expose workspace files to Visual Studio Code via vscode:// URIs\",\n      uriTemplate: \"vscode://workspace/{path}\",\n      mimeType: \"text/plain\",\n      arguments: sharedArguments,\n      load: async (args: { path: string }) => {\n        const text = await readWorkspaceFile(root, args.path);\n        return { text };\n      },\n    },\n    {\n      name: \"cursor-workspace-file\",\n      description: \"Expose workspace files to Cursor via cursor:// URIs\",\n      uriTemplate: \"cursor://workspace/{path}\",\n      mimeType: \"text/plain\",\n      arguments: sharedArguments,\n      load: async (args: { path: string }) => {\n        const text = await readWorkspaceFile(root, args.path);\n        return { text };\n      },\n    },\n    {\n      name: \"copilot-workspace-file\",\n      description:\n        \"Expose workspace files to GitHub Copilot via copilot:// URIs\",\n      uriTemplate: \"copilot://workspace/{path}\",\n      mimeType: \"text/plain\",\n      arguments: sharedArguments,\n      load: async (args: { path: string }) => {\n        const text = await readWorkspaceFile(root, args.path);\n        return { text };\n      },\n    },\n  ];\n}\n\nfunction initializeWorkspaceRoot(): string {\n  const configured = process.env[WORKSPACE_ROOT_ENV];\n  if (configured && configured.trim().length > 0) {\n    return path.resolve(configured.trim());\n  }\n  return process.cwd();\n}\n\nexport function buildDocPrompts(): InputPrompt<undefined>[] {\n  const root = getWorkspaceRoot();\n  const fileBasedPrompts = discoverDocPrompts(root).map((prompt) => ({\n    name: `doc/${prompt.name}`,\n    description: prompt.description,\n    load: async () => prompt.content,\n  }));\n\n  const integrationPrompts = CLIENT_INTEGRATIONS.map<InputPrompt<undefined>>(\n    (integration) => ({\n      name: `integration/${integration.id}`,\n      description: `${integration.display} integration guidance`,\n      load: async () =>\n        `You are coordinating with ${integration.display}. ${integration.instructions}\\n\\nTools available:\\n- document-code\\n- apply-code-change\\n\\nEnsure responses include actionable steps for the client.`,\n    })\n  );\n\n  return [...fileBasedPrompts, ...integrationPrompts];\n}\n\nfunction resolveInWorkspace(root: string, targetPath: string): string {\n  const resolved = path.isAbsolute(targetPath)\n    ? path.normalize(targetPath)\n    : path.resolve(root, targetPath);\n\n  const relative = path.relative(root, resolved);\n  if (relative.startsWith(\"..\") || path.isAbsolute(relative)) {\n    throw new WorkspaceError(\n      `Path ${targetPath} escapes the workspace root at ${root}`\n    );\n  }\n\n  return resolved;\n}\n\nasync function readWorkspaceFile(\n  root: string,\n  target: string\n): Promise<string> {\n  try {\n    const absolute = resolveInWorkspace(root, target);\n    return fs.readFileSync(absolute, \"utf8\" as BufferEncoding);\n  } catch (error) {\n    if (error instanceof WorkspaceError) {\n      await throwUserError(error.message);\n    }\n    /* istanbul ignore next */\n    throw error;\n  }\n}\n\nfunction discoverDocPrompts(root: string): DocPrompt[] {\n  const discovered: DocPrompt[] = [];\n\n  for (const directory of PROMPT_DIRECTORIES) {\n    const promptDir = path.join(root, directory);\n    if (!fs.existsSync(promptDir) || !fs.statSync(promptDir).isDirectory()) {\n      continue;\n    }\n\n    for (const entry of fs.readdirSync(promptDir)) {\n      const fullPath = path.join(promptDir, entry);\n      if (!fs.statSync(fullPath).isFile()) continue;\n\n      const name = path.parse(entry).name;\n      const content = fs.readFileSync(fullPath, \"utf8\");\n      const title = toTitleCase(name.replace(/[-_]/g, \" \"));\n      const description = extractDescription(content, fullPath);\n\n      discovered.push({\n        name,\n        title,\n        description,\n        content,\n        absolutePath: fullPath,\n      });\n    }\n  }\n\n  const unique = new Map<string, DocPrompt>();\n  for (const prompt of discovered) {\n    if (!unique.has(prompt.name)) {\n      unique.set(prompt.name, prompt);\n    }\n  }\n\n  return Array.from(unique.values()).sort((a, b) =>\n    a.name.localeCompare(b.name)\n  );\n}\n\nfunction selectPrompt(prompts: DocPrompt[], requestedName: string): DocPrompt {\n  const direct = prompts.find((prompt) => prompt.name === requestedName);\n  if (direct) return direct;\n\n  const fallback = prompts.find(\n    (prompt) => prompt.name === DEFAULT_PROMPT_NAME\n  );\n  if (fallback) return fallback;\n\n  if (!prompts.length) {\n    throw new WorkspaceError(\"No documentation prompts available\");\n  }\n\n  return prompts[0];\n}\n\nfunction buildDocumentationPayload({\n  filePath,\n  fileContent,\n  prompt,\n  includePrompt,\n  includeCode,\n  includeMetadata,\n  additionalContext,\n}: {\n  filePath: string;\n  fileContent: string;\n  prompt: DocPrompt;\n  includePrompt: boolean;\n  includeCode: boolean;\n  includeMetadata: boolean;\n  additionalContext?: string;\n}): ContentResult {\n  const sections: string[] = [];\n\n  if (includeMetadata) {\n    sections.push(\n      `# Documentation Request\\n- prompt: ${prompt.name}\\n- file: ${filePath}`\n    );\n  }\n\n  if (includePrompt) {\n    sections.push(\n      `## Prompt Guidance (${prompt.title})\\n\\n${prompt.content.trim()}`\n    );\n  }\n\n  if (additionalContext?.trim()) {\n    sections.push(`## Additional Context\\n\\n${additionalContext.trim()}`);\n  }\n\n  if (includeCode) {\n    sections.push(\n      `## Source\\n\\n\\`\\`\\`${inferLanguageFromPath(filePath)}\\n${fileContent}\\n\\`\\`\\``\n    );\n  }\n\n  return {\n    content: [\n      {\n        type: \"text\",\n        text: sections.join(\"\\n\\n\"),\n      },\n    ],\n  } satisfies ContentResult;\n}\n\nfunction extractDescription(content: string, filePath: string): string {\n  const firstLine = content\n    .split(/\\r?\\n/)\n    .map((line) => line.trim())\n    .find((line) => line.length > 0);\n\n  return (\n    firstLine?.slice(0, 240) ??\n    `Documentation prompt loaded from ${path.basename(filePath)}`\n  );\n}\n\nfunction toTitleCase(value: string): string {\n  return value\n    .split(/\\s+/)\n    .filter(Boolean)\n    .map((part) => part.charAt(0).toUpperCase() + part.slice(1).toLowerCase())\n    .join(\" \");\n}\n\nfunction inferLanguageFromPath(filePath: string): string {\n  const extension = path.extname(filePath).toLowerCase();\n  switch (extension) {\n    case \".ts\":\n    case \".tsx\":\n      return \"ts\";\n    case \".js\":\n    case \".jsx\":\n      return \"js\";\n    case \".json\":\n      return \"json\";\n    case \".md\":\n      return \"md\";\n    default:\n      return \"text\";\n  }\n}\n\nexport function __resetWorkspaceRoot(root: string) {\n  setWorkspaceRoot(root);\n}\n\nexport { documentCodeSchema, codeChangeSchema };\n"]}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { FastMCP } from "fastmcp";
|
|
2
|
+
/**
|
|
3
|
+
* @description Function type for Decaf MCP modules
|
|
4
|
+
* @summary Defines the signature for MCP module functions that each Decaf module must export under the MCP_FILE_NAME file
|
|
5
|
+
* The function should return a Server object or a Promise that resolves to a Server object
|
|
6
|
+
*
|
|
7
|
+
* @typedef {Function} McpModule
|
|
8
|
+
* @return {Server|Promise<Server>} A Command object or Promise that resolves to a Server object
|
|
9
|
+
* @memberOf module:CLI
|
|
10
|
+
*/
|
|
11
|
+
export type McpModule = {
|
|
12
|
+
enrich(mcp: FastMCP): FastMCP | Promise<FastMCP>;
|
|
13
|
+
PACKAGE_NAME: string;
|
|
14
|
+
VERSION: string;
|
|
15
|
+
};
|
package/lib/esm/types.js
ADDED
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEZhc3RNQ1AgfSBmcm9tIFwiZmFzdG1jcFwiO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBGdW5jdGlvbiB0eXBlIGZvciBEZWNhZiBNQ1AgbW9kdWxlc1xuICogQHN1bW1hcnkgRGVmaW5lcyB0aGUgc2lnbmF0dXJlIGZvciBNQ1AgbW9kdWxlIGZ1bmN0aW9ucyB0aGF0IGVhY2ggRGVjYWYgbW9kdWxlIG11c3QgZXhwb3J0IHVuZGVyIHRoZSBNQ1BfRklMRV9OQU1FIGZpbGVcbiAqIFRoZSBmdW5jdGlvbiBzaG91bGQgcmV0dXJuIGEgU2VydmVyIG9iamVjdCBvciBhIFByb21pc2UgdGhhdCByZXNvbHZlcyB0byBhIFNlcnZlciBvYmplY3RcbiAqXG4gKiBAdHlwZWRlZiB7RnVuY3Rpb259IE1jcE1vZHVsZVxuICogQHJldHVybiB7U2VydmVyfFByb21pc2U8U2VydmVyPn0gQSBDb21tYW5kIG9iamVjdCBvciBQcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gYSBTZXJ2ZXIgb2JqZWN0XG4gKiBAbWVtYmVyT2YgbW9kdWxlOkNMSVxuICovXG5leHBvcnQgdHlwZSBNY3BNb2R1bGUgPSB7XG4gIGVucmljaChtY3A6IEZhc3RNQ1ApOiBGYXN0TUNQIHwgUHJvbWlzZTxGYXN0TUNQPjtcbiAgUEFDS0FHRV9OQU1FOiBzdHJpbmc7XG4gIFZFUlNJT046IHN0cmluZztcbn07XG4iXX0=
|
package/lib/esm/utils.d.ts
CHANGED
|
@@ -1,19 +1,60 @@
|
|
|
1
|
+
import { McpModule } from "./types";
|
|
1
2
|
/**
|
|
2
|
-
* @
|
|
3
|
-
* @
|
|
4
|
-
* @summary Concatenates "Hello World" with a given string. Despite its name, it's a simple string concatenation operation.
|
|
5
|
-
*
|
|
6
|
-
* @param {string} [arg1="default"] - The string to append to "Hello World". If not provided, defaults to "default".
|
|
7
|
-
* @return {string} The resulting concatenated string
|
|
3
|
+
* @description Utility class for CLI operations
|
|
4
|
+
* @summary A static utility class that provides methods for loading modules, retrieving package information, and initializing CLI commands
|
|
8
5
|
*
|
|
9
6
|
* @example
|
|
10
|
-
* //
|
|
11
|
-
*
|
|
7
|
+
* // Initialize a Command object with package information
|
|
8
|
+
* const command = new Command();
|
|
9
|
+
* CLIUtils.initialize(command, './path/to/package');
|
|
12
10
|
*
|
|
13
|
-
*
|
|
14
|
-
*
|
|
15
|
-
* complexFunction("!");
|
|
11
|
+
* // Load a CLI module from a file
|
|
12
|
+
* const module = await CLIUtils.loadFromFile('./path/to/cli-module.js');
|
|
16
13
|
*
|
|
17
|
-
* @
|
|
14
|
+
* @class McpUtils
|
|
18
15
|
*/
|
|
19
|
-
export declare
|
|
16
|
+
export declare class McpUtils {
|
|
17
|
+
/**
|
|
18
|
+
* @description Dynamically imports a module file
|
|
19
|
+
* @summary Loads a JavaScript file and returns it as a CliModule, handling both ESM and CommonJS formats
|
|
20
|
+
*
|
|
21
|
+
* @param {string} path The file path to the module to load
|
|
22
|
+
* @return {Promise<McpModule>} A promise that resolves to the loaded CliModule
|
|
23
|
+
*/
|
|
24
|
+
static loadFromFile(path: string): Promise<McpModule>;
|
|
25
|
+
/**
|
|
26
|
+
* @description Normalizes module imports to handle both ESM and CommonJS formats
|
|
27
|
+
* @summary Properly imports JavaScript files regardless of their module format by handling the ESM wrapper for CommonJS modules
|
|
28
|
+
*
|
|
29
|
+
* @template T The type of the imported module
|
|
30
|
+
* @param {Promise<T>} importPromise The promise returned by the dynamic import
|
|
31
|
+
* @return {Promise<T>} A promise that resolves to the normalized module
|
|
32
|
+
* @private
|
|
33
|
+
*/
|
|
34
|
+
static normalizeImport<T>(importPromise: Promise<T>): Promise<T>;
|
|
35
|
+
/**
|
|
36
|
+
* @description Retrieves and parses the package.json file
|
|
37
|
+
* @summary Reads the package.json file from the specified path and parses it into a JavaScript object
|
|
38
|
+
*
|
|
39
|
+
* @param {string} basePath The base path where the package.json file is located
|
|
40
|
+
* @return {Record<string, unknown>} The parsed package.json content as an object
|
|
41
|
+
* @private
|
|
42
|
+
*/
|
|
43
|
+
private static getPackage;
|
|
44
|
+
/**
|
|
45
|
+
* @description Returns the version from package.json
|
|
46
|
+
* @summary Retrieves the version field from the package.json file at the specified path
|
|
47
|
+
*
|
|
48
|
+
* @param {string} basePath The base path where the package.json file is located
|
|
49
|
+
* @return {string} The package version string
|
|
50
|
+
*/
|
|
51
|
+
static packageVersion(basePath: string): string;
|
|
52
|
+
/**
|
|
53
|
+
* @description Returns the name from package.json
|
|
54
|
+
* @summary Retrieves the name field from the package.json file at the specified path and extracts the package name without the scope
|
|
55
|
+
*
|
|
56
|
+
* @param {string} basePath The base path where the package.json file is located
|
|
57
|
+
* @return {string} The package name without the scope (e.g., "cli" from "@decaf-ts/cli")
|
|
58
|
+
*/
|
|
59
|
+
static packageName(basePath: string): string;
|
|
60
|
+
}
|