MCP Starter Kit
MCP Server Starter Kit
A production-ready TypeScript template for building Model Context Protocol (MCP) servers. Skip the boilerplate and ship working tools to Claude and other MCP clients in minutes.
What's included
Working MCP server using the official
@modelcontextprotocol/sdk3 example tools you can use as-is or adapt:
fetch_url— fetch web content with configurable limits and domain blockingread_file/list_directory— safe filesystem access with path traversal protectiontransform_data— convert between JSON, CSV, TSV, Markdown table, and plain text
TypeScript throughout — strict mode, typed inputs/outputs, Zod validation
Error handling patterns — every tool returns a typed
ToolResult<T>with ok/error discriminationEnvironment-based config — all limits and paths configurable via
.envStructured logging — stderr-only logger (MCP protocol uses stdout)
Test suite — 19 tests with Vitest covering all three tools
Build scripts —
npm run build,npm run dev,npm test,npm run typecheck
Requirements
Node.js 18 or higher
npm 9 or higher
Quick start
# 1. Install dependencies
npm install
# 2. Configure environment
cp .env.example .env
# Edit .env — at minimum, set FILE_READER_ROOT to a safe directory
# 3. Build
npm run build
# 4. Run
npm startDevelopment mode
npm run devUses tsx for live reload — no build step required during development.
Connect to Claude Desktop
Add this to your Claude Desktop config (~/Library/Application Support/Claude/claude_desktop_config.json on macOS, %APPDATA%\Claude\claude_desktop_config.json on Windows):
{
"mcpServers": {
"my-server": {
"command": "node",
"args": ["/absolute/path/to/mcp-starter-kit/dist/index.js"],
"env": {
"FILE_READER_ROOT": "/path/to/allowed/directory",
"LOG_LEVEL": "info"
}
}
}
}Restart Claude Desktop. Your tools will appear in the tool picker.
Connect to Claude Code
Add to .claude/settings.json:
{
"mcpServers": {
"my-server": {
"command": "node",
"args": ["/absolute/path/to/mcp-starter-kit/dist/index.js"]
}
}
}Tools reference
fetch_url
Fetches the text content of a URL.
Parameter | Type | Required | Description |
| string | yes | HTTP or HTTPS URL to fetch |
| object | no | Additional request headers |
| number | no | Request timeout (100–30000ms, default from env) |
Returns the response body, status code, content type, and a truncated flag if the response exceeded FETCH_MAX_BYTES.
read_file
Reads a file within the configured FILE_READER_ROOT.
Parameter | Type | Required | Description |
| string | yes | Relative path from root |
|
| no | Encoding (default: utf8) |
| number | no | Max bytes to read (default: 1MB) |
Path traversal (../) is blocked at the resolver level.
list_directory
Lists files and directories within the configured root.
Parameter | Type | Required | Description |
| string | no | Relative directory path (default: |
| boolean | no | List nested files (default: false) |
transform_data
Converts data between formats.
Parameter | Type | Required | Description |
| string | yes | Raw input data |
|
| yes | Input format |
|
| yes | Output format |
| boolean | no | Pretty-print JSON (default: true) |
| boolean | no | Include CSV/TSV header row (default: true) |
| string | no | Custom delimiter for CSV/TSV parsing |
Configuration
All configuration is via environment variables. See .env.example for the full list.
Variable | Default | Description |
|
| Server identity reported to clients |
|
| Server version |
|
| Max response size for web fetcher (bytes) |
|
| Default fetch timeout (ms) |
| (empty) | Comma-separated blocked hostnames |
|
| Root directory for file access |
|
| Max input characters for transformer |
|
| Logging level (debug/info/warn/error) |
Adding your own tools
Create
src/tools/my-tool.ts— export an async function that returnsToolResult<YourType>Add input/output types to
src/types.tsusing Zod schemasRegister the tool in
src/index.tswithserver.tool(name, description, schema, handler)Write tests in
src/tools/my-tool.test.ts
The pattern used by all three example tools:
export async function myTool(input: MyToolInput): Promise<ToolResult<MyToolOutput>> {
// validate, execute, return { ok: true, data: ... } or { ok: false, error: "...", code: "..." }
}Project structure
mcp-starter-kit/
├── src/
│ ├── index.ts # Server entry point — tool registration
│ ├── config.ts # Environment variable loading
│ ├── logger.ts # Stderr logger
│ ├── types.ts # Shared types and Zod schemas
│ └── tools/
│ ├── web-fetcher.ts
│ ├── web-fetcher.test.ts (add your own)
│ ├── file-reader.ts
│ ├── file-reader.test.ts
│ ├── data-transformer.ts
│ └── data-transformer.test.ts
├── dist/ # Compiled output (after npm run build)
├── .env.example
├── package.json
├── tsconfig.json
└── vitest.config.tsRunning tests
npm test # Run once
npm run test:watch # Watch modeLicense
MIT
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/idapixl/mcp-starter-kit'
If you have feedback or need assistance with the MCP directory API, please join our Discord server