<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Claude-Alain Martin</title>
    <description>The latest articles on DEV Community by Claude-Alain Martin (@cammaccreator).</description>
    <link>https://dev.to/cammaccreator</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3860566%2Febeab91d-ec3f-4846-b8e1-fe5fca9b6a5d.png</url>
      <title>DEV Community: Claude-Alain Martin</title>
      <link>https://dev.to/cammaccreator</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/cammaccreator"/>
    <language>en</language>
    <item>
      <title>I built an MCP server without the @modelcontextprotocol/sdk — here's what I learned</title>
      <dc:creator>Claude-Alain Martin</dc:creator>
      <pubDate>Tue, 12 May 2026 13:00:51 +0000</pubDate>
      <link>https://dev.to/cammaccreator/i-built-an-mcp-server-without-the-modelcontextprotocolsdk-heres-what-i-learned-4eo0</link>
      <guid>https://dev.to/cammaccreator/i-built-an-mcp-server-without-the-modelcontextprotocolsdk-heres-what-i-learned-4eo0</guid>
      <description>&lt;p&gt;I shipped a Model Context Protocol server last month. It's live on Anthropic's official registry as &lt;code&gt;io.github.cammac-creator/openswissdata&lt;/code&gt;. It exposes nine tools over JSON-RPC 2.0 to any MCP-compatible client (Claude Desktop, Cursor, Cline, you name it).&lt;/p&gt;

&lt;p&gt;I built it without &lt;code&gt;@modelcontextprotocol/sdk&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Not because the SDK is bad — it isn't. But because for my stack, the friction of fitting it in wasn't paid back by any feature I needed. This post is the honest write-up of that decision, the wire protocol I had to implement instead, and why I'd do it the same way again.&lt;/p&gt;

&lt;h2&gt;
  
  
  The hook: SDK ≠ obligation
&lt;/h2&gt;

&lt;p&gt;There's an implicit assumption when you discover a new protocol: "I need to install the SDK." The SDK is the blessed path. It's in the docs. It's what every blog post imports on line 1.&lt;/p&gt;

&lt;p&gt;But an SDK is a particular set of tradeoffs frozen into code. Sometimes those tradeoffs match yours. Sometimes they don't. The MCP wire protocol — the thing the SDK ultimately speaks to clients — is published, small, and stable. So before reaching for the import, it's worth asking: what does the SDK &lt;em&gt;give me&lt;/em&gt; that I can't get by reading the spec?&lt;/p&gt;

&lt;p&gt;For my project, the honest answer was: nothing I needed for the MVP.&lt;/p&gt;

&lt;h2&gt;
  
  
  What the SDK actually wants from you
&lt;/h2&gt;

&lt;p&gt;The &lt;code&gt;@modelcontextprotocol/sdk&lt;/code&gt; exports a &lt;code&gt;Server&lt;/code&gt; class. To use it, you give it two things: a server identity, and a &lt;code&gt;Transport&lt;/code&gt;. The Transport is where the friction lives.&lt;/p&gt;

