152 lines
4.6 KiB
JavaScript
152 lines
4.6 KiB
JavaScript
#!/usr/bin/env node
|
|
|
|
// src/bin/fastmcp.ts
|
|
import { execa } from "execa";
|
|
import yargs from "yargs";
|
|
import { hideBin } from "yargs/helpers";
|
|
await yargs(hideBin(process.argv)).scriptName("fastmcp").command(
|
|
"dev <file>",
|
|
"Start a development server",
|
|
(yargs2) => {
|
|
return yargs2.positional("file", {
|
|
demandOption: true,
|
|
describe: "The path to the server file",
|
|
type: "string"
|
|
}).option("watch", {
|
|
alias: "w",
|
|
default: false,
|
|
describe: "Watch for file changes and restart server",
|
|
type: "boolean"
|
|
}).option("verbose", {
|
|
alias: "v",
|
|
default: false,
|
|
describe: "Enable verbose logging",
|
|
type: "boolean"
|
|
});
|
|
},
|
|
async (argv) => {
|
|
try {
|
|
const command = argv.watch ? `npx @wong2/mcp-cli npx tsx --watch ${argv.file}` : `npx @wong2/mcp-cli npx tsx ${argv.file}`;
|
|
if (argv.verbose) {
|
|
console.log(`[FastMCP] Starting server: ${command}`);
|
|
console.log(`[FastMCP] File: ${argv.file}`);
|
|
console.log(
|
|
`[FastMCP] Watch mode: ${argv.watch ? "enabled" : "disabled"}`
|
|
);
|
|
}
|
|
await execa({
|
|
shell: true,
|
|
stderr: "inherit",
|
|
stdin: "inherit",
|
|
stdout: "inherit"
|
|
})`${command}`;
|
|
} catch (error) {
|
|
console.error(
|
|
"[FastMCP Error] Failed to start development server:",
|
|
error instanceof Error ? error.message : String(error)
|
|
);
|
|
if (argv.verbose && error instanceof Error && error.stack) {
|
|
console.error("[FastMCP Debug] Stack trace:", error.stack);
|
|
}
|
|
process.exit(1);
|
|
}
|
|
}
|
|
).command(
|
|
"inspect <file>",
|
|
"Inspect a server file",
|
|
(yargs2) => {
|
|
return yargs2.positional("file", {
|
|
demandOption: true,
|
|
describe: "The path to the server file",
|
|
type: "string"
|
|
});
|
|
},
|
|
async (argv) => {
|
|
try {
|
|
await execa({
|
|
stderr: "inherit",
|
|
stdout: "inherit"
|
|
})`npx @modelcontextprotocol/inspector npx tsx ${argv.file}`;
|
|
} catch (error) {
|
|
console.error(
|
|
"[FastMCP Error] Failed to inspect server:",
|
|
error instanceof Error ? error.message : String(error)
|
|
);
|
|
process.exit(1);
|
|
}
|
|
}
|
|
).command(
|
|
"validate <file>",
|
|
"Validate a FastMCP server file for syntax and basic structure",
|
|
(yargs2) => {
|
|
return yargs2.positional("file", {
|
|
demandOption: true,
|
|
describe: "The path to the server file",
|
|
type: "string"
|
|
}).option("strict", {
|
|
alias: "s",
|
|
default: false,
|
|
describe: "Enable strict validation (type checking)",
|
|
type: "boolean"
|
|
});
|
|
},
|
|
async (argv) => {
|
|
try {
|
|
const { existsSync } = await import("fs");
|
|
const { resolve } = await import("path");
|
|
const filePath = resolve(argv.file);
|
|
if (!existsSync(filePath)) {
|
|
console.error(`[FastMCP Error] File not found: ${filePath}`);
|
|
process.exit(1);
|
|
}
|
|
console.log(`[FastMCP] Validating server file: ${filePath}`);
|
|
const command = argv.strict ? `npx tsc --noEmit --strict ${filePath}` : `npx tsc --noEmit ${filePath}`;
|
|
try {
|
|
await execa({
|
|
shell: true,
|
|
stderr: "pipe",
|
|
stdout: "pipe"
|
|
})`${command}`;
|
|
console.log("[FastMCP] \u2713 TypeScript compilation successful");
|
|
} catch (tsError) {
|
|
console.error("[FastMCP] \u2717 TypeScript compilation failed");
|
|
if (tsError instanceof Error && "stderr" in tsError) {
|
|
console.error(tsError.stderr);
|
|
}
|
|
process.exit(1);
|
|
}
|
|
try {
|
|
await execa({
|
|
shell: true,
|
|
stderr: "pipe",
|
|
stdout: "pipe"
|
|
})`node -e "
|
|
(async () => {
|
|
try {
|
|
const { FastMCP } = await import('fastmcp');
|
|
await import('file://${filePath}');
|
|
console.log('[FastMCP] ✓ Server structure validation passed');
|
|
} catch (error) {
|
|
console.error('[FastMCP] ✗ Server structure validation failed:', error.message);
|
|
process.exit(1);
|
|
}
|
|
})();
|
|
"`;
|
|
} catch {
|
|
console.error("[FastMCP] \u2717 Server structure validation failed");
|
|
console.error("Make sure the file properly imports and uses FastMCP");
|
|
process.exit(1);
|
|
}
|
|
console.log(
|
|
"[FastMCP] \u2713 All validations passed! Server file looks good."
|
|
);
|
|
} catch (error) {
|
|
console.error(
|
|
"[FastMCP Error] Validation failed:",
|
|
error instanceof Error ? error.message : String(error)
|
|
);
|
|
process.exit(1);
|
|
}
|
|
}
|
|
).help().parseAsync();
|
|
//# sourceMappingURL=fastmcp.js.map
|