About
Modex (MOdel + ContEXt) is a Clojure library that lets you augment your AI with new tools, resources and prompts.
Modex implements (most of) the Model Context Protocol to build MCP Servers & Clients in 'native' Clojure.
Because it's native Clojure, you don't need to deal with Anthropic's MCP Java SDK.
Modex implements the stdio
transport, so no need for a proxy like
mcp-proxy to translate between SSE <=> stdio or vice versa.
Screenshot of Modex in Action
Claude Desktop can talk to a Modex MCP Server via its MCP client:
Table of Contents
- Quickstart
- What is MCP?
- What can Modex do?
- Detailed Step-by-Step Instructions
- Implementation
- Project Status
- Rationale
- FAQ
- Licence
Example Tools
- Datomic MCP uses Modex to expose Datomic tools so your models can query DB schema and data in dev or prod.
Quickstart
git clone git@github.com:theronic/modex.git
cd modex
./build.sh
builds an uberjar attarget/modex-mcp-0.1.0.jar
.- Open your Claude Desktop Config at
~/Library/Application\ Support/Claude/claude_desktop_config.json
- Configure a new MCP Server that will run the uberjar at its full path:
{
"mcpServers": {
"modex-mcp-hello-world": {
"command": "java",
"args": ["-jar", "/Users/your-username/code/modex/target/modex-mcp-0.1.0.jar"]
}
},
"globalShortcut": ""
}
- Restart Claude Desktop to activate your new MCP Server + tools :)
- Tell Claude to "run the inc tool with 123", authorize the tool and you should see an output of 124.
What is MCP?
MCP lets you augment your AI models with Tools, Resources & Prompts:
- Tools are things it can do, like query a database (e.g. Datomic).
- Resources are files and data it can read, like PDF bank statements.
- Prompts are templated messages and workflows.
For example, you could make a tool that fetches purchases from your bank's API and let the AI categorize your expenses, and then use another tool to write those expense categories to your accounting system, or to a database. Pretty cool.
What can Modex do?
Full Example
There is an MCP server example in src/modex/mcp/core.clj that defines an MCP server with some basic tools.
Your MCP client (e.g. Claude Desktop) can connect to this server and use exposed tools to provide additional context to your AI models.
Describe a tool with the tool
macro:
(require '[modex.mcp.tools :as tools])
(def add-tool (tools/tool (add [x y] (+ x y))))
Invoke a Tool with invoke-tool
:
Invocation uses a map of arguments like an MCP client would for a tools/call
request:
(tools/invoke-tool add-tool {:x 5 :y 6}) ; Modex will map these arguments and call the handler.
=> 11
Define a Toolset with tools
macro
The tools
macro just calls the tool
macro and puts tools in a map keyed on tool name (keyword):
(require '[modex.mcp.tools :as tools])
(def my-tools
"Define your tools here."
(tools/tools
(greet
"Greets a person by name."
[^{:type :string :doc "A person's first name." :required true} first-name
^{:type :string :doc "A person's last name (optional)." :required false} last-name]
[(str "Hello from Modex, " (if last-name ; args can optional
(str first-name " " last-name)
first-name) "!")])
(add
"Adds two numbers."
[^{:type :number :doc "First number to add."} a
^{:type :number :doc "Second number to add."} b]
(+ a b))
(subtract
"Subtracts two numbers (- a b)"
[^{:type :number :doc "First number."} a
^{:type :number :doc "Second number."} b]
(- a b))))
Create a Modex MCP Server + tools:
(require '[modex.mcp.server :as server])
(def my-mcp-server
"Here we create a reified instance of AServer. Only tools are presently supported."
(server/->server
{:name "Modex MCP Server"
:version "0.0.1"
:initialize (fn [] "do long-running, blocking I/O setup here")
:tools my-tools
:prompts nil
:resources nil}))
Start your MCP Server
(server/start-server! my-mcp-server)
Or put that in your -main
function.
Data Structures
Records
- Tool:
(defrecord Tool [name doc args handler])
- Parameter – a Tool Parameter.
(defrecord Parameter [name doc type required])
Protocols
Modex exposes an AServer
protocol and a DSL to define tools protocols that describe MCP servers, which expose tools, resources & prompts.
AServer Protocol:
(defprotocol AServer
(protocol-version [this])
(server-name [this])
(version [this])
(capabilities [this])
(initialize [this])
(list-tools [this])
(call-tool [this tool-name arg-map])
(list-resources [this])
(list-prompts [this]))
Detailed Step-by-Step Instructions
Step 1: Build the Uberjar
Before you can run it, you have to build it first. The build outputs an uberjar, which is like a Java executable.
clojure -T:build uber
or run the helper which does that:
./build.sh
(you might need to run chmod +x build.sh
)
Step 2: Open Claude Desktop Config
Open your Claude Desktop Configuration file, claude_desktop_config.json
, which on MacOS should be at:
~/Library/Application\ Support/Claude/claude_desktop_config.json
Step 3: Configure your MCP Server
Add an element under mcpServers
so it looks like this:
{
"mcpServers": {
"modex": {
"command": "java",
"args": ["-jar", "/Users/your-username/code/modex/target/modex-mcp-0.1.0.jar"]
}
},
"globalShortcut": ""
}
This tells Claude Desktop there is a tool named modex
and it can connect to by running java -jar /path/to/your/uber.jar
.
The way this works is that your local MCP Client (i.e. Claude Desktop), starts your MCP server process and communicates with it via stdin/stdout pipes.
Step 4: Restart Claude Desktop
You should now be able to ask Claude "run foo", or "what does foo say?" and it will run
the foo
tool and reply with the response, "Hello, AI!".
Implementation
Modex implements an MCP client & server in Clojure that is mostly compliant with the 2024-11-05 MCP Spec.
Messages are encoded using the JSON-RPC 2.0 wire format.
There are 3 message types:
- Requests have
{:keys [id method ?params]}
- Responses have
{:keys [id result ?error]}
- Notifications have
{:keys [method ?params}
MCP supports two transport types:
- [x] stdio/stdout – implemented in Modex.
- [ ] Server-Sent Events (SSE) – not implemented yet. Useful for restricted networks
Project Status
- [x] Passing tests
- [x] Ergonomics (AServer / AClient protocol?)
- [x] Tools
- [ ] nREPL for live changes to running process
- [ ] Resources
- [ ] Prompts
- [ ] SSE support
Rationale
There is an existing library mcp-clj that uses SSE, so it requires mcp-proxy to proxy from SSE <=> stdio. I was annoyed by this, so I made Modex.
FAQ
Can I modify the server while an MCP Client (like Claude Desktop) is connected?
Not yet, but I'll add an nREPL soon so you can eval changes while Claude Desktop is connected to the process without rebuilding the uberjar.
Btw. I tried to get it to run clojure -M -m modex.mcp.server
, but you can't set Claude Desktop's working directory.
So currently, I rebuild the uberjar and restart Claude Desktop. Will fix.
License
In summary:
- Free for non-commercial use: Use it, modify it, share it under GPLv3 at no cost, just keep it open source.
- Commercial use: Want to keep your changes private? Pay $20 once-off for a perpetual commercial license. This covers the cost of my AI tokens to keep building this in public.
This tool is licensed under the GNU General Public License v3.0 (GPLv3). You are free to use, modify, and distribute it, provided that any derivative works are also licensed under the GPLv3 and made open source. This ensures the tool remains freely available to the community while requiring transparency for any changes.
If you wish to use or modify this tool in a proprietary project—without releasing your changes under the GPLv3—you may purchase a commercial license. This allows you to keep your modifications private for personal or commercial use. To obtain a commercial license, please contact me at modex@petrus.co.za.
Author(s)
Recommend MCP