&lt;p&gt;There are basically two production transports:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;STDIO&lt;/strong&gt; — designed for child processes spawned by desktop clients. Claude Desktop or Cursor launch your server as a subprocess and talk to it over stdin/stdout. Not what you want if you're hosting a public HTTP endpoint.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;StreamableHTTPServerTransport&lt;/strong&gt; — the SDK's HTTP transport. It expects Express-style &lt;code&gt;req&lt;/code&gt;/&lt;code&gt;res&lt;/code&gt; objects and runs its own session manager. Sessions, SSE channels, the whole envelope.&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;I run my backend on &lt;strong&gt;Hono&lt;/strong&gt; (because it's tiny, edge-friendly, and fits on Bun/Node/workers). Hono doesn't speak Express's &lt;code&gt;req&lt;/code&gt;/&lt;code&gt;res&lt;/code&gt; shape — it gives you a &lt;code&gt;Context&lt;/code&gt; with &lt;code&gt;c.req.json()&lt;/code&gt; and &lt;code&gt;c.json()&lt;/code&gt;. So integrating &lt;code&gt;StreamableHTTPServerTransport&lt;/code&gt; would mean either (a) adapting between Hono and Express semantics inside every request, or (b) carving out a side-channel &lt;code&gt;http.createServer()&lt;/code&gt; listener just for MCP. Both are uglier than what I ended up doing.&lt;/p&gt;

&lt;p&gt;And the SSE / session resumption that the StreamableHTTPServerTransport exists to provide? My nine tools are all synchronous request/response. There's nothing to stream. No long-running task to resume. The transport's value proposition didn't apply to my use case.&lt;/p&gt;

&lt;p&gt;So I read the spec.&lt;/p&gt;

&lt;h2&gt;
  
  
  The wire protocol, minus the marketing
&lt;/h2&gt;

&lt;p&gt;The MCP spec at &lt;a href="https://modelcontextprotocol.io" rel="noopener noreferrer"&gt;modelcontextprotocol.io&lt;/a&gt; is more approachable than people assume. For a tools-only server like mine, you need to handle exactly three JSON-RPC 2.0 methods:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;initialize&lt;/code&gt; — client says hi, you respond with your protocol version and capabilities&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;tools/list&lt;/code&gt; — client asks what you can do, you enumerate registered tools&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;tools/call&lt;/code&gt; — client invokes a tool by name with arguments, you dispatch and return&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Plus the standard JSON-RPC 2.0 error envelope (&lt;code&gt;-32700&lt;/code&gt; parse error, &lt;code&gt;-32600&lt;/code&gt; invalid request, &lt;code&gt;-32601&lt;/code&gt; method not found, &lt;code&gt;-32602&lt;/code&gt; invalid params, &lt;code&gt;-32603&lt;/code&gt; internal error).&lt;/p&gt;

&lt;p&gt;That's it. That's the protocol surface. Maybe 80 lines of types and 100 lines of dispatch logic.&lt;/p&gt;

&lt;h2&gt;
  
  
  The minimalist code
&lt;/h2&gt;

&lt;p&gt;Here's the entire JSON-RPC error/response shape — all you need to wire-conform:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// 5 standard JSON-RPC 2.0 error codes — the whole error surface&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;ERR&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="na"&gt;PARSE_ERROR&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;32700&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="na"&gt;INVALID_REQUEST&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;32600&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="na"&gt;METHOD_NOT_FOUND&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;32601&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="na"&gt;INVALID_PARAMS&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;32602&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="na"&gt;INTERNAL_ERROR&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;32603&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;as&lt;/span&gt; &lt;span class="kd"&gt;const&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="c1"&gt;// Two helpers — every response goes through one of these&lt;/span&gt;
&lt;span class="kd"&gt;function&lt;/span&gt; &lt;span class="nf"&gt;err&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;id&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;code&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;message&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;data&lt;/span&gt;&lt;span class="p"&gt;?)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;jsonrpc&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;2.0&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;id&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;error&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;code&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;message&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;...(&lt;/span&gt;&lt;span class="nx"&gt;data&lt;/span&gt; &lt;span class="o"&gt;!==&lt;/span&gt; &lt;span class="kc"&gt;undefined&lt;/span&gt; &lt;span class="p"&gt;?&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;data&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{})&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="p"&gt;};&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;span class="kd"&gt;function&lt;/span&gt; &lt;span class="nf"&gt;ok&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;id&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;result&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;jsonrpc&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;2.0&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;id&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;result&lt;/span&gt; &lt;span class="p"&gt;};&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Tools are just an interface — name, description, JSON schema for inputs, and a handler:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="kr"&gt;interface&lt;/span&gt; &lt;span class="nx"&gt;Tool&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="nl"&gt;name&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="kr"&gt;string&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
  &lt;span class="nl"&gt;description&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="kr"&gt;string&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
  &lt;span class="nl"&gt;inputSchema&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nb"&gt;Readonly&lt;/span&gt;&lt;span class="o"&gt;&amp;lt;&lt;/span&gt;&lt;span class="nb"&gt;Record&lt;/span&gt;&lt;span class="o"&gt;&amp;lt;&lt;/span&gt;&lt;span class="kr"&gt;string&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;unknown&lt;/span&gt;&lt;span class="o"&gt;&amp;gt;&amp;gt;&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
  &lt;span class="c1"&gt;// Handlers may be sync (CSV lookup) or async (embedding pipeline).&lt;/span&gt;
  &lt;span class="c1"&gt;// The dispatch layer awaits the return value uniformly.&lt;/span&gt;
  &lt;span class="nl"&gt;handler&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;args&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;unknown&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nx"&gt;ToolResult&lt;/span&gt; &lt;span class="o"&gt;|&lt;/span&gt; &lt;span class="nb"&gt;Promise&lt;/span&gt;&lt;span class="o"&gt;&amp;lt;&lt;/span&gt;&lt;span class="nx"&gt;ToolResult&lt;/span&gt;&lt;span class="o"&gt;&amp;gt;&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;TOOLS&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="k"&gt;readonly&lt;/span&gt; &lt;span class="nx"&gt;Tool&lt;/span&gt;&lt;span class="p"&gt;[]&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;[&lt;/span&gt;
  &lt;span class="nx"&gt;tariffLookupTool&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="nx"&gt;kycCheckTool&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="nx"&gt;crossWalkTool&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="c1"&gt;// ... six more&lt;/span&gt;
