Install the mod from Modrinth or CurseForge. Pick the jar that matches your Minecraft version, drop it in
mods/alongside Fabric API, and launch. Full walkthrough in Quick start below.
A Model Context Protocol (MCP) server that runs inside Minecraft Java Edition as a Fabric mod. It exposes the server-side Minecraft API and the Fabric API as MCP tools so an MCP client — Claude Desktop, Cursor, or any agent that speaks MCP — can read and manipulate a live world programmatically.
An agent built this over MCP — authored as a parametric voxel model, then placed
in a single block_fill_batch and verified with block_render_region (the
server-side render that produced no screenshot dependency — it reads block map
colours straight from the world).
And Ghasticlawd — the project's Ghast × Claude mascot — voxelized, placed via
block_fill_batch, and confirmed with block_render_region.
And terrain, too — this red-rock canyon, inspired by the national parks of the
American West, was generated as a hydraulically eroded heightfield,
render-checked, then materialized into the world through block_fill_batch and
verified with block_render_region.
The mod ships separate jars for each supported Minecraft version. The build matrix in v0.2.0 is 1.21.11, 26.1.1, and 26.1.2.
The same jar exposes two MCP servers. The main entrypoint serves the world tools on
minecraft-java (default port 8765) wherever a server runs — dedicated or single-player. The
client entrypoint runs inside a real, rendered client and serves an inspection-only surface
on minecraft-java-client (default port 8766): view_capture returns the player's actual
first-person frame as a PNG, and sense_* / client_status read client-side perception. It is the
way an agent SEES the world as a player does — the real rendered pixels the headless server can't
produce. Run server-only, client-only (single-player gives you both endpoints from one process), or
a server+client combo. See
docs/configuration.md.
This mod depends on Minecraft's server-side internals via the Fabric API. The API surface evolves between Minecraft versions and a Minecraft update can change, deprecate, or remove methods this mod relies on. Pin your Minecraft installation, your Fabric API jar, and this mod's jar to known-good versions and upgrade them together. Treat the stack as experimental.
This server pairs with a companion Claude Code plugin, and the whole thing is the Java Edition counterpart of an existing Bedrock stack:
| Repository | Edition | Role |
|---|---|---|
minecraft-java-fabric-mcp-server (this repo) |
Java Edition | The MCP server, packaged as a Fabric mod. Loads inside Minecraft and exposes the Java API as MCP tools. |
minecraft-java-fabric-claude-plugin |
Java Edition | The companion Claude Code plugin. Bundles the MCP connection plus setup and builder skills/agents that drive this server. |
minecraft-bedrock-mcp-server |
Bedrock Edition | The Node/TypeScript MCP server for the Bedrock Dedicated Server. |
minecraft-bedrock-mcp-behavior-pack |
Bedrock Edition | The companion behavior pack for the Bedrock setup. |
The plugin is optional — any MCP client (Claude Desktop, Cursor, a hand-configured claude mcp add)
can talk to this server directly — but on Claude Code it's the fastest path: it wires up the
connection and ships the building skills and agents. See
Connect an MCP client below.
The Java and Bedrock stacks are independent — installing one has no effect on the other. They share a tone and configuration style but expose different (overlapping) tool surfaces because the Java and Bedrock Script APIs differ substantially.
MCP client (Claude Desktop, Cursor, …)
│ MCP over Streamable HTTP, localhost-only by default
▼
[ Fabric mod: HTTP transport ] ←—— Host / Origin / optional bearer validation
│
▼
[ Protocol layer ] ←—— JSON-RPC 2.0 over Streamable HTTP, tool registry
│
▼
[ Runtime: MinecraftMainThreadExecutor ] ←—— marshals every world touch onto the server main thread
│
▼
[ Adapter layer ] ←—— version-stable interface; per-version implementations
│
▼
the loaded Minecraft world (integrated single-player, or dedicated server)
Every tool call:
- Arrives on an HTTP thread of the embedded JDK
HttpServer. - Passes Host / Origin / (optional) bearer / body-size / rate-limit checks.
- Is dispatched by the MCP layer to a tool implementation.
- The tool submits any Minecraft API access through the main-thread executor.
- The executor completes the work on the next server tick and returns the result.
- The HTTP thread serializes the result to JSON and responds.
See docs/architecture.md for the detailed layering rationale.
End-to-end: a clean machine to "Claude making it rain chicken jockeys in your village." The default
configuration binds to 127.0.0.1:8765, no token required, with strict Host/Origin validation
that defends against DNS-rebinding and CSRF.
The mod runs inside the Minecraft launcher's JVM, which the launcher itself manages — for typical single-player play you don't need to install Java separately. (You only need a system JDK if you're building from source.)
If you don't already have it, get the Minecraft Launcher from minecraft.net/download. Launch it once, sign in, and load a vanilla world to confirm the install works.
Download the Fabric Loader installer and run it. Pick the
Minecraft version you want (one of 1.21.11, 26.1.1, or 26.1.2 — the versions this mod
ships jars for) and click Install. The installer registers a new "fabric-loader-…" profile in the
Minecraft Launcher and creates a mods/ folder under your game directory.
- This mod — get it from your preferred mod host and pick the jar whose suffix matches your
Minecraft version (e.g.
minecraft-fabric-mcp-0.2.0+1.21.11.jar):- Modrinth (recommended)
- CurseForge
- or the GitHub Releases page.
- Fabric API: pick the version that matches the same MC version.
Drop both jars into your Minecraft mods/ directory:
- Windows:
%APPDATA%\.minecraft\mods\ - macOS:
~/Library/Application Support/minecraft/mods/ - Linux:
~/.minecraft/mods/
Open the Minecraft Launcher, select the fabric-loader-… profile from the dropdown, click Play,
and open any world (existing or new). When the world finishes loading, the integrated server starts
and the MCP listener binds to http://127.0.0.1:8765/mcp. Confirm with:
curl http://127.0.0.1:8765/healthz
# → {"status":"ok"}The mod speaks MCP over Streamable HTTP. Connect from whichever client you use.
Claude Code (CLI) — via the plugin (recommended): the companion
minecraft-java-fabric-claude-plugin
wires up the MCP connection and installs the setup and building skills/agents. From a Claude Code
session, run:
/plugin marketplace add chapmanjw/minecraft-java-fabric-claude-plugin
/plugin install minecraft-java@minecraft-java-claude
Restart Claude Code afterward. See the plugin's README for what each skill and agent does.
Claude Code (CLI) — manual: if you'd rather connect the raw tools without the plugin, run once from a terminal —
claude mcp add --transport http minecraft-java http://localhost:8765/mcpClaude Desktop: edit the desktop app's claude_desktop_config.json (Settings → Developer → Edit
Config) and add:
{
"mcpServers": {
"minecraft-java": {
"command": "npx",
"args": ["-y", "mcp-remote", "http://localhost:8765/mcp"]
}
}
}Then restart Claude Desktop. For authenticated / remote setups, see docs/claude-desktop-integration.md and docs/cursor-integration.md.
In any Claude session with the MCP server connected, ask it natural-language things like:
- "Who's online in Minecraft right now?" →
player_list_online - "Give me 64 diamonds." →
player_give_item - "Set it to noon and clear the weather." →
level_set_time,level_set_weather - "Build me a 5×5×5 box of glass at my feet." →
block_fill_region - "Find a nearby village." →
entity_queryforminecraft:villager - "Make it rain chicken jockeys over the village." → 20×
command_executewithsummon minecraft:chicken <x> <y> <z> {Passengers:[{id:"minecraft:zombie",IsBaby:1b}]}pluslevel_set_weatherset tothunderfor atmosphere.
That last one is the canonical smoke test. If chickens with baby zombie riders rain down from the sky over your village while a thunderstorm rolls in, the whole stack is wired up correctly.
The full single-player walkthrough (with dedicated-server and LAN variants) lives in docs/setup-singleplayer.md. For a dedicated server with bearer auth and remote access, see docs/setup-dedicated-server.md.
All configuration is JSON-on-disk at <game dir>/config/minecraft_fabric_mcp/config.json, with every field
overridable by an environment variable (MCP_<UPPER_SNAKE_FIELD>). Defaults are baked into the mod —
the file only needs to exist if you're overriding something.
| Field | Default | Description |
|---|---|---|
host |
127.0.0.1 |
Bind address. Non-loopback requires auth_required=true and allow_remote=true. |
port |
8765 |
Listen port. |
auth_required |
false |
When true, every request must carry Authorization: Bearer <bearer_token>. |
bearer_token |
(null) | When auth_required=true and this is null, the mod generates a 32-byte hex token and writes it back into the config file. |
allow_remote |
false |
Required true when host is non-loopback. Belt-and-suspenders so you don't accidentally expose the listener. |
allowed_origins |
[] |
Browser-style Origin allow-list. Default rejects any request carrying an Origin header. |
command_timeout_ms |
15000 |
Per-tool wait for main-thread work. |
rate_limit_rpm |
60 |
Per-client requests per minute. |
max_body_bytes |
16777216 |
Max request body. |
event_buffer_size |
1024 |
Default per-subscription ring buffer size for events_* tools. |
queue_max |
256 |
Max concurrent async jobs (huge bulk operations). |
log_level |
info |
SLF4J log level. |
tls_cert_path / tls_key_path |
(null) | If both set, the listener serves HTTPS. Both must be set, or both null. |
metrics_enabled |
false |
Reserved for the Prometheus /metrics endpoint (v0.2.0). |
included_categories |
[] |
Allow-list of domain categories. If non-empty it replaces the default-ON set. One of blocks, structures, world, entities, players, items, gameplay, scripting, registries, server. |
excluded_categories |
[] |
Domain categories to drop. Applied after the allow-list (or the default-ON set). |
max_access |
write |
Access cap: read, write, or admin. Tools above the cap are dropped. The default write keeps admin/destructive tools off until you opt in. |
exclude_write_tools |
false |
Legacy switch, equivalent to max_access: read. When true, only read tools register. |
See Tool categories, access levels & defaults for the full model and worked examples.
See docs/configuration.md for the full reference and recipe-style examples.
- The default bind is
127.0.0.1with no token. Anything on the same machine can talk to the MCP endpoint; nothing else can. - The
Hostheader is validated against the bind address — DNS rebinding attacks that resolve an attacker-controlled domain to 127.0.0.1 still get rejected because theirHostheader is wrong. - The
Originheader is validated againstallowed_origins(empty by default). Browsers always sendOriginon cross-origin requests; legitimate MCP clients (mcp-remote, Cursor) do not. - Binding non-loopback requires both
allow_remote: trueandauth_required: true. On first run with auth enabled, the mod generates a 32-byte hex bearer token, writes it back to the config file (with POSIX 600 permissions where supported), and logs it once.
Full threat model in docs/security.md.
The mod registers tools whose names follow the Minecraft Java API class convention rather than
the Bedrock mc_<domain>_<action> style. Tools group by the class or concept they operate on:
| Domain | Examples |
|---|---|
server_* |
server_get_status, server_set_motd, server_save_all_worlds |
level_* |
level_set_time, level_set_weather, level_create_explosion, level_get_biome_at |
block_* |
block_get_state, block_fill_region, block_fill_batch, block_scan_summary, block_render_region |
block_entity_* |
block_entity_get_nbt, block_entity_set_nbt |
entity_* |
entity_summon, entity_teleport, entity_apply_effect |
player_* |
player_list_online, player_give_item, player_send_title |
inventory_* |
inventory_get, inventory_set_slot, inventory_count_items |
itemstack_* |
itemstack_describe, itemstack_drop_at |
command_* |
command_execute, command_execute_as |
scoreboard_* |
scoreboard_add_objective, scoreboard_set_score, scoreboard_add_team |
data_* |
data_storage_get, data_attachment_set |
structure_* |
structure_save_from_world, structure_load_to_world |
datapack_* |
datapack_enable, datapack_disable |
loot_table_* / recipe_* / tag_* / resource_loader_* |
registry-style read tools |
events_* |
events_subscribe, events_poll, events_unsubscribe |
A full reference with JSON Schemas and per-version annotations lives in docs/tools.md.
The 183 tools are filtered along two independent axes so you can narrow the surface by subsystem and by risk at the same time:
- Domain — which part of Minecraft the tool touches: blocks, entities, the server itself, and so on. Ten categories, each tied to a tool-name prefix. This is the "I only care about world-building, hide the scoreboard plumbing" axis.
- Access — how dangerous the tool is:
read(inspects state, mutates nothing) <write(the normal case: changes a block, an entity, an item) <admin(world-wide, server-lifecycle, or destructive — explosions, world-border resizing, difficulty and game-rule changes, datapack toggles, kicking players, registering commands). This is the "let the agent build, but don't let it reshape the world border or reload resources" axis.
An operator filters on both: pick the subsystems you want, then cap the blast radius.
| Category | Domains / prefixes | Purpose | Default | ~Tools |
|---|---|---|---|---|
blocks |
block_*, block_entity_* |
Read, place, fill, clone, scan, render, and erode blocks and block-entity NBT | ON | 20 |
structures |
structure_* |
Save / load / list named structure templates and their files | ON | 9 |
world |
level_*, worldborder_* |
Time, weather, biomes, dimensions, spawn, features, particles, explosions, the world border | ON | 31 |
entities |
entity_* |
Summon, query, teleport, tag, damage, effect, and edit-NBT for non-player entities | ON | 17 |
players |
player_*, player_screen_* |
Inspect and act on connected players: inventory, XP, gamemode, messages, kick | opt-in | 16 |
items |
inventory_*, itemstack_*, item_modify_* |
Inventory slots, item-stack inspection, container item edits | ON | 9 |
gameplay |
scoreboard_*, bossbar_*, advancement_* |
Scoreboards, teams, boss bars, advancements — the game-state logic layer | opt-in | 30 |
scripting |
command_*, function_*, schedule_*, events_*, data_storage_*, data_attachment_* |
Commands, datapack functions, scheduled work, event subscriptions, custom data | ON | 21 |
registries |
recipe_*, loot_table_*, tag_*, content_registry_*, resource_loader_*, resource_condition_*, fluid_storage_* |
Recipe / loot / tag / resource lookups and content-registry edits | opt-in | 21 |
server |
server_*, datapack_* |
Server status, MOTD, world saves, resource reloads, datapack enable/disable | ON | 9 |
Counts are the live 26.1.2 surface; earlier Minecraft targets register fewer (a tool gated on a
Fabric module or MC version drops out before categorization). Run tools/list against your build
for the exact set.
An empty config (or no config file at all) registers the lean default set: about 102 of the 183 tools. The rule is simple:
- Default-ON domains:
blocks,structures,world,entities,items,scripting,server. The builder-and-operator core. - Opt-in domains:
players,gameplay,registries. Useful, but noise for most world-building sessions — turn them on when you need them. - Default access cap:
max_access = write. Read and write tools register; admin tools do not.
So the default surface is every read/write tool in the seven ON domains. The other 81 stay off until
you ask for them: 67 in the three opt-in domains (players 16 + gameplay 30 + registries 21)
and 14 admin tools that sit in ON domains but are suppressed by the write cap.
The 15 admin-tagged tools (all opt-in via max_access: admin):
worldborder_set_size worldborder_add_size worldborder_set_center
worldborder_set_warning_blocks worldborder_set_warning_time
worldborder_set_damage_amount worldborder_set_damage_buffer
level_set_difficulty level_set_game_rule level_create_explosion
command_register server_reload_resources
datapack_enable datapack_disable player_kick
Fourteen of those sit in default-ON domains (only player_kick is in the opt-in players domain),
so on a default config they are present-but-suppressed by the write cap. Raise the cap to admin
and they appear.
For each supported tool, in order:
- Domain allow-list. If
included_categoriesis non-empty, keep the tool only if its category is listed. Otherwise keep it only if its category is default-ON. - Domain exclude. Drop the tool if its category is in
excluded_categories. - Access cap. Drop the tool if its access rank exceeds
max_access(read<write<admin).exclude_write_tools: trueis the legacy spelling ofmax_access: read.
included_categories is an outright replacement for the default-ON set — list it and the ON/opt-in
distinction no longer applies; you get exactly the categories you named (still subject to the
exclude list and the access cap).
(a) World-builder — the default. No config needed. You get the ~102-tool core: build blocks and structures, shape the world, summon and edit entities, move items, run commands and functions, save the server. No admin reshaping, no scoreboard/registry clutter.
{}(b) Read-only monitor. Inspection only — nothing mutates. Good for an observer agent or a dashboard.
{
"max_access": "read"
}({"exclude_write_tools": true} does the same thing.)
(c) Datapack developer. The default core plus the gameplay and registry surfaces for
advancements, loot, recipes, and tags. Still capped at write.
{
"included_categories": [
"blocks", "structures", "world", "entities", "items",
"scripting", "server", "gameplay", "registries"
]
}There is no "default-ON plus these two" shorthand — the allow-list is all-or-nothing, so when you need an opt-in domain, name the whole set you want (the seven ON domains plus your additions), as above.
(d) Full operator. Every category, every access level — the entire 183-tool surface, including admin/destructive tools.
{
"included_categories": [
"blocks", "structures", "world", "entities", "players", "items",
"gameplay", "scripting", "registries", "server"
],
"max_access": "admin"
}This is also how you restore the old behavior. Before this change an empty config registered all 183 tools; now an empty config gives the lean ~102. The snippet above brings the full surface back.
command_execute (and command_execute_as) are tagged write, not admin, because they are the
workhorse of almost every session. But a command can do anything the server console can —
/difficulty, /worldborder, /gamerule, /datapack, /stop. The access cap classifies the MCP
tool, not the command string it carries, so max_access: write is not an airtight sandbox: an
agent with command_execute can reach admin-level effects by typing the command directly.
For a genuinely hardened deployment, drop the command surface entirely — exclude scripting, or use
an allow-list that omits it:
{
"included_categories": ["blocks", "structures", "world", "entities", "items", "server"],
"max_access": "write"
}The category and access filters limit what an agent can easily do and shrink the tool list it reasons over; they are not a substitute for the network-level controls (loopback bind, bearer auth, Host/Origin validation) described in docs/security.md.
The mod ships one jar per Minecraft version. The v0.2.0 build matrix:
| Minecraft | Required JDK | Mappings | Mod jar suffix |
|---|---|---|---|
| 1.21.11 | 21 | Mojang official | +1.21.11 |
| 26.1.1 | 25 | unobfuscated | +26.1.1 |
| 26.1.2 | 25 | unobfuscated | +26.1.2 |
./gradlew chiseledBuild produces all three jars in one invocation. Each Stonecutter version
subproject has its own versions/<ver>/build.gradle.kts because the Fabric Loom plugin ID
differs across Minecraft versions (legacy fabric-loom for 1.21.x; net.fabricmc.fabric-loom
LoomNoRemap variant for 26.1+).
Each tool declares the Minecraft and Fabric API constraints it needs via the @McpTool
annotation, and the registration layer filters tools at startup. A tool that requires
fabric-biome-api-v1 only registers when that module is loaded; a tool restricted to MC 26.1+
won't appear in the tool list on 1.21.11. Run tools/list against your specific build to see what's
available.
Full detail in docs/version-compatibility.md.
The mod speaks MCP over Streamable HTTP. Claude Desktop connects via the
mcp-remote adapter, which bridges a stdio entry to
the remote HTTP endpoint.
{
"mcpServers": {
"minecraft-java": {
"command": "npx",
"args": ["-y", "mcp-remote", "http://localhost:8765/mcp"]
}
}
}That's the default-config form (localhost, no token). For the authenticated form and the path to Claude Desktop's config file on macOS / Windows, see docs/claude-desktop-integration.md. For Cursor, see docs/cursor-integration.md.
| Route | Method(s) | Purpose |
|---|---|---|
/mcp |
POST |
Single JSON-RPC request → JSON-RPC response |
/mcp |
GET |
Server-to-client SSE stream (no spontaneous messages in v0.2.0) |
/mcp |
DELETE |
Close session (no-op in the stateless v0.2.0 dispatcher) |
/mcp |
OPTIONS |
CORS preflight |
/healthz |
GET |
Liveness probe (no auth) |
# Build every version in the matrix (produces 3 jars)
./gradlew chiseledBuild
# Build a single version
./gradlew :1.21.11:build
./gradlew :26.1.2:build
# Switch the Stonecutter active subproject (used as the default for plain `./gradlew build`)
./gradlew "Reset active project" -Pversion=1.21.11
./gradlew buildOutput jars land in versions/<mcver>/build/libs/minecraft-fabric-mcp-<modver>+<mcver>.jar.
Requirements: Gradle 9.4.0+ (the wrapper bundles it). The build needs both JDK 21
(toolchain target for 1.21.x) and JDK 25 (toolchain target for 26.1.x; the Gradle daemon
must also run on JDK 25 because Loom enforces it at configure time). Set them via standard
env vars: JDK_21 and JDK_25, or JAVA_HOME_21_X64 and JAVA_HOME_25_X64. Gradle's
toolchain locator picks each one up automatically on CI (actions/setup-java exposes them)
and on developer machines (Corretto's Windows installer installs both into
C:\Program Files\Amazon Corretto\).
This mod is a stable foundation for separately built MCP clients and agents. Its public contract, governed by semantic versioning, is:
- tool names,
- tool input schemas,
- tool output (the
resultfield of the response envelope), - the MCP wire protocol revision (
2025-06-18in v0.2.0), - the configuration schema and environment variable names.
Internal layering (transport, runtime, adapter implementations) is not part of the contract and may change at any time.
The underlying Minecraft + Fabric API surface is not part of our contract — a Minecraft update can still break behavior even when our public contract is unchanged. Pin versions; upgrade them in lock-step.
./gradlew spotlessApply # auto-format
./gradlew check # spotless, checkstyle, tests
./gradlew chiseledTest # unit tests across every version
./gradlew "Reset active project" -Pversion=26.1.2
./gradlew runServer # launches the Fabric dev server with the mod loadedSee CONTRIBUTING.md for the development workflow, code-style notes, and PR checklist.
See SECURITY.md. Use GitHub Security Advisories — don't open a public issue.
MIT — see LICENSE.



