cd /news/developer-tools/spring-boot-mcp-server-in-2026-the-t… · home topics developer-tools article
[ARTICLE · art-40187] src=dev.to ↗ pub= topic=developer-tools verified=true sentiment=· neutral

Spring Boot MCP Server in 2026: The Transport Trap That Wastes Your Weekend

A developer built a Spring Boot MCP server that exposes database queries, REST API calls, and file system access as tools for AI agents like Claude, GitHub Copilot, and Cursor. The project highlights a common transport trap where using the wrong Maven starter (stdio vs. SSE) causes silent failures with no error messages. The fix is a single configuration line matching the starter to the client type.

read3 min views1 publishedJun 26, 2026

I spent an afternoon staring at "connection refused" on my first MCP server.

The fix was one config line. Here's what no README tells you upfront.

Model Context Protocol is the standard that lets AI agents — Claude, GitHub

Copilot, Cursor — call your code as a tool. Instead of the AI just generating

text, it can actually invoke your functions and get real data back.

Think of it as giving Claude a set of keys to specific doors in your Java

backend. It asks "can you run this query?" — your MCP server runs it, returns

the result — Claude uses that result in its response.

For Java teams, this is significant. There are millions of Spring Boot services

sitting in production right now that AI agents can't touch. MCP changes that.

Here's the full working server — a Spring Boot MCP server that exposes

database queries, REST API calls, and file system access as tools

any AI agent can call.

Maven dependencies:

<dependency>
    <groupId>org.springframework.ai</groupId>
    <artifactId>spring-ai-starter-mcp-server-webmvc</artifactId>
</dependency>
<dependencyManagement>
  <dependencies>
    <dependency>
      <groupId>org.springframework.ai</groupId>
      <artifactId>spring-ai-bom</artifactId>
      <version>1.0.0</version>
      <type>pom</type>
      <scope>import</scope>
    </dependency>
  </dependencies>
</dependencyManagement>

Your first tool:

@Service
public class DatabaseMcpTools {

    @Autowired private JdbcTemplate jdbc;

    @Tool(description = "Run a read-only SQL query on the application database")
    public String queryDatabase(
        @ToolParam(description = "SQL SELECT query to execute") String sql
    ) {
        if (!sql.trim().toUpperCase().startsWith("SELECT")) {
            return "Error: only SELECT queries are permitted";
        }
        return jdbc.queryForList(sql).toString();
    }

    @Tool(description = "List all tables in the database schema")
    public String listTables() {
        return jdbc.queryForList(
            "SELECT table_name FROM information_schema.tables " +
            "WHERE table_schema = 'public'"
        ).toString();
    }
}

application.yml

:

spring:
  ai:
    mcp:
      server:
        name: my-mcp-server
        version: 1.0.0
        instructions: "Provides database query and table listing tools."

Run it:

mvn spring-boot:run

Add to ~/Library/Application Support/Claude/claude_desktop_config.json

:

{
  "mcpServers": {
    "my-spring-server": {
      "command": "java",
      "args": ["-jar", "/absolute/path/to/your-server.jar"]
    }
  }
}

Restart Claude Desktop. You should see a 🔨 hammer icon in the chat input.

Click it — your tool names should appear. Type:

"List all the tables in the database"

Claude calls your tool, your Spring Boot logs fire, Claude gets real data back.

That's your first working MCP integration.

Here's what burned me. There are two transports and they are not

interchangeable:

Client Transport Maven starter
Claude Desktop, Claude Code CLI stdio (subprocess) spring-ai-starter-mcp-server
VS Code, Cursor, Windsurf SSE (HTTP) spring-ai-starter-mcp-server-webmvc

The failure mode is brutal: no error message. Claude Desktop just shows

no tools. VS Code just shows no server. The process starts fine. Logs look

fine. The handshake silently fails.

The rule: if the client is an IDE connecting over HTTP, use the webmvc

starter. If the client is a CLI spawning your jar as a subprocess, use the

plain starter without webmvc

.

For VS Code / Cursor, add to .vscode/mcp.json

while the app is running:

{
  "servers": {
    "my-spring-server": {
      "type": "sse",
      "url": "http://localhost:8080/sse"
    }
  }
}

1. Guard against path traversal in file tools:

Path target = BASE_DIR.resolve(userInput).normalize();
if (!target.startsWith(BASE_DIR)) return "Error: access denied";

2. Guard against SQL writes:

if (!sql.trim().toUpperCase().startsWith("SELECT")) 
    return "Error: only SELECT queries are permitted";

3. Never return null from a @Tool method — return empty string instead.

4. Use absolute paths in your Claude Desktop config, not ~/

or ./

.

5. Add a docker-compose.yml so clients can run it with one command:

services:
  mcp-server:
    build: .
    ports:
      - "8080:8080"
    environment:
      - SPRING_DATASOURCE_URL=${DB_URL}
      - SPRING_DATASOURCE_USERNAME=${DB_USER}
      - SPRING_DATASOURCE_PASSWORD=${DB_PASSWORD}

Everything above plus file system tools, REST API wrapper, and setup

guides for both transports:

github.com/anirbandashfx-commits/spring-boot-mcp-server

Building a custom MCP server for your Java team?

Connect on LinkedIn

── more in #developer-tools 4 stories · sorted by recency
── more on @spring boot 3 stories trending now
sponsored brought to you by zahid.host 4,200+ EU-deployed projects
reading about agents? ship yours in a single git push.

Run your AI side-project on zahid.host

EU-based hosting, git-push deploys, automatic HTTPS, no cold starts. Free tier with a custom domain — perfect for shipping the agent you just read about.

$git push zahid main
Live at https://your-agent.zahid.host
Get free account → Pricing
from €0/mo · no card required
LIVE [news/spring-boot-mcp-serv…] indexed:0 read:3min 2026-06-26 ·