&lt;span class="p"&gt;]&lt;/span&gt; &lt;span class="k"&gt;as&lt;/span&gt; &lt;span class="kd"&gt;const&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;TOOLS_BY_NAME&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nb"&gt;Object&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;fromEntries&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;TOOLS&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;map&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;t&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="nx"&gt;t&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;name&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;t&lt;/span&gt;&lt;span class="p"&gt;]));&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;The &lt;code&gt;initialize&lt;/code&gt; handler advertises what you support. The spec is loud about &lt;code&gt;protocolVersion&lt;/code&gt; — pin it to a known date:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;PROTOCOL_VERSION&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;2025-06-18&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;SERVER_INFO&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;name&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;openswissdata-mcp&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;version&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;0.2.0&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;as&lt;/span&gt; &lt;span class="kd"&gt;const&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="c1"&gt;// In the dispatch switch:&lt;/span&gt;
&lt;span class="k"&gt;case&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;initialize&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nf"&gt;ok&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;id&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="na"&gt;protocolVersion&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;PROTOCOL_VERSION&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="na"&gt;serverInfo&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="p"&gt;...&lt;/span&gt;&lt;span class="nx"&gt;SERVER_INFO&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt;
    &lt;span class="na"&gt;capabilities&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;tools&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;listChanged&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="kc"&gt;false&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt;
  &lt;span class="p"&gt;});&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;code&gt;tools/list&lt;/code&gt; is a one-liner over the registry — clients use this to discover what's available, so the schema you emit here is the contract:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="kd"&gt;function&lt;/span&gt; &lt;span class="nf"&gt;listTools&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="na"&gt;tools&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;TOOLS&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;map&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;t&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;({&lt;/span&gt;
      &lt;span class="na"&gt;name&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;t&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;name&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
      &lt;span class="na"&gt;description&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;t&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;description&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
      &lt;span class="na"&gt;inputSchema&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;t&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;inputSchema&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="p"&gt;})),&lt;/span&gt;
  &lt;span class="p"&gt;};&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;code&gt;tools/call&lt;/code&gt; is where the actual work happens. Validate, look up, dispatch:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="k"&gt;case&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;tools/call&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;params&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;r&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;params&lt;/span&gt; &lt;span class="k"&gt;as&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;name&lt;/span&gt;&lt;span class="p"&gt;?:&lt;/span&gt; &lt;span class="kr"&gt;string&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt; &lt;span class="nl"&gt;arguments&lt;/span&gt;&lt;span class="p"&gt;?:&lt;/span&gt; &lt;span class="nx"&gt;unknown&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="o"&gt;|&lt;/span&gt; &lt;span class="kc"&gt;undefined&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
  &lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="o"&gt;!&lt;/span&gt;&lt;span class="nx"&gt;params&lt;/span&gt; &lt;span class="o"&gt;||&lt;/span&gt; &lt;span class="k"&gt;typeof&lt;/span&gt; &lt;span class="nx"&gt;params&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;name&lt;/span&gt; &lt;span class="o"&gt;!==&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;string&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nf"&gt;err&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;id&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;ERR&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;INVALID_PARAMS&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;params.name (string) is required&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
  &lt;span class="p"&gt;}&lt;/span&gt;
  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;tool&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;TOOLS_BY_NAME&lt;/span&gt;&lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="nx"&gt;params&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;name&lt;/span&gt;&lt;span class="p"&gt;];&lt;/span&gt;
  &lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="o"&gt;!&lt;/span&gt;&lt;span class="nx"&gt;tool&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nf"&gt;err&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;id&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;ERR&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;METHOD_NOT_FOUND&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="s2"&gt;`Unknown tool: &lt;/span&gt;&lt;span class="p"&gt;${&lt;/span&gt;&lt;span class="nx"&gt;params&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;name&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="s2"&gt;`&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
  &lt;span class="p"&gt;}&lt;/span&gt;
  &lt;span class="c1"&gt;// Always await — handler can be sync or Promise-returning; await of a&lt;/span&gt;
  &lt;span class="c1"&gt;// non-promise is a no-op, so this is safe and uniform.&lt;/span&gt;
  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;result&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;tool&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;handler&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;params&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;arguments&lt;/span&gt; &lt;span class="o"&gt;??&lt;/span&gt; &lt;span class="p"&gt;{});&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nf"&gt;ok&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;id&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;result&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;And mounting it inside Hono is one route:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;Hono&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;hono&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;dispatch&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;../../mcp/server.js&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="k"&gt;export&lt;/span&gt; &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;mcpRoute&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Hono&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;

