mcp
MCP Crystal Shard
A lightweight, reusable Crystal implementation of the Model Context Protocol (MCP) for building JSON-RPC servers that expose tools to AI assistants.
Features
- Tool Registry: Automatic registration of tools via macro-based inheritance
- JSON-RPC 2.0: Full compliance with MCP protocol specifications
- Server-Sent Events: Real-time communication support
- Dependency Injection: Clean separation via interfaces for auth, logging, and config
- Kemal Integration: Easy integration with Kemal web framework
Quick Start
Add to your shard.yml:
dependencies:
mcp:
github: ralsina/mcp
Simple stdio MCP Server
require "mcp"
# Define your tool by inheriting from AbstractTool
class MyTool < MCP::AbstractTool
@@tool_name = "my_tool"
@@tool_description = "A simple example tool that returns 42"
@@tool_input_schema = {
"type" => "object",
"properties" => {} of String => String,
}.to_json
def invoke(params : Hash(String, JSON::Any), env : HTTP::Server::Context? = nil)
{
"result" => 42
}
end
end
# Start a simple stdio MCP server
MCP::StdioHandler.start_server
That's it! Your MCP server is ready to handle requests via stdin/stdout.
Web Server Integration
require "kemal"
require "mcp"
# Create your tools
class EchoTool < MCP::AbstractTool
@@tool_name = "echo"
@@tool_description = "Echoes back the input message"
@@tool_input_schema = {
"type" => "object",
"properties" => {
"message" => {
"type" => "string",
"description" => "Message to echo back",
},
},
"required" => ["message"],
}.to_json
def invoke(params : Hash(String, JSON::Any), env : HTTP::Server::Context? = nil)
message = params["message"]?.try(&.as_s) || "no message"
{
"echo" => message
}
end
end
# Set up MCP server
mcp_server = MCP::Server.new
# Kemal routes
post "/mcp" do |env|
# Simple user ID extraction (implement your own auth)
user_id = env.request.headers["User-Id"]? || "anonymous"
response = mcp_server.handle_request(env, user_id)
env.response.content_type = "application/json"
response
end
get "/mcp" do |env|
user_id = env.request.headers["User-Id"]? || "anonymous"
mcp_server.handle_sse(env, user_id)
end
Kemal.run
Architecture
The shard provides these core components:
MCP::AbstractTool: Base class for all tools with automatic registrationMCP::Server: Handles JSON-RPC requests and SSE connectionsMCP::Handler: Kemal integration layer- Interface classes: For auth, logging, and configuration injection
Tool Registration
Tools are automatically registered when the class is defined:
class MyTool < MCP::AbstractTool
# Tool metadata
@@tool_name = "my_tool"
@@tool_description = "Does something useful"
@@tool_input_schema = {
"type" => "object",
"properties" => {
"input" => {
"type" => "string",
"description" => "Input parameter",
},
},
"required" => ["input"],
}.to_json
# Implementation
def invoke(params : Hash(String, JSON::Any), env : HTTP::Server::Context? = nil)
# Your tool logic here
input_value = params["input"]?.try(&.as_s) || ""
{
"result" => "processed: #{input_value}"
}
end
end
Protocol Support
- JSON-RPC 2.0: Core protocol for request/response
- MCP 2024-11-05: Standard protocol version
- Tools/List: List available tools with metadata
- Tools/Call: Execute tools with parameters
- Server-Sent Events: Real-time updates and streaming
Examples
See the examples/ directory for complete working examples:
Basic Example (examples/basic/)
- Simple stdio MCP server
- Demonstrates core tool registration and JSON-RPC handling
- Perfect for command-line MCP servers
Web Example (examples/web/)
- Web-based MCP server using Kemal framework
- HTTP JSON-RPC endpoint and Server-Sent Events
- Demonstrates authentication, logging, and HTTP context access
- Includes built-in web interface for testing
License
MIT License - see LICENSE file for details.
Repository
mcp
Owner
Statistic
- 3
- 0
- 0
- 1
- 1
- 8 days ago
- October 29, 2025
License
MIT License
Links
Synced at
Sat, 08 Nov 2025 23:50:42 GMT
Languages