EMCP
A minimal Elixir MCP (Model Context Protocol) server.
Setup
1. Initialize the session store
Add the session store initialization to your application's start/2:
defmodule MyApp.Application do
use Application
def start(_type, _args) do
EMCP.SessionStore.ETS.init()
children = [
# ...
]
Supervisor.start_link(children, strategy: :one_for_one)
end
end
You can use a custom session store by implementing the EMCP.SessionStore behaviour.
2. Define a server
defmodule MyApp.MCPServer do
use EMCP.Server,
name: "my-app",
version: "1.0.0",
tools: [MyApp.Tools.Echo],
prompts: [MyApp.Prompts.CodeReview],
resources: [MyApp.Resources.Readme],
resource_templates: [MyApp.ResourceTemplates.UserProfile],
# You can optionally add these meta fields as well:
title: "My App",
description: "MCP server for My App",
instructions: "Use the tools to interact with My App.",
end3. Mount the transport
Add the StreamableHTTP transport to your Phoenix router. Mount it outside any pipeline since EMCP handles content negotiation itself:
scope "/mcp" do
forward "/", EMCP.Transport.StreamableHTTP, server: MyApp.MCPServer
end4. Add tools to your server
defmodule MyApp.Tools.Echo do
@behaviour EMCP.Tool
@impl EMCP.Tool
def name, do: "echo"
@impl EMCP.Tool
def description, do: "Echoes back the provided message"
@impl EMCP.Tool
def input_schema do
%{
type: :object,
properties: %{
message: %{type: :string}
},
required: [:message]
}
end
@impl EMCP.Tool
def call(_conn, %{"message" => message}) do
EMCP.Tool.response([%{"type" => "text", "text" => message}])
end
# Return errors with:
# EMCP.Tool.error("something went wrong")
endPrompts
Prompts are reusable templates that return structured messages:
defmodule MyApp.Prompts.CodeReview do
@behaviour EMCP.Prompt
@impl EMCP.Prompt
def name, do: "code_review"
@impl EMCP.Prompt
def description, do: "Reviews code with optional focus area"
@impl EMCP.Prompt
def arguments do
[
%{name: "code", description: "The code to review", required: true},
%{name: "focus", description: "Optional area to focus on"}
]
end
@impl EMCP.Prompt
def template(_conn, %{"code" => code} = args) do
focus = args["focus"]
user_text =
if focus,
do: "Review this code, focusing on #{focus}:\n\n#{code}",
else: "Review this code:\n\n#{code}"
%{
"description" => "Code review prompt",
"messages" => [
%{"role" => "user", "content" => %{"type" => "text", "text" => user_text}},
%{"role" => "assistant", "content" => %{"type" => "text", "text" => "I'll review the code you've provided."}}
]
}
end
endResources
Resources expose data that clients can read. A static resource has a fixed URI:
defmodule MyApp.Resources.Readme do
@behaviour EMCP.Resource
@impl EMCP.Resource
def uri, do: "file:///project/readme"
@impl EMCP.Resource
def name, do: "readme"
@impl EMCP.Resource
def description, do: "The project README"
@impl EMCP.Resource
def mime_type, do: "text/plain"
@impl EMCP.Resource
def read(_conn), do: File.read!("README.md")
endResource templates use URI patterns so clients can request dynamic content:
defmodule MyApp.ResourceTemplates.UserProfile do
@behaviour EMCP.ResourceTemplate
@impl EMCP.ResourceTemplate
def uri_template, do: "db:///users/{user_id}/profile"
@impl EMCP.ResourceTemplate
def name, do: "user_profile"
@impl EMCP.ResourceTemplate
def description, do: "A user profile by ID"
@impl EMCP.ResourceTemplate
def mime_type, do: "application/json"
@impl EMCP.ResourceTemplate
def read(_conn, "db:///users/" <> rest) do
case String.split(rest, "/") do
[user_id, "profile"] ->
user = MyApp.Repo.get!(MyApp.User, user_id)
{:ok, JSON.encode!(user)}
_ ->
{:error, "Resource not found"}
end
end
def read(_conn, _uri), do: {:error, "Resource not found"}
end
When a client calls resources/read, the server first tries an exact URI match against static resources. If none match, it tries each resource template in order until one handles the URI.
Origin validation
To prevent DNS rebinding attacks, you can enable origin validation on the transport. When enabled, only requests with an Origin header matching the allowed list will be accepted. Requests without an Origin header (e.g. from CLI tools) are always allowed.
forward "/", EMCP.Transport.StreamableHTTP,
server: MyApp.MCPServer,
validate_origin: Mix.env() == :prod,
allowed_origins: ["example.com", "staging.example.com"]Allowed origins can be specified as:
-
An empty list:
[]- blocks all incoming requests with an Origin header -
Full URLs:
"https://example.com"— matches the exact scheme and host, plus any port -
Bare domains:
"example.com"— matches any scheme (httporhttps) and any port
Origin matching is case-insensitive. If you want to allow all origins, just set validate_origin: false or remove this configuration completely.
Session management
By default, StreamableHTTP silently re-creates expired or unknown sessions.
Your EMCP server might receive unknown sessions for example if you run your app on multiple servers and your load-balancer does not route the client HTTP requests based on the mcp-session-id header. If your load-balancer routes an HTTP request to a server that didn't initialize the MCP session, the server would return a 404 - Session not found.
For stateless HTTP requests (which almost all MCP requests are anyways), it does not matter whether a session is initialized or not. That's why StreamableHTTP recreates such unknown sessions by default, a trade-off that deviates from the MCP spec to provide a better UX.
If you want to follow the MCP spec and return 404 for missing or expired sessions, set recreate_missing_session: false.
forward "/", EMCP.Transport.StreamableHTTP,
server: MyApp.MCPServer,
recreate_missing_session: falseBut be warned that without proper HTTP request routing, this might irritate your user since most LLM clients (e.g. Claude Code) don't recreate expired or unknown MCP sessions during an ongoing conversation and require the user to start a new conversation to reconnect. This is inconvenient to the user who loses the conversation context, but hey! - it's MCP-spec compliant!
STDIO Transport
For local development or CLI tools, you can use the STDIO transport instead. It reads JSON-RPC messages from stdin and writes responses to stdout.
Add it to your supervision tree:
children = [
{EMCP.Transport.STDIO, server: MyApp.MCPServer}
]
Configure it in Claude Code via .claude/settings.json:
{
"mcpServers": {
"my-app": {
"command": "mix",
"args": ["run", "--no-halt"],
"cwd": "/path/to/your/elixir/project"
}
}
}Acknowledgements
Based on the official Ruby MCP SDK reference implementation.
Inspired by these more complete Elixir implementations:
Development
The e2e tests use the MCP Inspector CLI. Install it before running tests:
cd test/inspector && bun installThen run the tests:
mix test