&lt;span class="nx"&gt;mcpRoute&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;post&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;/jsonrpc&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="k"&gt;async &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;c&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="kd"&gt;let&lt;/span&gt; &lt;span class="nx"&gt;body&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
  &lt;span class="k"&gt;try&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;body&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;c&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;req&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;json&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt;
  &lt;span class="k"&gt;catch&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;c&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;json&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;
      &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;jsonrpc&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;2.0&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;id&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="kc"&gt;null&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;error&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;code&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="o"&gt;-&lt;/span&gt;&lt;span class="mi"&gt;32700&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;message&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Parse error&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt;
      &lt;span class="mi"&gt;400&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="p"&gt;);&lt;/span&gt;
  &lt;span class="p"&gt;}&lt;/span&gt;
  &lt;span class="c1"&gt;// JSON-RPC 2.0 supports batched requests — handle both shapes&lt;/span&gt;
  &lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nb"&gt;Array&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;isArray&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;body&lt;/span&gt;&lt;span class="p"&gt;))&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;c&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;json&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nb"&gt;Promise&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;all&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;body&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;map&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;r&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nf"&gt;dispatch&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;r&lt;/span&gt;&lt;span class="p"&gt;))));&lt;/span&gt;
  &lt;span class="p"&gt;}&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;c&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;json&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nf"&gt;dispatch&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;body&lt;/span&gt;&lt;span class="p"&gt;));&lt;/span&gt;
