mcpkit
Provides a SQLite-backed CRUD example using better-sqlite3 with WAL mode, enabling persistent data operations.
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@mcpkitcreate a tool that adds two numbers"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
mcpkit
The TypeScript toolkit for building MCP servers without the boilerplate.
Define a tool with a Zod schema and a handler. Get a working Model Context Protocol server back — schema generation, input validation, error envelopes, transport wiring, all done.
import { defineServer, defineTool } from 'mcpkit';
import { z } from 'zod';
const server = defineServer({
name: 'demo',
version: '0.1.0',
tools: [
defineTool({
name: 'add',
description: 'Add two numbers.',
input: z.object({ a: z.number(), b: z.number() }),
handler: ({ a, b }) => `${a + b}`,
}),
],
});
await server.start();That's a real, functioning MCP server. Run it with mcpkit dev and point any
MCP-aware client at it.
why this exists
Writing an MCP server with the official SDK is fine, but you end up doing the same plumbing every time:
declaring the tool list in one place
declaring a separate JSON Schema for each tool
writing a switch over tool names in the call handler
coercing handler returns into the protocol's content envelope
wiring up a transport
catching errors and converting them into the right
isErrorshape
mcpkit collapses all of that into defineTool + defineServer. The schema
is generated from your Zod type, validation runs before your handler, errors
turn into proper protocol responses, and a string return becomes a text
content block. You stay in the layer that actually matters — what the tool
does — and skip the layer that doesn't.
with vs without
Same tool, written against the bare SDK and against mcpkit:
const server = new Server(
{ name: 'demo', version: '0.1.0' },
{ capabilities: { tools: {} } },
);
server.setRequestHandler(
ListToolsRequestSchema,
async () => ({
tools: [
{
name: 'add',
description: 'Add two numbers.',
inputSchema: {
type: 'object',
properties: {
a: { type: 'number' },
b: { type: 'number' },
},
required: ['a', 'b'],
},
},
],
}),
);
server.setRequestHandler(
CallToolRequestSchema,
async (req) => {
if (req.params.name === 'add') {
const { a, b } = req.params.arguments as {
a: number; b: number;
};
return {
content: [{ type: 'text', text: `${a + b}` }],
};
}
throw new Error('unknown tool');
},
);
await server.connect(new StdioServerTransport());const server = defineServer({
name: 'demo',
version: '0.1.0',
tools: [
defineTool({
name: 'add',
description: 'Add two numbers.',
input: z.object({
a: z.number(),
b: z.number(),
}),
handler: ({ a, b }) => `${a + b}`,
}),
],
});
await server.start();The right column has the same wire-level behavior, plus input validation,
plus typed handler arguments, plus an isError envelope on uncaught throws.
install
npm install mcpkit zodOr scaffold a fresh project (recommended for a first server):
npx mcpkit create my-server
cd my-server
npm run devYou'll get a small project with a working stdio server, three example tools,
and a tsconfig.json set up for strict mode. Replace the example tools with
yours and ship.
the cli
mcpkit create [target] scaffold a new server from a template
mcpkit dev run with hot reload (uses tsx under the hood)
mcpkit build compile to dist/
mcpkit inspect launch the official inspector against your servercreate ships with four templates today:
template | what you get |
| local MCP server over stdio. most clients want this. |
| network-reachable server over the streamable HTTP transport. |
| stdio server with HTTP-fetching tools (timeouts wired in). |
| stdio server with a SQLite-backed CRUD example (better-sqlite3, WAL). |
the api
defineTool
defineTool({
name: string, // [a-zA-Z0-9_-]+
description: string, // shown to the client / LLM
input: z.ZodType, // Zod schema; converted to JSON Schema for you
handler: (input) => string | ToolContent | ToolContent[] | { content, isError? }
})The handler input is fully typed via z.infer. Returning a string wraps it
as a single text content block — that's the common case. Throwing inside a
handler turns into an isError: true response automatically; if you want to
shape the error message, pass an onToolError handler to defineServer.
defineServer
defineServer({
name: string,
version: string,
description?: string,
tools?: ToolDefinition[],
resources?: ResourceDefinition[],
prompts?: PromptDefinition[],
onToolError?: (err, toolName) => ToolResult,
onEvent?: (event: ServerEvent) => void,
})Returns a DefinedServer with:
.start({ transport: 'stdio' })— connect a transport and serve..connect(transport)— connect a transport instance you constructed yourself (HTTP, custom, anything that quacks like aTransport)..stop()— close the active transport and the underlying server..raw— the underlying SDKServerif you need to do something exotic.
resources and prompts
Same declarative shape:
defineResource({
uri: 'file:///etc/hosts',
name: 'hosts',
mimeType: 'text/plain',
read: async () => ({ text: await fs.readFile('/etc/hosts', 'utf8') }),
});
definePrompt({
name: 'summarize',
description: 'Summarize a chunk of text.',
arguments: z.object({ text: z.string() }),
build: ({ text }) => ({
messages: [{ role: 'user', content: { type: 'text', text: `Summarize:\n${text}` } }],
}),
});observability
onEvent gets a structured callback for every tool call, resource read, and
prompt fetch — start time, end time, latency, error, a per-call requestId
to correlate. You can plug it into anything: pino, console, OpenTelemetry,
your homemade aggregator. There's also a built-in for the simple case:
import { defineServer, consoleLogger, jsonLogger } from 'mcpkit';
const server = defineServer({
name: 'demo',
version: '0.1.0',
onEvent: consoleLogger(), // → pretty stderr lines
// or: onEvent: jsonLogger() // → one JSON object per line, on stderr
tools: [...]
});Logging always goes to stderr — stdout is reserved for protocol traffic on stdio transports.
testing
mcpkit/testing exposes an in-process client that talks to your server
over an in-memory transport — no subprocess, no stdio piping, no flaky
process teardown. Same client a real consumer would use, just routed through
RAM.
import { describe, it, expect } from 'vitest';
import { createTestClient, expectToolError, snapshotTools } from 'mcpkit/testing';
import { server } from '../src/index.js';
describe('add', () => {
it('adds', async () => {
const client = await createTestClient(server);
const result = await client.callTool('add', { a: 2, b: 3 });
expect(result.text).toBe('5');
expect(result.isError).toBe(false);
await client.close();
});
it('rejects bad input', async () => {
const client = await createTestClient(server);
const text = await expectToolError(client, 'add', { a: 'nope', b: 1 });
expect(text).toMatch(/invalid/i);
await client.close();
});
it("doesn't drift its public surface", () => {
expect(snapshotTools(server)).toMatchSnapshot();
});
});design choices worth knowing
Zod, not raw JSON Schema. You write the type once. Validation, generated JSON Schema for the protocol, and TypeScript inference for the handler all fall out of the same source. Trying to keep three definitions in sync is the boilerplate this project exists to delete.
Errors are values, not exceptions. A handler that throws becomes an
isError: true content envelope. The client sees a sensible response instead
of a transport-level failure. If you'd rather format the error yourself,
override onToolError.
Transport-agnostic core. The same defineServer works over stdio, the
streamable HTTP transport, the in-memory test transport, or anything else
that implements the SDK's Transport interface. The http-streaming
template shows the wiring.
Strict mode by default. Templates ship with strict: true and
noUncheckedIndexedAccess. The library itself compiles under the same
settings. If you find a hole in the types, that's a bug.
Listener errors are swallowed. If your onEvent handler throws, your
tool calls keep working. Observability bugs shouldn't be load-bearing.
faq
Does this lock me into mcpkit forever?
No. Every helper has an escape hatch — server.raw gives you the underlying
SDK Server, and you can setRequestHandler on it directly if you need
something the kit doesn't model yet. The kit is a layer on top, not a
replacement.
Why Zod 3 and not 4?
Zod 4 is great but the ecosystem (notably zod-to-json-schema) is still
catching up. We'll move when it's stable in production. If you're already
on Zod 4, the schema interfaces are compatible enough — file an issue if you
hit a wall.
Does it support resources and prompts, not just tools?
Yes. defineResource and definePrompt are first-class. They're less
commonly used than tools, so most examples lead with tools — but the wiring
is identical.
Streamable HTTP, SSE, both?
Streamable HTTP. The older HTTP+SSE flavor is still in the SDK but is being
phased out — if you have a reason to need it, defineServer is transport-
agnostic and you can pass any Transport instance via .connect().
Production-ready? The library is small and the surface is intentionally narrow. The official SDK does the heavy lifting underneath. Pin a version, write tests for your tools (the in-process client makes this easy), and you're set.
what this is not
not a hosted service. you build, you deploy.
not an agent framework. it builds the server side of MCP, not the client.
not opinionated about your domain. tools are functions; what they do is your problem.
roadmap
more templates (oauth-protected, edge runtime, drizzle/postgres).
a
mcpkit publishcommand that lints + packages + tags a release.richer testing helpers (fuzz a tool's input, schema diff against a baseline).
optional OpenTelemetry adapter for
onEvent.
If something's missing, open an issue with a sketch of the API you'd want.
license
MIT.
This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
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/EuKennedy/mcpkit'
If you have feedback or need assistance with the MCP directory API, please join our Discord server