Skip to content

Latest commit

 

History

History
419 lines (314 loc) · 16.6 KB

File metadata and controls

419 lines (314 loc) · 16.6 KB
PrintMCP

From "I want to print a coffee cup" to a finished print — driven by your AI agent.

PrintMCP is a Model Context Protocol server that automates the entire 3D-printing pipeline: find a model, slice it to G-code, and print it — all through tools an AI assistant can call.

CI Python Protocol Built with uv Pipeline License: Apache 2.0


🧭 The pipeline

PrintMCP is built in three levels. Each level is a group of tools, and together they form one continuous workflow from a vague idea to plastic on the bed.

flowchart LR
    Start(["🗣️ User request"]) --> S1
    subgraph L1["1 · Source — Thingiverse"]
        direction TB
        S1["search_models"] --> S2["get_model"] --> S3["download_model"]
    end
    subgraph L2["2 · Slice — Cura"]
        direction TB
        SL["cura_slice_model"]
    end
    subgraph L3["3 · Print — OctoPrint"]
        direction TB
        P1["upload_file"] --> P2["start_print"] --> P3["get_job"]
    end
    S3 --> SL --> P1
    P3 --> Done(["✅ Finished print"])
Loading
Level Scope Backend Status
1 · Source Search for and download 3D model files Thingiverse REST API ✅ Implemented
2 · Slice Slice models into printer-ready G-code Ultimaker Cura (headless CuraEngine) ✅ Implemented
3 · Print Upload, start, monitor, and control prints OctoPrint REST API ✅ Implemented

📑 Table of contents


✨ Highlights

  • One server, the whole pipeline. Search → download → slice → print, without leaving your assistant.
  • License-aware sourcing. Every model's license is surfaced before you download, so you don't misuse it.
  • Real slicing, real estimates. Drives the same CuraEngine that Ultimaker Cura ships, and reports the true print time and filament usage from the engine itself.
  • Safe by default. Tools that physically actuate the printer (heaters, motors) refuse to act unless you explicitly pass confirm=true — see Safety model.
  • Structured or human output. Every tool accepts response_format (markdown or json).
  • Zero-secret-in-output design. Credentials live in .env and are sent only to the services they belong to.

📚 Documentation

Full guides, a per-tool reference, and step-by-step tutorials live in docs/.

Start here Then
📖 Documentation hub the index of everything below
🚀 Getting Started install, configure, connect a client
⚙️ Configuration every env var + where files are stored
🛡️ Safety Model how confirm=true protects your machine
🔧 Troubleshooting common errors → fixes
🏛️ Architecture how it's built (for contributors)

Tutorials — learn to 3D print by chatting with your assistant:

  1. Find Something to Print
  2. Get It Print-Ready
  3. Send It to Your Printer
  4. From Idea to Object

For developerstool reference & API: the shared invocation contract, plus per-level details for Thingiverse, Cura, and OctoPrint.


🛡️ Safety model

Warning

Level 3 drives a real machine. A bad tool call could heat a nozzle, move an axis, or abandon a print.

To prevent accidents, every physical-actuation tool defaults to a dry run. Called without confirm, it describes exactly what would happen and sends nothing to the printer. Pass confirm=true to actually act.

octoprint_start_print(path="cup.gcode")              → 🟡 dry run: "this would start printing cup.gcode"
octoprint_start_print(path="cup.gcode", confirm=true) → 🟢 starts the print

Note

Read-only tools (octoprint_get_status, octoprint_list_files, octoprint_get_job) never need confirm. Temperatures are also sanity-checked (e.g. the bed is capped at ~140 °C) so a typo can't command a wild value.


🧰 Tool reference

All tools accept response_format (markdown or json). Tools marked 🔒 physically actuate the printer and require confirm=true; without it they return a harmless dry-run preview.

Level 1 · Source — Thingiverse

Tool What it does
thingiverse_search_models Keyword search for printable "things"; returns candidates with IDs.
thingiverse_get_model Details for one thing: license, description, and downloadable files.
thingiverse_download_model Download a thing's files (printable models by default) to local disk.

Level 2 · Slice — Cura

Tool What it does
cura_slice_model Slice a local .stl/.obj/.3mf/.amf/.ply into G-code. Choose printer, layer height, infill, supports, adhesion, and temperatures; returns the G-code path plus estimated print time and filament.