&lt;span class="p"&gt;});&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;That's the runtime path. Around 200 lines of TypeScript including the dispatch switch, the error helpers, the tool registry, and the route. Add OAuth and scope checks on top if you need them — but the protocol core stays at 200 lines.&lt;/p&gt;

&lt;h2&gt;
  
  
  What I actually got out of doing it this way
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;Total control of the request envelope.&lt;/strong&gt; When something goes wrong on the wire — and something always goes wrong on the wire when you launch — I'm reading my own code, not stepping through a Transport abstraction. The dispatch function is one switch statement. There's nowhere for a bug to hide.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Zero dependencies I'm not using.&lt;/strong&gt; No SSE machinery, no session manager, no Express adapter, no schema validation library that overlaps with the Zod I already had. The &lt;code&gt;node_modules&lt;/code&gt; footprint of the MCP code path is just Hono.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Native Hono integration.&lt;/strong&gt; OAuth middleware, rate limiting, observability — all of it composes the same way it does for the rest of my API. My MCP route sits next to my Stripe webhook and my health probe and uses the same middleware stack. No special case.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Easier mental model for testing.&lt;/strong&gt; &lt;code&gt;dispatch(rpcRequest, authContext)&lt;/code&gt; is a pure-ish function. Vitest calls it with a fake request, asserts the response shape. No transport to mock, no port to bind, no subprocess to spawn.&lt;/p&gt;

&lt;h2&gt;
  
  
  The trade-offs (be honest)
&lt;/h2&gt;

&lt;p&gt;I'm not pretending this is free. Going SDK-less means:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;No SSE / streaming.&lt;/strong&gt; If a tool needs to push partial results or resume an interrupted task, I'd have to build that. For nine synchronous lookups, I don't. If that changes — I'll switch.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;No session manager.&lt;/strong&gt; Each request is independent. Stateful conversations (which MCP supports via session IDs) aren't on my surface.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;No free updates.&lt;/strong&gt; When the spec evolves and adds a new method, I have to add a case to my switch. The SDK would have shipped that for me. So far the protocol has been stable enough that this hasn't bitten me.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;You have to read the spec.&lt;/strong&gt; Which honestly I think is a feature, not a bug. If you're shipping an MCP server you should know what's on the wire.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;There's also the STDIO question. Claude Desktop and Cursor launch MCP servers as child processes over stdin/stdout — they don't make HTTP calls. For that, I do use the SDK, in a separate &lt;code&gt;npx @openswissdata/mcp&lt;/code&gt; binary that's &lt;em&gt;only&lt;/em&gt; an STDIO bridge:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;Server&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;@modelcontextprotocol/sdk/server/index.js&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="c1"&gt;// ...this binary just proxies STDIO → HTTPS over to the real server.&lt;/span&gt;
&lt;span class="c1"&gt;// Using the SDK here is correct — STDIO transport is exactly what it's for.&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;That's the right place for the SDK: a thin bridge between an OS-level subprocess transport and my HTTP endpoint. The SDK does the part it was designed for, and my dispatcher does the part it was designed for. Both files end up small.&lt;/p&gt;

&lt;h2&gt;
  
  
  What this powers
&lt;/h2&gt;

&lt;p&gt;The custom dispatcher runs in production at &lt;code&gt;mcp.openswissdata.com&lt;/code&gt; — that's the live MCP endpoint behind &lt;a href="https://openswissdata.com" rel="noopener noreferrer"&gt;openswissdata.com&lt;/a&gt;, the first MCP server originating from Switzerland on Anthropic's registry. It serves Swiss customs tariffs, FINMA financial entities, and a few other Swiss reference datasets to AI agents.&lt;/p&gt;

&lt;p&gt;Whole code is Apache-2.0 licensed and on GitHub: &lt;a href="https://github.com/cammac-creator/openswissdata" rel="noopener noreferrer"&gt;github.com/cammac-creator/openswissdata&lt;/a&gt;. The dispatcher I described lives at &lt;code&gt;src/mcp/server.ts&lt;/code&gt; and the Hono route at &lt;code&gt;src/routes/mcp/index.ts&lt;/code&gt;. Copy it, paste it, adapt it — that's literally why the license is what it is.&lt;/p&gt;

