A library for building MCP (Model Context Protocol) servers in Scala 3, based on Tapir. Describe MCP tools with type-safe input, expose them over a JSON-RPC HTTP API.
Integrates with any Scala stack, using any of the HTTP server implementations supported by Tapir.
Add the dependency to your build.sbt:
libraryDependencies += "com.softwaremill.chimp" %% "core" % "0.1.5"Below is a self-contained, scala-cli-runnable example:
//> using dep com.softwaremill.chimp::core:0.1.5
import chimp.*
import sttp.tapir.*
import sttp.tapir.server.netty.sync.NettySyncServer
// define the input type for your tool
case class AdderInput(a: Int, b: Int) derives io.circe.Codec, Schema
@main def mcpApp(): Unit =
  // describe the tool providing the name, description, and input type
  val adderTool = tool("adder").description("Adds two numbers").input[AdderInput]
  // combine the tool description with the server-side logic
  val adderServerTool = adderTool.handle(i => Right(s"The result is ${i.a + i.b}"))
  // create the MCP server endpoint; it will be available at http://localhost:8080/mcp  
  val mcpServerEndpoint = mcpEndpoint(List(adderServerTool), List("mcp"))
  // start the server
  NettySyncServer().port(8080).addEndpoint(mcpServerEndpoint).startAndWait()Are available here.
Chimp implements the HTTP transport of the MCP protocol (version 2025-03-26). Only tools are supported, via the following JSON-RPC commands:
- Initialization and capabilities negotiation (initialize)
- Listing available tools (tools/list)
- Invoking a tool (tools/call)
All requests and responses use JSON-RPC 2.0. Tool input schemas are described using JSON Schema, auto-generated from Scala types.
- Use tool(name)to start defining a tool.
- Add a description and annotations for metadata and hints.
- Specify the input type (must have a Circe Codecand TapirSchema).
- Provide the server logic as a function from input to Either[String, String](or a generic effect type).- Use handleto connect the tool definition with the server logic when the use of headers is not required.
- Use handleWithHeadersto connect the tool definition with the server logic when headers are required.
 
- Use 
- Create a Tapir endpoint by providing your tools to mcpEndpoint
- Start an HTTP server using your preferred Tapir server interpreter.
Chimp uses the circe JSON library, to decode the incoming input, as well as handle JSON-RPC envelopes (both for responses and requests).
When using ZIO, you might have to explicitly state the effect type that you are using, as the Tapir-ZIO integration requires
a RIO[R, A] effect (which is an alias for ZIO[R, Throwable, A]), for example:
val myServerTool = myTool.serverLogic[[X] =>> RIO[Any, X]]: (input, headers) => 
  ZIO.succeed(???)Contributions are welcome! Please open issues or pull requests.
We offer commercial support for Tapir and related technologies, as well as development services. Contact us to learn more about our offer!
Copyright (C) 2025 SoftwareMill https://softwaremill.com.