Level 3 · Print — OctoPrint

Tool What it does
octoprint_get_status Connection state, printer state, and live tool/bed temperatures. 👁️
octoprint_list_files List G-code files on the server, with their server-side paths. 👁️
octoprint_get_job Active job: file, % complete, elapsed, and estimated time remaining. 👁️
octoprint_connect Open or close the printer's serial connection. 🔒
octoprint_upload_file Upload a local .gcode to the server (optionally select/print). 🔒¹
octoprint_start_print Select a server-side file and begin printing. 🔒
octoprint_control_job Pause, resume, or cancel the running job. 🔒
octoprint_set_temperature Set a tool (nozzle) or bed target temperature. 🔒
octoprint_home Home one or more axes. 🔒
octoprint_move Jog the print head by a relative offset. 🔒

👁️ read-only · 🔒 requires confirm=true · ¹ uploading alone is safe; only starting a print needs confirm.


📦 Requirements

Needed for Notes
Python ≥ 3.10 everything
uv everything Project & dependency manager.
Thingiverse API token Level 1 Free — register an app.
Ultimaker Cura Level 2 Its bundled CuraEngine is auto-detected on Windows.
An OctoPrint server + API key Level 3 Any printer running OctoPrint on your network.

Tip

The levels are independent. You can use Level 1 with just a Thingiverse token, add Cura later for slicing, and wire up OctoPrint whenever your printer is ready.


🚀 Quick start

Install from PyPI

uv tool install printmcp      # or:  pipx install printmcp

This puts a printmcp command on your PATH. You can also run it without installing:

uvx printmcp

Install from source

git clone https://github.com/SourceBox-LLC/PrintMCP
cd PrintMCP
uv sync                       # creates .venv, installs PrintMCP + dev tools

Then: configure and run

# Configure your secrets (in the working directory you'll run from)
cp .env.example .env          # Windows: copy .env.example .env
#   …then edit .env (see Configuration below)

# Run the server (stdio — it blocks waiting for an MCP client; that's expected)
printmcp                      # if installed; from source: uv run printmcp

You normally won't run the server by hand — you register it with a client, which launches it for you.


⚙️ Configuration

PrintMCP reads configuration from environment variables. A .env file in the project root is loaded automatically (and is git-ignored, so your secrets stay local).

Variable Required Default Purpose
THINGIVERSE_TOKEN Level 1 Thingiverse REST API App Token.
PRINTMCP_DOWNLOAD_DIR ~/PrintMCP/downloads Where downloaded models are saved.
PRINTMCP_CURA_DIR auto-detected Ultimaker Cura install folder (e.g. C:\Program Files\UltiMaker Cura 5.11.0). Set only if auto-detection fails.
PRINTMCP_CURAENGINE <cura>/CuraEngine.exe Full path to the CuraEngine executable, if it lives outside the Cura folder.
OCTOPRINT_URL Level 3 Base URL of your OctoPrint server, e.g. http://octopi.local or http://192.168.1.50:80.
OCTOPRINT_API_KEY Level 3 OctoPrint API key. Sent only in the X-Api-Key header to OCTOPRINT_URL.
Where do I get each credential?
  • Thingiverse token — Create an app at https://www.thingiverse.com/apps/create and copy its App Token.
  • OctoPrint API key — In OctoPrint: Settings → API (global key), or generate a per-user Application Key under your user account. Either works.

🔌 Register with an MCP client

Automatic

The setup script detects your installed MCP clients (Claude Code, Claude Desktop, Cursor, Windsurf, opencode), lets you pick one, and configures it for you:

# Windows (PowerShell)
.\scripts\setup-mcp.ps1
# macOS / Linux
./scripts/setup-mcp.sh

Important

Close the client first. MCP clients rewrite their config when they exit, so a setup that runs while the client is open gets clobbered. If the client you choose is running, the script tells you to quit it and run again. See scripts/README.md.

Manual

Claude Desktop (claude_desktop_config.json)
{
  "mcpServers": {
    "printmcp": {
      "command": "uv",
      "args": [
        "run",
        "--directory",
        "C:\\Users\\Sbuss\\Documents\\Software Development\\Projects\\PrintMCP",
        "printmcp"
      ]
    }
  }
}
Claude Code CLI
claude mcp add printmcp -- uv run --directory "C:\Users\Sbuss\Documents\Software Development\Projects\PrintMCP" printmcp

Tip

Point --directory at wherever you cloned PrintMCP. The client launches the server and talks to it over stdio.


💡 Example: coffee cup to print

You don't call tools — you just talk to your assistant, and it drives the whole pipeline. A full run sounds like this:

💬 You: "I want to 3D print a coffee cup. Find one, get it ready for my Ender 3, and start it."

🤖 Assistant: "I found a few coffee cups on Thingiverse — here are the top three…"

💬 You: "The first one."

🤖 Assistant: "Good pick — it's licensed Creative Commons (free to print). I've downloaded and sliced it for your Ender-3 Pro: about 6h 31m and ~25.5 m of filament. Before I start the physical printer — ready for me to begin?"

💬 You: "Yes, go."

🤖 Assistant: "Printing now! 🎉 Ask me any time for progress."

💬 You: (later) "How's it going?"

🤖 Assistant: "About 42% done — roughly 2h 14m left."

One request, one finished cup. The assistant handles the busywork (searching, downloading, slicing, uploading); you make the decisions that matter — which model, and the go-ahead to start.

Note

Anything that touches the physical printer — heating, moving, starting a print — pauses for your confirmation first. Everything up to slicing is purely local and needs no go-ahead. See the Safety Model.

Want the step-by-step version? The tutorials walk through it in plain conversation.


🗂️ Project structure

PrintMCP/
├── src/printmcp/
│   ├── app.py          # shared FastMCP instance
│   ├── server.py       # entry point — registers tools, runs over stdio
│   ├── config.py       # env-driven config (tokens, Cura paths, OctoPrint URL/key)
│   ├── thingiverse.py  # Level 1 — search & download
│   ├── cura.py         # Level 2 — slice via CuraEngine
│   └── octoprint.py    # Level 3 — print management
├── docs/                # full documentation (guides, tool reference, tutorials)
│   ├── README.md        # documentation hub
│   ├── getting-started.md, configuration.md, safety.md, troubleshooting.md, architecture.md
│   ├── tools/           # per-level tool reference
│   └── tutorials/       # step-by-step walkthroughs
├── tests/               # offline tests (validation + mock-transport HTTP)
├── .env.example
├── pyproject.toml
└── README.md

🧪 Development

uv run pytest                       # tests
uvx ruff check src tests            # lint
uvx ruff format --check src tests   # formatting

CI runs all three on every push and PR (tests across Python 3.10–3.13).

The suite is fully offline — it requires no token, network, or Cura install and runs in under two seconds. It covers:

  • Input validation & helpers — tool registration, filename sanitization, slice-input ranges, stats parsing.
  • The Level 3 safety gate — every actuating tool refuses to act (and sends nothing) without confirm=true.
  • OctoPrint HTTP plumbing — using a mock transport, it asserts the exact method, path, JSON body, and X-Api-Key header of every request, that responses parse correctly, and that error statuses (401/409/ connection-refused) map to friendly messages — all without a printer.

Notable changes are tracked in CHANGELOG.md; releasing is documented in docs/RELEASING.md.


🗺️ Roadmap

  • More print backends — a Moonraker/Klipper backend exposing moonraker_* tools alongside the OctoPrint ones, so non-OctoPrint printers work too.
  • Slicing depth — more printer profiles and quality presets, and surfacing CuraEngine warnings (e.g. a model larger than the build volume).
  • More model sources — Printables, MyMiniFactory, and others behind a shared search interface.

⚖️ Model licensing

Models on Thingiverse carry their own licenses (often Creative Commons; some are non-commercial). thingiverse_get_model and thingiverse_download_model surface each model's license — respect it before reusing, remixing, or selling a print.


📄 License

Licensed under the Apache License 2.0 — © 2026 SourceBox LLC.

You may freely use, modify, and distribute this software, including for commercial purposes, provided you retain the copyright and license notices and state any significant changes. The license also includes an express patent grant. It comes with no warranty. See the LICENSE file for the full text.

Note

This license covers PrintMCP's own code only. Third-party tools it drives (Ultimaker Cura, OctoPrint) and any 3D models you download carry their own licenses — see Model licensing above.