&lt;h2&gt;
  
  
  When you should use the SDK anyway
&lt;/h2&gt;

&lt;p&gt;I don't want to oversell this. The SDK is the right call when:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;You're shipping a &lt;strong&gt;STDIO-launched desktop integration&lt;/strong&gt; — that's its native habitat.&lt;/li&gt;
&lt;li&gt;You need &lt;strong&gt;SSE / streaming&lt;/strong&gt; for long-running tools or resumable sessions.&lt;/li&gt;
&lt;li&gt;You want &lt;strong&gt;schema-validated request types&lt;/strong&gt; out of the box (the SDK ships Zod schemas for every wire message).&lt;/li&gt;
&lt;li&gt;You don't want to track spec drift in your own switch statement.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;What the SDK is &lt;em&gt;not&lt;/em&gt; the right call for: an HTTP server in a non-Express stack, with synchronous tools, where you want one set of middleware to handle everything. That's where the friction of fitting the SDK in starts costing more than reading the spec directly.&lt;/p&gt;

&lt;h2&gt;
  
  
  Takeaway
&lt;/h2&gt;

&lt;p&gt;"Should I use the SDK" is a real question, not a default. The MCP wire protocol is small enough that a custom dispatcher is a weekend, not a quarter. If the SDK gives you something concrete — STDIO, SSE, sessions — use it. If it gives you abstractions you'd have to fight to wedge into your stack, you have permission to write the 200 lines yourself.&lt;/p&gt;

&lt;p&gt;The protocol is the interface. The SDK is one implementation of it. Sometimes the second one is yours.&lt;/p&gt;




&lt;p&gt;Wire protocol spec: &lt;a href="https://modelcontextprotocol.io" rel="noopener noreferrer"&gt;modelcontextprotocol.io&lt;/a&gt;&lt;br&gt;
Source code (Apache-2.0): &lt;a href="https://github.com/cammac-creator/openswissdata" rel="noopener noreferrer"&gt;github.com/cammac-creator/openswissdata&lt;/a&gt;&lt;/p&gt;

</description>
      <category>mcp</category>
      <category>typescript</category>
      <category>ai</category>
      <category>indiehacker</category>
    </item>
    <item>
      <title>Building an IBAN Validation API with Hono, SQLite, and MCP</title>
      <dc:creator>Claude-Alain Martin</dc:creator>
      <pubDate>Sat, 04 Apr 2026 07:05:29 +0000</pubDate>
      <link>https://dev.to/cammaccreator/building-an-iban-validation-api-with-hono-sqlite-and-3mp</link>
      <guid>https://dev.to/cammaccreator/building-an-iban-validation-api-with-hono-sqlite-and-3mp</guid>
      <description>&lt;p&gt;title: Building an IBAN Validation API with Hono, SQLite, and MCP&lt;br&gt;&lt;br&gt;
  published: true&lt;br&gt;&lt;br&gt;
  tags: typescript, api, webdev, ai&lt;br&gt;&lt;br&gt;
  cover_image: &lt;a href="https://api.ibanforge.com/og-image.png" rel="noopener noreferrer"&gt;https://api.ibanforge.com/og-image.png&lt;/a&gt;                                                                    &lt;/p&gt;




&lt;p&gt;# Building an IBAN Validation API with Hono, SQLite, and MCP                                                           &lt;/p&gt;

