This project provides a Model Context Protocol (MCP) server wrapper around the esp-rainmaker-cli
Python library.
It allows MCP-compatible clients (like LLMs or applications such as Cursor, Claude Desktop, and Windsurf) to interact with your ESP RainMaker devices using the official CLI.
The Model Context Protocol (MCP) is a standardized framework that enables AI systems to interact with external tools, data sources, and services in a unified manner. Introduced by Anthropic and adopted by major AI organizations, MCP acts as a universal interface, much like USB-C for hardware, allowing seamless integration across different platforms.
- Unified Interaction: MCP allows AI models to access and control IoT devices using natural language prompts, making interactions more intuitive and accessible.
- Real-time Control: With MCP, users can execute actions such as turning devices on/off, adjusting settings, and managing schedules directly through AI interfaces.
- Local Server, Cloud-Backed Control: The ESP RainMaker MCP server runs locally and stores credentials on your machine. However, device management actions are performed via the official ESP RainMaker cloud APIs through the esp-rainmaker-cli.
By integrating MCP, the ESP RainMaker platform enhances its capabilities, allowing tools like Claude, Cursor, Windsurf, and Gemini CLI to manage IoT devices efficiently and securely.
- Python: Version 3.10 or higher
- uv: The
uv
Python package manager. Install from Astral's uv documentation. - ESP RainMaker CLI Login: You must have successfully logged into ESP RainMaker using the standard
esp-rainmaker-cli login
command in your terminal at least once. This server relies on the credentials saved by that process. - RainMaker Nodes added into your account since onboarding isn't supported by the MCP server.
-
Clone the Repository:
git clone https://github.com/espressif/esp-rainmaker-mcp.git cd esp-rainmaker-mcp
-
Install Dependencies using uv: This command installs
esp-rainmaker-cli
,mcp[cli]
, and any other dependencies listed inpyproject.toml
into a virtual environment managed byuv
.uv sync
(This assumes
uv
is installed) -
Login to ESP Rainmaker using
esp-rainmaker-cli
uv run esp-rainmaker-cli login
Note
Direct login via username/password within MCP is not supported for security reasons. Please use the standard CLI login flow first.
To add this project as an MCP server in supported MCP clients (Cursor, Claude Desktop, Windsurf, and Gemini CLI), you'll need to add the same JSON configuration to each client's config file. The configuration is identical across all clients:
Use the following JSON configuration for all MCP clients:
{
"mcpServers": {
"ESP-RainMaker-MCP": {
"command": "uv",
"args": [
"run",
"--with",
"esp-rainmaker-cli",
"--with",
"mcp[cli]",
"mcp",
"run",
"<absolute_path_to_repo>/server.py"
]
}
}
}
Important
Replace <absolute_path_to_repo>/server.py
with the actual absolute path to the server.py
file within the cloned esp-rainmaker-mcp
directory on your system.
-
Open Cursor and click on the settings (gear icon) at the top right.
-
Navigate to "Tools & Integrations" from the settings menu.
-
Click on "MCP Tools" in the integrations section.
-
Click on "New MCP Server" to add a new server.
-
This will open the mcp.json file. Add the JSON configuration shown above.
-
Open Claude Desktop and go to Settings -> Developer -> Edit Config.
-
This will open the configuration file (claude_desktop_config.json). Add the JSON configuration shown above.
-
Save the changes and restart Claude Desktop to apply the new settings.
-
Open Windsurf and look for the hammer-type icon under the chat text input area.
-
Click on the hammer icon and select "Configure" from the options. This will open the plugins window.
-
Click on "View raw config" which will show you the
~/.codium/windsurf/mcp_config.json
file. -
Add the JSON configuration shown above to the file.
-
Save the changes and click on "Refresh" under the chat text window to load the ESP RainMaker MCP tools.
- Locate your Gemini CLI settings file. On macOS, this is typically at
~/.gemini/settings.json
. - Open the
settings.json
file in your preferred text editor. - Add the JSON configuration shown above to the
mcpServers
section of the file. If the section does not exist, create it as shown in the example. - Save the file and restart Gemini CLI if it is running.
Note
The configuration for all four applications (Cursor, Claude Desktop, Windsurf, and Gemini CLI) is the same, so you can use the same JSON structure for all of them.
Note
The --with
arguments ensure uv
includes the necessary dependencies when running the mcp run
command.
This server acts as a bridge. It uses the mcp
library to handle the Model Context Protocol communication. When a tool is called:
- It uses functions from the installed
esp-rainmaker-cli
library. - The library functions read locally stored authentication tokens.
- It makes the necessary API calls to the ESP RainMaker cloud.
- It returns the results (or errors) back through the MCP protocol.
This MCP server exposes the following tools for interacting with ESP RainMaker:
login_instructions()
:- Provides instructions (formatted with Markdown) on how to log in using the standard
esp-rainmaker-cli login
command in your terminal. This server relies on the external CLI's browser-based login flow to securely store credentials. Rendering as Markdown depends on the MCP client's capabilities.
- Provides instructions (formatted with Markdown) on how to log in using the standard
check_login_status()
:- Checks if a valid login session exists based on credentials stored locally by
esp-rainmaker-cli
. Confirms if the server can communicate with the ESP RainMaker backend.
- Checks if a valid login session exists based on credentials stored locally by
get_nodes()
:- Lists all node IDs associated with the logged-in user.
get_node_details(node_id: str = None, fields: str = None, name: str = None, type_: str = None)
:- Get detailed information for nodes including config, status, and params.
- Supports filtering and field selection:
fields
: comma-separated list of fields to include (e.g. "node_id,name,type,config,params,status.connectivity,fw_version,mapping_timestamp")name
: substring match (user-visible name from params)type_
: substring match (device type)node_id
: single node ID (for one node) or None (for all)
- Returns a dict (single node) or list of dicts (all nodes).
- Example:
get_node_details(ctx, fields="node_id,name,type")
get_node_status(node_id: str)
:- Get the online/offline connectivity status for a specific node ID.
get_params(node_id: str)
:- Get current parameter values for a device.
set_params(node_id: str, params_dict: dict)
:- Set parameters for one or more devices.
node_id
: Single ID or comma-separated list (e.g., "light1,light2")params_dict
: Parameters to set, e.g.,{"Light": {"Power": true}}
get_schedules(node_id: str)
:- Get schedules for a device.
set_schedule(node_id: str, operation: str, ...)
:- Manage device schedules.
operation
: "add", "edit", "remove", "enable", or "disable"- For add/edit: Provide
name
,trigger
, andaction
- Common triggers:
- Daily 8 AM:
{"m": 480, "d": 127}
- Weekdays 6:30 PM:
{"m": 1110, "d": 31}
- Daily 8 AM:
- Example action:
{"Light": {"Power": true}}
-
create_group(name: str, group_type: str = None, ...)
:- Create a home or room.
- Required:
name
,group_type
("home" or "room") - For rooms:
parent_group_id
required - Example:
create_group("Living Room", "room", parent_group_id="home_id")
-
get_group_details(group_id: str = None, include_nodes: bool = False)
:- Get group information. For all groups, use
group_id=None
. - Set
include_nodes=True
to include device details. - Returns: Group hierarchy, members, and metadata.
- Get group information. For all groups, use
-
update_group(group_id: str, ...)
:- Update group properties or manage devices.
- Optional:
name
,description
,add_nodes
,remove_nodes
- Examples:
- Rename:
update_group("group_id", name="New Name")
- Add devices:
update_group("group_id", add_nodes="light1,light2")
- Rename:
-
add_device_to_room(device_node_id: str, room_group_id: str)
:- Add device to room (handles parent group automatically).
- Example:
add_device_to_room("light1", "kitchen_id")
This project is licensed under the terms specified in the LICENSE file.