loading…
Search for a command to run...
loading…
An MCP server for TouchDesigner that lets you control TouchDesigner with Claude
An MCP server for TouchDesigner that lets you control TouchDesigner with Claude
An MCP server for TouchDesigner that lets you control TouchDesigner with Claude (or any Model Context Protocol client). Create operators, set parameters, and wire networks inside a running TouchDesigner project just by describing what you want — "make a noise texture chained into a level" — and watch the operators appear in your patch.
Built for Claude Code but works with any MCP-compatible client (Cline, Continue, Claude Desktop, etc.).
MCP Client ──stdio──▶ server.js ──HTTP :9980──▶ Web Server DAT ─▶ /project1
(Claude…) (this repo) (inside TD) (your network)
node -v. Install via brew install node or nodejs.org.npm install -g @anthropic-ai/claude-code, then run claude once to log in.git clone https://github.com/aliphi/touchdesigner-mcp.git
cd touchdesigner-mcp
npm install
That installs the two dependencies (@modelcontextprotocol/sdk, zod). Don't run node server.js yourself — Claude Code will start it for you in step 3.
Shortcut: open Touchdesigner/TouchdesignerClaudeMCP.toe — it's a ready-to-use TD project with the Web Server DAT already created, configured on port 9980, and wired to the callback below. Open it, make sure the Web Server DAT's Active toggle is on, and skip to step 3.
Otherwise, to set it up by hand in your own project:
/project1).9980Verify it's working: open http://localhost:9980 in your browser. You should see:
{"status": "connected", "project": "project1"}
If you don't, the DAT isn't Active or the port isn't 9980. Fix that before continuing.
From inside the touchdesigner-mcp folder, run:
claude mcp add touchdesigner -- node "$(pwd)/server.js"
That's it — Claude Code now knows how to launch the server.
Edit ~/.claude.json and add a touchdesigner entry under mcpServers:
{
"mcpServers": {
"touchdesigner": {
"command": "node",
"args": ["/absolute/path/to/touchdesigner-mcp/server.js"]
}
}
}
Replace /absolute/path/to/ with your real path.
Make sure TD is open with the Web Server DAT active, then in any terminal run claude and try:
List what's in my TouchDesigner project
Create a noiseTOP feeding a levelTOP, then a compositeTOP
Build me a basic 3D scene: a sphere, a camera, a light, and a renderTOP that outputs it all
Claude will use the tools below to build the network inside your running TD project. Switch back to TD to watch the operators appear.
| Tool | What it does |
|---|---|
td_list_operators |
List everything at a given path. Always run this first. |
td_create_operator |
Create a new operator (TOP/CHOP/SOP/DAT/COMP) at a position |
td_set_parameter |
Set a single parameter on an operator |
td_connect |
Wire one operator's output into another's input |
td_run_python |
Escape hatch — run arbitrary Python inside TD |
Conventions Claude follows (defined in CLAUDE.md):
/project1server.js Node MCP server (the thing Claude launches)
td_webserver_callback.py Paste this into TD's Web Server DAT
CLAUDE.md Conventions Claude follows when building networks
package.json Node deps
Touchdesigner/TouchdesignerClaudeMCP.toe Ready-to-use TD project with the Web Server DAT pre-wired
"Error connecting to TouchDesigner: fetch failed"
TD isn't reachable. Check, in order: (a) TD is open, (b) the Web Server DAT exists, (c) its port is 9980, (d) Active is on, (e) http://localhost:9980 returns the JSON above in your browser.
Claude says it doesn't have TouchDesigner tools
The MCP server isn't registered. Run claude mcp list — touchdesigner should appear. If it doesn't, redo step 3 with an absolute path. After registering, fully quit and reopen Claude Code.
"Cannot find module '@modelcontextprotocol/sdk'"
You skipped npm install. Run it inside the touchdesigner-mcp folder.
Operators get created but td_set_parameter errors with "no attribute 'par.foo'"
That parameter name doesn't exist on that operator type. Ask Claude to use td_run_python with dir(op('/project1/x').par) to list valid parameters.
Port 9980 already in use
Either kill whatever's using it (lsof -i :9980) or change the port in both server.js (the TD_PORT constant near the top) and the Web Server DAT in TD.
The MCP server has no domain logic — it serializes Python strings and POSTs them to http://localhost:9980. The Web Server DAT's callback tries eval(code) first, and if that's a SyntaxError, falls back to exec(code). The last expression in the script becomes the HTTP response body, which is why every built-in tool ends with an f-string summary like f"Connected {…} -> {…}".
Security note: the callback runs arbitrary Python with no auth. Keep localhost only — don't expose port 9980 to your network.
Run in your terminal:
claude mcp add touchdesigner-mcp -- npx