&lt;p&gt;I recently shipped &lt;a href="https://ibanforge.com" rel="noopener noreferrer"&gt;IBANforge&lt;/a&gt;, a free API for IBAN validation and BIC/SWIFT lookup. v1.1.0 adds&lt;br&gt;
   Swiss clearing data (1,190 BC-Nummer entries from SIX with SIC, euroSIC, Instant Payments), 85 EMI/neobank&lt;br&gt;&lt;br&gt;
  classifications, and 5 MCP tools for AI agents. In this article, I'll walk through the key architectural decisions and&lt;br&gt;
  share real code from the project.                                                                                      &lt;/p&gt;

&lt;p&gt;## Why Hono Over Express&lt;/p&gt;

&lt;p&gt;When I started IBANforge, I considered Express, Fastify, and Hono. I went with Hono for three reasons:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;Performance&lt;/strong&gt; -- Hono is built for edge runtimes and benchmarks significantly faster than Express on Node.js&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;TypeScript-first&lt;/strong&gt; -- Full type inference on routes, middleware, and context
&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Lightweight middleware&lt;/strong&gt; -- Built-in CORS, compression, and logging with zero config&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;Here's how the main app comes together:                                                                                &lt;/p&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;
typescript                                                                                                          
  import { Hono } from 'hono';                                                                                           
  import { compress } from 'hono/compress';                       
  import { cors } from 'hono/cors';        
  import { logger } from 'hono/logger';                                                                                  

  const app = new Hono();                                                                                                

  app.use('*', cors({ origin: '*' }));
  app.use('*', logger());                                                                                                
  app.use('*', compress());

  // x402 payment middleware only on paid routes                  
  app.use('/v1/*', createX402Middleware());                                                                              

  // Routes                                                                                                              
  app.route('/', ibanValidate);                                   
  app.route('/', bicLookup);                                                                                             
  app.route('/', health);                                         

  The route handler for IBAN validation is clean and readable:                                                           

  ibanValidate.post('/v1/iban/validate', async (c) =&amp;gt; {                                                                  
    const start = performance.now();                                                                                     
    const body = await c.req.json&amp;lt;{ iban?: unknown }&amp;gt;();          

    const result = validateIBAN(body.iban as string);             

    // Enrich with BIC, SEPA info, issuer classification, and risk indicators
    enrichResult(result);

    result.processing_ms = Math.round(
      (performance.now() - start) * 100                                                                                  
    ) / 100;                                                                                                             

    return c.json(result);                                                                                               
  });                                                             

  No decorators, no class inheritance, no magic -- just functions.                                                       

  SQLite for Lookup Data                                                                                                 

  IBANforge stores 121,000+ BIC/SWIFT entries from GLEIF (the Global Legal Entity Identifier Foundation). The data is    
  CC0-licensed, free to use.                                      

  Why SQLite instead of PostgreSQL?                                                                                      

  - Zero infrastructure -- The database is a single file shipped inside the Docker image                                 
  - Read performance -- Queries take &amp;lt;10ms for exact BIC lookups  
  - Simplicity -- No connection pools, no migrations server, no managed database costs                                   

  The BIC lookup uses prepared statements with an LRU cache on top:                                                      

  import { getBicDB } from './db.js';                                                                                    
  import { LRUCache } from './cache.js';                                                                                 

  const bicCache = new LRUCache&amp;lt;BICRow | null&amp;gt;(2000);                                                                    

  function lookupByBic11(bic11: string): BICRow | null {                                                                 
    const db = getBicDB();                                        
    const stmt = db.prepare(                                                                                             
      'SELECT * FROM bic_entries WHERE bic11 = ? LIMIT 1'         
    );                                                                                                                   
    return (stmt.get(bic11) as BICRow) ?? null;                                                                          
  }

  export function lookup(bic: string): BICRow | null {                                                                   
    const cached = bicCache.get(bic);
    if (cached !== undefined) return cached;                                                                             

    const result = bic.length === 11                                                                                     
      ? lookupByBic11(bic)
      : lookupByBic11(bic + 'XXX');                                                                                      

    bicCache.set(bic, result);
    return result;                                                                                                       
  }                                                               

  The LRU cache wraps the main lookup() entry point, so repeated lookups for the same BIC code are sub-microsecond. For  
  the initial lookup, SQLite returns in ~2-5ms -- fast enough for real-time validation.

  MCP Integration: How AI Agents Use the API                                                                             

  This is what makes IBANforge different from existing IBAN APIs. The Model Context Protocol (MCP) lets AI agents like   
  Claude discover and call API tools natively.                    

  Here's how we expose IBAN validation as an MCP tool:                                                                   

  import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';                                                   
  import { z } from 'zod';                                                                                               

  const server = new McpServer({                                                                                         
    name: 'ibanforge',                                            
    version: '1.1.0',
  });

  server.registerTool(
    'validate_iban',                                                                                                     
    {                                                             
      title: 'Validate IBAN',
      description: 'Validate a single IBAN and retrieve BIC/SWIFT info.',                                                
      inputSchema: {                                                                                                     
        iban: z.string().describe('IBAN to validate. Spaces accepted.'),                                                 
      },                                                                                                                 
      annotations: {                                              
        readOnlyHint: true,                                                                                              
        idempotentHint: true,                                     
      },
    },
    async ({ iban }) =&amp;gt; {
      const result = validateIBAN(iban);
      enrichResult(result);
      return {
        content: [{                                                                                                      
          type: 'text',
          text: JSON.stringify(result, null, 2),                                                                         
        }],                                                       
      };
    },
  );

  Once configured, an AI agent can say "validate this IBAN: CH93 0076 2011 6238 5295 7" and get structured bank data back
   -- no prompt engineering required.

  The MCP server runs over stdio transport, which means any MCP-compatible client (Claude Desktop, Cursor, custom agents)
   can plug it in with a single config entry.

  The x402 Micropayment Model                                                                                            

  Instead of API keys and monthly subscriptions, IBANforge uses x402 -- an HTTP-native payment protocol. The idea is     
  simple: the API responds with 402 Payment Required and the client pays per-call in USDC on Base L2.

  Pricing:                                                                                                               
  - IBAN validation: $0.005 per call
  - BIC lookup: $0.003 per call                                                                                          
  - Batch validation: $0.002 per IBAN                             

  During launch, all endpoints are free. The x402 middleware is configured but not enforced yet. When it goes live,
  there's no signup, no API key management, no billing dashboard -- just pay and use.                                    

  // x402 middleware applied only to /v1/* routes                                                                        
  app.use('/v1/*', createX402Middleware());                                                                              

  The Numbers                                                                                                            

  The entire infrastructure costs ~$6/month:                                                                             

  ┌───────────────────┬────────────┐                                                                                     
  │     Component     │    Cost    │                              
  ├───────────────────┼────────────┤                                                                                     
  │ Railway (API)     │ $5/month   │
  ├───────────────────┼────────────┤                                                                                     
  │ Vercel (frontend) │ $0/month   │                              
  ├───────────────────┼────────────┤                                                                                     
  │ Domain            │ ~$1/month  │
  ├───────────────────┼────────────┤                                                                                     
  │ GLEIF data        │ free (CC0) │                              
  └───────────────────┴────────────┘                                                                                     

  The SQLite database is 121,197 BIC entries. No managed database fees.                                                  

  Try It                                                                                                                 

  - Playground: ibanforge.com/playground -- test IBAN validation and BIC lookup interactively                            
  - API Docs: ibanforge.com/docs -- full reference with curl, Python, and TypeScript examples
  - GitHub: github.com/cammac-creator/ibanforge -- MIT license, self-hostable                                            

  If you're building payment workflows, KYC pipelines, or AI agents that handle bank data -- give it a try. I'd love to  
  hear your feedback.                                                                                                    

  ---                
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;

</description>
      <category>typescript</category>
      <category>api</category>
      <category>ai</category>
      <category>webdev</category>
    </item>
  </channel>
</rss>
