Skip to content

Commit 91101ee

Browse files
committed
feat: add cli tool
1 parent 85624c6 commit 91101ee

11 files changed

+684
-17
lines changed

package-lock.json

+28-9
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+6-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "mcp-framework",
3-
"version": "0.1.8",
3+
"version": "0.1.9",
44
"description": "Framework for building Model Context Protocol (MCP) servers in Typescript",
55
"type": "module",
66
"author": "Alex Andru <[email protected]>",
@@ -16,7 +16,8 @@
1616
"dist"
1717
],
1818
"bin": {
19-
"mcp-build": "./dist/cli/build.js"
19+
"mcp": "./dist/cli/index.js",
20+
"mcp-build": "./dist/cli/framework/build.js"
2021
},
2122
"scripts": {
2223
"build": "tsc",
@@ -36,6 +37,9 @@
3637
"@modelcontextprotocol/sdk": "^0.6.0"
3738
},
3839
"dependencies": {
40+
"@types/prompts": "^2.4.9",
41+
"commander": "^12.1.0",
42+
"prompts": "^2.4.2",
3943
"zod": "^3.23.8"
4044
},
4145
"devDependencies": {

src/cli/build.ts src/cli/framework/build.ts

+6-6
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,12 @@ import { spawnSync } from "child_process";
33
import { readFileSync, writeFileSync } from "fs";
44
import { join } from "path";
55

6+
export function buildFramework() {
7+
runTsc();
8+
addShebang();
9+
console.log("MCP Build complete");
10+
}
11+
612
function runTsc() {
713
const tscPath = join(process.cwd(), "node_modules", ".bin", "tsc");
814
const tsc = spawnSync(tscPath, [], {
@@ -30,9 +36,3 @@ function addShebang() {
3036
process.exit(1);
3137
}
3238
}
33-
34-
runTsc();
35-
36-
addShebang();
37-
38-
console.log("MCP Build complete");

src/cli/index.ts

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
#!/usr/bin/env node
2+
import { Command } from "commander";
3+
import { createProject } from "./project/create.js";
4+
import { addTool } from "./project/add-tool.js";
5+
import { addPrompt } from "./project/add-prompt.js";
6+
import { addResource } from "./project/add-resource.js";
7+
import { buildFramework } from "./framework/build.js";
8+
9+
const program = new Command();
10+
11+
program
12+
.name("mcp")
13+
.description("CLI for managing MCP server projects")
14+
.version("0.1.8");
15+
16+
program
17+
.command("build-framework", { hidden: true })
18+
.description("Build the MCP framework")
19+
.action(buildFramework);
20+
21+
program
22+
.command("create")
23+
.description("Create a new MCP server project")
24+
.argument("[name]", "project name")
25+
.action(createProject);
26+
27+
program
28+
.command("add")
29+
.description("Add a new component to your MCP server")
30+
.addCommand(
31+
new Command("tool")
32+
.description("Add a new tool")
33+
.argument("[name]", "tool name")
34+
.action(addTool)
35+
)
36+
.addCommand(
37+
new Command("prompt")
38+
.description("Add a new prompt")
39+
.argument("[name]", "prompt name")
40+
.action(addPrompt)
41+
)
42+
.addCommand(
43+
new Command("resource")
44+
.description("Add a new resource")
45+
.argument("[name]", "resource name")
46+
.action(addResource)
47+
);
48+
49+
program.parse();

src/cli/project/add-prompt.ts

+86
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
import { writeFile, mkdir } from "fs/promises";
2+
import { join } from "path";
3+
import prompts from "prompts";
4+
import { toPascalCase } from "../utils/string-utils.js";
5+
import { validateMCPProject } from "../utils/validate-project.js";
6+
7+
export async function addPrompt(name?: string) {
8+
await validateMCPProject();
9+
10+
let promptName = name;
11+
if (!promptName) {
12+
const response = await prompts([
13+
{
14+
type: "text",
15+
name: "name",
16+
message: "What is the name of your prompt?",
17+
validate: (value: string) =>
18+
/^[a-z0-9-]+$/.test(value)
19+
? true
20+
: "Prompt name can only contain lowercase letters, numbers, and hyphens",
21+
},
22+
]);
23+
24+
if (!response.name) {
25+
console.log("Prompt creation cancelled");
26+
process.exit(1);
27+
}
28+
29+
promptName = response.name;
30+
}
31+
32+
if (!promptName) {
33+
throw new Error("Prompt name is required");
34+
}
35+
36+
const className = toPascalCase(promptName);
37+
const fileName = `${className}Prompt.ts`;
38+
const promptsDir = join(process.cwd(), "src/prompts");
39+
40+
try {
41+
await mkdir(promptsDir, { recursive: true });
42+
43+
const promptContent = `import { MCPPrompt } from "mcp-framework";
44+
import { z } from "zod";
45+
46+
interface ${className}Input {
47+
message: string;
48+
}
49+
50+
class ${className}Prompt extends MCPPrompt<${className}Input> {
51+
name = "${promptName}";
52+
description = "${className} prompt description";
53+
54+
schema = {
55+
message: {
56+
type: z.string(),
57+
description: "Message to process",
58+
required: true,
59+
},
60+
};
61+
62+
async generateMessages({ message }: ${className}Input) {
63+
return [
64+
{
65+
role: "user",
66+
content: {
67+
type: "text",
68+
text: message,
69+
},
70+
},
71+
];
72+
}
73+
}
74+
75+
export default ${className}Prompt;`;
76+
77+
await writeFile(join(promptsDir, fileName), promptContent);
78+
79+
console.log(
80+
`Prompt ${promptName} created successfully at src/prompts/${fileName}`
81+
);
82+
} catch (error) {
83+
console.error("Error creating prompt:", error);
84+
process.exit(1);
85+
}
86+
}

src/cli/project/add-resource.ts

+73
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
import { writeFile, mkdir } from "fs/promises";
2+
import { join } from "path";
3+
import prompts from "prompts";
4+
import { validateMCPProject } from "../utils/validate-project.js";
5+
import { toPascalCase } from "../utils/string-utils.js";
6+
7+
export async function addResource(name?: string) {
8+
await validateMCPProject();
9+
10+
let resourceName = name;
11+
if (!resourceName) {
12+
const response = await prompts([
13+
{
14+
type: "text",
15+
name: "name",
16+
message: "What is the name of your resource?",
17+
validate: (value: string) =>
18+
/^[a-z0-9-]+$/.test(value)
19+
? true
20+
: "Resource name can only contain lowercase letters, numbers, and hyphens",
21+
},
22+
]);
23+
24+
if (!response.name) {
25+
console.log("Resource creation cancelled");
26+
process.exit(1);
27+
}
28+
29+
resourceName = response.name;
30+
}
31+
32+
if (!resourceName) {
33+
throw new Error("Resource name is required");
34+
}
35+
36+
const className = toPascalCase(resourceName);
37+
const fileName = `${className}Resource.ts`;
38+
const resourcesDir = join(process.cwd(), "src/resources");
39+
40+
try {
41+
await mkdir(resourcesDir, { recursive: true });
42+
43+
const resourceContent = `import { MCPResource, ResourceContent } from "mcp-framework";
44+
45+
class ${className}Resource extends MCPResource {
46+
uri = "resource://${resourceName}";
47+
name = "${className}";
48+
description = "${className} resource description";
49+
mimeType = "application/json";
50+
51+
async read(): Promise<ResourceContent[]> {
52+
return [
53+
{
54+
uri: this.uri,
55+
mimeType: this.mimeType,
56+
text: JSON.stringify({ message: "Hello from ${className} resource" }),
57+
},
58+
];
59+
}
60+
}
61+
62+
export default ${className}Resource;`;
63+
64+
await writeFile(join(resourcesDir, fileName), resourceContent);
65+
66+
console.log(
67+
`Resource ${resourceName} created successfully at src/resources/${fileName}`
68+
);
69+
} catch (error) {
70+
console.error("Error creating resource:", error);
71+
process.exit(1);
72+
}
73+
}

0 commit comments

Comments
 (0)