MCP server to drive Gearotons M17 open-source servomotors from natural language (Claude / any MCP client) over RS-485. Auto-discovers serial ports and motors; exposes the full firmware command set.
Project description
servomotor-mcp
Drive open-source Gearotons M17 servomotors from natural language.
An MCP server that exposes the M17 — a NEMA-17 integrated, closed-loop, RS-485 servomotor — to Claude Desktop, Claude Code, or any MCP client. Control real motors by just asking:
"Find my motors and rotate the one on the bench two full turns, slowly."
Nothing is hardcoded: the server discovers your serial ports, auto-detects the
motors on the bus (the firmware's "Detect devices" command), and exposes the entire
firmware command set — every command in the servomotor library's catalog becomes an
MCP tool automatically (48 commands as of library 0.10.0), plus a few high-level tools
for everyday moves. It ships with a mock backend, so you can try the whole thing with
no hardware.
The first servomotor with an official MCP server. Open hardware, open firmware, open software — and now an open, AI-native control interface.
Quickstart (no hardware, ~2 minutes)
# Run the server directly with uv (recommended):
uvx --from servomotor-mcp servomotor-mcp
# or install it:
pip install servomotor-mcp
servomotor-mcp
Then add it to Claude Desktop — copy the block from
examples/claude_desktop_config.json into your
claude_desktop_config.json, restart Claude Desktop, and ask:
"What serial ports do you see? Connect and find my motors."
See examples/demo_prompts.md for a scripted demo.
Drive real motors
Plug an M17 (or a daisy-chain of them) into a USB↔RS-485 adapter and install the
[serial] extra — that's it, no configuration:
pip install 'servomotor-mcp[serial]' # pulls in the Gearotons servomotor library
servomotor-mcp
With the servomotor library installed the server uses the real serial backend
automatically. In a session, the model then:
list_serial_ports— enumerates the machine's ports (macOS/dev/cu.*, WindowsCOM*, Linux/dev/ttyUSB*), flagging USB serial adapters;connect— opens the port you (or it) picked, at 230400 baud;- auto-detects every motor on that bus (unique ID + alias) — no address maps to write;
- drives them. Tell it in plain English which adapter to use if you have several.
Tools
High-level (discovery + everyday motion):
| Tool | What it does |
|---|---|
list_serial_ports |
Enumerate serial ports with USB metadata (call first). |
connect / disconnect |
Open a port and auto-detect the motors on that bus. |
detect_devices |
Re-scan the bus (reboots the motors on it). |
list_motors |
Detected motors with live position/voltage/temperature/status. |
move_to / move_relative |
Absolute / relative moves in degrees; waits for completion. |
stop |
Emergency-stop one or all motors. |
get_motor_status |
One motor's snapshot, fatal errors decoded to plain English. |
run_sequence |
Choreographed steps ("draw a square"), incl. raw command steps. |
Plus one tool per firmware command, generated from the library's command catalog:
enable_mosfets, go_to_position, move_with_velocity, move_with_acceleration,
multimove, homing, zero_position, get_position, get_temperature,
set_device_alias, set_pid_constants, system_reset, vibrate, ping, … — anything
the motor can do, the model can do. Motors are addressed by their alias number, their
16-hex-digit unique ID, or "all" (broadcast). Values are in friendly units (degrees,
seconds, degrees/s, volts, °C); the server converts to firmware units.
How it works
natural language → Claude → MCP tool calls → this server → RS-485 → M17 motors
The server is a thin layer over the Gearotons servomotor Python library. The library is
data-driven — motor_commands.json defines every firmware command — and the server turns
that same catalog into MCP tools, so new library commands appear automatically. Tool calls
are forwarded straight to the hardware — no software clamping; full multi-turn travel, any
speed. The motor's own firmware protections (over-current / over-voltage /
over-temperature) still apply. The same tools run against the mock backend
(GEAROTONS_MOTOR_BACKEND=mock) for development and CI.
Environment variables (all optional):
GEAROTONS_MOTOR_BACKEND—auto(default: serial when theservomotorlibrary is installed, else mock),serial, ormock.GEAROTONS_SERIAL_PORT— default port forconnectwhen the model doesn't pass one.GEAROTONS_DEFAULT_SPEED_DPS— default speed formove_to/move_relative(180).
Develop / test
pip install -e '.[dev]'
GEAROTONS_MOTOR_BACKEND=mock pytest # catalog + mock-bus + server-tool tests
hardware_tests/ contains scripts that exercise the real serial path end to end
(port sweep, full command suite, stdio MCP session) against a bench motor.
Status
- ✅ Full firmware command surface (48 commands), serial-port discovery, bus
auto-detection — verified on a physical M17 (fw 0.15.3.0) over a real stdio MCP
session and via
uvx, on all four test adapters (motor found only where it truly is). - ✅ Mock backend + 39 unit tests, no hardware needed.
- ✅ Cross-platform port handling (macOS / Windows / Linux) via pyserial enumeration.
License
MIT. Hardware, firmware, and software for the M17 are open-source — see github.com/tomrodinger/servomotor.
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file servomotor_mcp-0.3.0.tar.gz.
File metadata
- Download URL: servomotor_mcp-0.3.0.tar.gz
- Upload date:
- Size: 114.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7b90007f54ae05bb2f0b705aad9a801bc48a468a17d4780cc54dc47311f2ec08
|
|
| MD5 |
2ffa675735c125ef892f73c9856f959f
|
|
| BLAKE2b-256 |
5d7f28faf787f43bce3a80cb8f76eb353ae13b88d05760167e3f3a7888152eaa
|
Provenance
The following attestation bundles were made for servomotor_mcp-0.3.0.tar.gz:
Publisher:
publish.yml on Gearotons/servomotor-mcp
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
servomotor_mcp-0.3.0.tar.gz -
Subject digest:
7b90007f54ae05bb2f0b705aad9a801bc48a468a17d4780cc54dc47311f2ec08 - Sigstore transparency entry: 2053613454
- Sigstore integration time:
-
Permalink:
Gearotons/servomotor-mcp@54ead742b28611a866bf51207dc56432ff13b53c -
Branch / Tag:
refs/tags/v0.3.0 - Owner: https://github.com/Gearotons
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@54ead742b28611a866bf51207dc56432ff13b53c -
Trigger Event:
push
-
Statement type:
File details
Details for the file servomotor_mcp-0.3.0-py3-none-any.whl.
File metadata
- Download URL: servomotor_mcp-0.3.0-py3-none-any.whl
- Upload date:
- Size: 41.9 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
007e3b43b1955277e004738e811b7c099fd68a722fa66c6ce8e376b2b3e03faf
|
|
| MD5 |
b3f21455384fe8d75ce8c665398fdbdf
|
|
| BLAKE2b-256 |
88d44b694d4c5f25e2a7e2bdf0b5b80bd10f64ccc2d929cfbce26547378ffd54
|
Provenance
The following attestation bundles were made for servomotor_mcp-0.3.0-py3-none-any.whl:
Publisher:
publish.yml on Gearotons/servomotor-mcp
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
servomotor_mcp-0.3.0-py3-none-any.whl -
Subject digest:
007e3b43b1955277e004738e811b7c099fd68a722fa66c6ce8e376b2b3e03faf - Sigstore transparency entry: 2053613897
- Sigstore integration time:
-
Permalink:
Gearotons/servomotor-mcp@54ead742b28611a866bf51207dc56432ff13b53c -
Branch / Tag:
refs/tags/v0.3.0 - Owner: https://github.com/Gearotons
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@54ead742b28611a866bf51207dc56432ff13b53c -
Trigger Event:
push
-
Statement type: