tuya-smart-control
Control Tuya smart home devices via natural language. Use when the user asks to control smart devices (turn on/off lights, AC, plugs, adjust brightness/temperature/mode), query device status or list devices, manage homes and rooms, rename devices, check weather by location, send notifications (SMS, voice call, email, or App push), view device data statistics (e.g. energy/power consumption), capture snapshots/short videos from IPC cameras, or subscribe to real-time device events (property changes, online/offline status) via WebSocket. Requires TUYA_API_KEY.
git clone https://github.com/tuya/tuya-openclaw-skills
T=$(mktemp -d) && git clone --depth=1 https://github.com/tuya/tuya-openclaw-skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/tuya-smart-control" ~/.claude/skills/tuya-tuya-openclaw-skills-tuya-smart-control && rm -rf "$T"
T=$(mktemp -d) && git clone --depth=1 https://github.com/tuya/tuya-openclaw-skills "$T" && mkdir -p ~/.openclaw/skills && cp -r "$T/tuya-smart-control" ~/.openclaw/skills/tuya-tuya-openclaw-skills-tuya-smart-control && rm -rf "$T"
tuya-smart-control/SKILL.mdTuya Smart Home Device Control Skill
Basic Information
- Official Website: https://www.tuya.com/
- Source Code: https://github.com/tuya/tuya-openclaw-skills
- Authentication: Via Header
Authorization: Bearer {Api-key} - Credentials: Read from environment variable
. Base URL is auto-detected from API key prefix. SeeTUYA_API_KEY
for the prefix-to-region mapping table. You can override by settingreferences/api-conventions.md
.TUYA_BASE_URL - API Reference: See individual files under
references/ - Python SDK: See
scripts/tuya_api.py - Device Message Client: See
(real-time WebSocket subscription)scripts/tuya_device_mq_client.py
Environment Variable Configuration
Set the following environment variable before use:
export TUYA_API_KEY="your-tuya-api-key" # TUYA_BASE_URL is optional — auto-detected from API key prefix # Override only if needed: export TUYA_BASE_URL="https://openapi.tuyaus.com"
The same
TUYA_API_KEY is used for both the REST API and WebSocket message subscription. The WebSocket URI is auto-detected from the API key prefix (same 7 data centers as the REST API). See references/device-message.md for the full mapping table.
The skill will not load if the
TUYA_API_KEY environment variable is missing.
Usage
Always prefer Method 1 (Command Line) — single command, no boilerplate code. It handles authentication, URL resolution, JSON serialization, and error handling automatically.
Method 1: Via Command Line (Recommended)
python3 {baseDir}/scripts/tuya_api.py <command> [params...] # Examples: python3 {baseDir}/scripts/tuya_api.py homes python3 {baseDir}/scripts/tuya_api.py devices python3 {baseDir}/scripts/tuya_api.py devices --home 5053559 python3 {baseDir}/scripts/tuya_api.py devices --room 123456 python3 {baseDir}/scripts/tuya_api.py device_detail <device_id> python3 {baseDir}/scripts/tuya_api.py model <device_id> python3 {baseDir}/scripts/tuya_api.py control <device_id> '{"switch_led":true}' python3 {baseDir}/scripts/tuya_api.py rename <device_id> "New Name" python3 {baseDir}/scripts/tuya_api.py weather 39.90 116.40 python3 {baseDir}/scripts/tuya_api.py sms "Your message" python3 {baseDir}/scripts/tuya_api.py voice "Your message" python3 {baseDir}/scripts/tuya_api.py mail "Subject" "Content" python3 {baseDir}/scripts/tuya_api.py push "Subject" "Content" python3 {baseDir}/scripts/tuya_api.py stats_config python3 {baseDir}/scripts/tuya_api.py stats_data <dev_id> <dp_code> <type> <start> <end> python3 {baseDir}/scripts/tuya_api.py ipc_pic_fetch <device_id> <consent> [pic_count] [home_id] python3 {baseDir}/scripts/tuya_api.py ipc_video_fetch <device_id> <duration> <consent> [home_id]
CLI validation rules:
supports only one scope flag at a time:devices
or--home <id>--room <id>
requirescontrol
to be a valid JSON object (not array/string)properties_json
validates coordinate range: latitudeweather
, longitude[-90, 90][-180, 180]
validatesstats_data
/start
formatend
and max 24-hour windowyyyyMMddHH
args:ipc_pic_fetch
— consent<device_id> <consent> [pic_count] [home_id]
= decrypted URL1
args:ipc_video_fetch
— duration in seconds (1-60)<device_id> <duration> <consent> [home_id]- Use
for command help and examplespython3 {baseDir}/scripts/tuya_api.py --help
Method 2: Via Python SDK
Use when you need to chain multiple API calls or do complex logic in a single script:
import sys sys.path.insert(0, "{baseDir}/scripts") from tuya_api import TuyaAPI api = TuyaAPI() homes = api.get_homes() devices = api.get_all_devices() detail = api.get_device_detail("device_id_here") result = api.issue_properties("device_id_here", {"switch_led": True, "bright_value": 500}) weather = api.get_weather(lat="39.90", lon="116.40") # IPC cloud capture — take a snapshot and get decrypted URL capture = api.ipc_ai_capture_pic_allocate_and_fetch("device_id_here", user_privacy_consent_accepted=True)
Method 3: Device Message Subscription (WebSocket)
Use when you need real-time device event monitoring (property changes, online/offline status):
import asyncio import os import sys sys.path.insert(0, "{baseDir}/scripts") from tuya_device_mq_client import TuyaDeviceMQClient async def main(): # Uses TUYA_API_KEY for auth; WebSocket URI auto-detected from key prefix client = TuyaDeviceMQClient( api_key=os.environ["TUYA_API_KEY"], device_ids=None, # None = all devices; or pass a list of device IDs ) @client.on_property_change async def on_prop(device_id, properties): for prop in properties: t = TuyaDeviceMQClient.format_timestamp(prop["time"]) print(f"[{t}] Device {device_id}: {prop['code']} = {prop['value']}") @client.on_online_status async def on_status(device_id, status, timestamp_ms): t = TuyaDeviceMQClient.format_timestamp(timestamp_ms) print(f"[{t}] Device {device_id} is now {status}") await client.connect() asyncio.run(main())
Important: The WebSocket client runs server-side only. It reuses the same
— no separate credentials needed. The WebSocket URI is auto-detected from the key prefix (same 7 data centers as the REST API). Notification throttling (minimum 30-minute cooldown) is mandatory when triggering notifications from device events. SeeTUYA_API_KEYfor message format details and more examples.references/device-message.md
Feature Overview
| Module | Capabilities | Reference |
|---|---|---|
| Home Management | List all homes, list rooms in a home | |
| Device Query | All devices, devices by home/room, single device detail (including current property states) | |
| Device Control | Query device Thing Model, issue property commands | |
| Device Management | Rename device | |
| Weather Service | Current and forecast weather | |
| Notifications | SMS, voice call, email, App push | |
| Data Statistics | Hourly statistics config query, statistics value query | |
| IPC Cloud Capture | Cloud snapshot and short video capture for IPC cameras | |
| Device Message Subscription | Real-time WebSocket subscription for device property changes and online/offline events | |
| Error Handling | Error codes and recovery strategies | |
| API Conventions | Request/response format, data center mapping | |
Core Workflows
Workflow 1: Device Control
When the user says things like "turn on the living room light" or "set the AC temperature to 26 degrees":
-
Locate the device — Find the target device based on the device name or location mentioned by the user. Follow this priority:
- Priority 1 — Room + category match: If the user mentions a room (e.g. "living room AC"), first query the home list → room list to match the room, then list devices in that room and match by
or devicecategory_namename - Priority 2 — Device name match: If the user only mentions a device name (e.g. "AC"), call "List All Devices" API and match by
first, then by devicecategory_name
fuzzy matchname - Priority 3 — Disambiguation: If multiple devices match, list all candidates with their room information and ask the user to choose
- Priority 1 — Room + category match: If the user mentions a room (e.g. "living room AC"), first query the home list → room list to match the room, then list devices in that room and match by
-
Get current state — Call the "Get Single Device Detail" API
- If
isresult
: the device does not exist or you have no permission — inform the user and stopnull - If
isonline
: the device is offline — tell the user "Device XX is currently offline, please check its power and network connection" and do not proceed furtherfalse - Only continue when
is valid andresult
isonlinetrue - The
field contains current values of each functional property (e.g. switch state, brightness, temperature)properties
- If
-
Query capabilities — Call the "Query Device Thing Model" API to get the device's supported property list
- Important: The
field is a JSON string that must be parsed again (e.g.result.model
) to obtain the property definitionsjson.loads(result["model"]) - Check each property's
:accessMode
(read-only): cannot be controlled, only queried — inform the user "this property is read-only"ro
(write-only): can be controlled but current value cannot be readwr
(read-write): can be both controlled and queriedrw
- Important: The
-
Map the command — Map the user's intent to Thing Model properties:
- Turn on/off → find a bool-type switch property (e.g.
,switch_led
)switch - Adjust brightness → find a value-type brightness property
- Adjust temperature → find a value-type temp property
- If the device does not support the requested function, inform the user and list supported functions
- Relative adjustments — When the user says "a bit brighter", "lower the temperature by 2 degrees", etc.:
- Read the current value from
in the device detail (Step 2)properties - Read
,min
,max
from the Thing Modelstep
(Step 3)typeSpec - Calculate the target value:
- Vague ("a bit", "a little") → current value ± (max - min) × 10%
- Specific ("by 2 degrees", "by 100") → current value ± the specified amount
- Clamp the target value within [min, max] and round to the nearest
step
- Read the current value from
- Validate value range: Before issuing, confirm the target value is within the
min/max rangetypeSpec
- Turn on/off → find a bool-type switch property (e.g.
-
Issue the command — Call the "Issue Properties" API using the Python SDK:
api.issue_properties(device_id, {property_code: value})- The SDK handles
JSON string serialization automaticallyproperties - If not using the SDK: the
field must be a JSON string, not a JSON object. You must double-serialize:properties{"properties": "{\"switch_led\":true}"}
- The SDK handles
-
Verify and return result — After issuing the command:
- Wait 1-2 seconds, then call "Get Device Detail" again to read the updated
properties - Compare the target value with the actual value to confirm execution
- If values match: inform the user the operation succeeded
- If values differ: tell the user "command sent, but the device state has not updated yet — there may be a delay"
- Wait 1-2 seconds, then call "Get Device Detail" again to read the updated
Workflow 2: Rename Device
- Locate the device using Workflow 1 Step 1 to obtain the device_id
- Call the "Rename Device" API with the new name
- Return the result
Workflow 3: Notifications
- Identify the message type: SMS / Voice / Email / App Push
- Extract required parameters (message content; email and push also need a subject)
- Call the corresponding API (all notification APIs are self-send — messages can only be sent to the current logged-in user)
- Return the send result
Workflow 4: Weather Query
- Obtain coordinates:
- First call Home List API and check the
/latitude
fieldslongitude - Note: the coordinate format is
— you must extract the{"Value": "30.3"}
field (e.g..Value
)home["latitude"]["Value"] - If the home has no location set, ask the user for their city and convert to coordinates (see common city coordinates in
)references/weather.md
- First call Home List API and check the
- Determine which weather attributes to query (default: temperature, humidity, weather condition)
- Call the weather query API
- Translate the returned data into a human-readable description
Workflow 5: Data Statistics
- Locate the device (same as Workflow 1 Step 1)
- Call the "Statistics Config Query" API to confirm whether the device has the corresponding statistics capability
- If available, call the "Statistics Value Query" API
- Time inference: Convert the user's natural language to
format:yyyyMMddHH- "today" → start = today 00:00, end = current hour
- "yesterday" → start = yesterday 00:00, end = yesterday 23:00
- The time range cannot exceed 24 hours per request — for longer ranges, make multiple requests and aggregate
- Format example:
= January 1, 2024 00:002024010100
- Time inference: Convert the user's natural language to
- Aggregate and return the results
Workflow 6: Device Status Query
When the user asks "Is the living room light on?" or "What's the AC set to?":
- Locate the device and get current state (same as Workflow 1 Steps 1-2; stop if device not found or offline)
- Read
values, cross-reference with the Thing Model property names/descriptions, and translate to natural language (e.g.properties
→ "the light is currently on")"switch_led": true
Workflow 7: Multi-Device Batch Control
When the user says "Turn off all lights" or "Set all ACs to 26 degrees":
- Call "List All Devices" API and filter matching devices by
or devicecategory_name
keywordname - For each matching device: check
status (skip offline devices and note them), then execute Workflow 1 Steps 3-6online - Aggregate results: report how many devices succeeded, which ones failed or were offline
- Add a brief delay (0.5-1s) between requests to avoid rate limiting
Workflow 8: IPC Cloud Capture
When the user asks to "take a photo with the camera" or "record a short video from the camera":
- Locate the IPC device — same as Workflow 1 Step 1, filter by camera category
- Determine capture type:
- Snapshot →
(optionalPIC
, 1-5)pic_count - Short video →
(optionalVIDEO
, 1-60, default 10)video_duration_seconds
- Snapshot →
- Privacy consent — Only set
when the user has explicitly agreed to receive decrypted playable URLs. Default touser_privacy_consent_accepted=true
unless the user declinestrue - Execute capture — Use the all-in-one helper methods:
- For PIC:
api.ipc_ai_capture_pic_allocate_and_fetch(device_id, user_privacy_consent_accepted=True, pic_count=1) - For VIDEO:
api.ipc_ai_capture_video_allocate_and_fetch(device_id, video_duration_seconds=5, user_privacy_consent_accepted=True) - These methods handle the full allocate → wait → poll → retry flow automatically
- For PIC:
- Return the result — Extract the URL from the resolve result:
- PIC with consent:
resolve["decrypt_image_url"] - VIDEO with consent:
(cover image may be null if still uploading)resolve["decrypt_video_url"] - If
is stillstatus
after all retries, inform the user that the device may be slow to upload and suggest trying again laterNOT_READY
- PIC with consent:
Workflow 9: IPC Visual Recognition
When the user asks "What's in front of my camera?", "Is there anyone at the door?", or "Describe what the camera sees":
- Capture a snapshot — Follow Workflow 8 Steps 1-4 to take a PIC capture with
user_privacy_consent_accepted=True - Get the image URL — Extract
from the capture result. If the resolve failed or returnedresolve["decrypt_image_url"]
, inform the user and stopNOT_READY - Download the image — Fetch the image content from the decrypted URL
- Send to AI vision model — Pass the image to the AI large model for visual understanding. Describe the image content in natural language based on the user's question:
- General question ("What's there?") → describe the overall scene, objects, and people
- Specific question ("Is there a package?", "Is anyone at the door?") → focus on answering the specific question
- Return the description — Respond to the user with the visual analysis result in conversational language
Workflow 10: Real-Time Device Monitoring
When the user asks to "monitor device changes in real time", "watch for property updates", or "notify me when a device goes offline":
- Determine scope — Ask which devices to monitor (all or specific device IDs). If specific, locate devices using Workflow 1 Step 1
- Determine event types — Property changes (
), online/offline status (on_property_change
), or bothon_online_status - Write the subscription script — Using
fromTuyaDeviceMQClient
:scripts/tuya_device_mq_client.py- Import and instantiate with
(WebSocket URI auto-detected from key prefix)api_key=os.environ["TUYA_API_KEY"] - Register appropriate handlers using decorators
- Call
to start listeningawait client.connect()
- Import and instantiate with
- Apply throttling — If the subscription triggers notifications or device control actions, implement a cooldown mechanism (minimum 30-minute interval for notifications)
- Cross-reference with REST API — Property codes from WebSocket events correspond to Thing Model codes. Use
to look up property names and value ranges when neededapi.get_device_model(device_id)
Workflow 11: Event-Driven Automation
When the user asks to "turn on the hallway light when the door opens" or "send me a notification when the AC turns off":
- Identify trigger and action — Parse the trigger device, trigger condition (property code + value), and the action to execute
- Locate devices — Use Workflow 1 Step 1 to find both the trigger device and the action device
- Write the automation script — Combine
for event listening withTuyaDeviceMQClient
for device control:TuyaAPI- Subscribe to the trigger device's property changes
- When the trigger condition is met, call
to control the action deviceapi.issue_properties() - Implement notification throttling (30-minute cooldown) if sending notifications
- Verify — Confirm the trigger condition and action mapping with the user before running
Important Notes
- Device name matching uses fuzzy matching; when multiple results are found, ask the user to confirm
- The statistics API time format is
, and the time range cannot exceed 24 hours per requestyyyyMMddHH - All four notification APIs are self-send only — messages can only be sent to the currently logged-in user
- The weather query requires latitude and longitude; if unavailable from the Home API, ask for the user's city
- Base URL is auto-detected from API key prefix. See
for detailsreferences/api-conventions.md - If you encounter issues, visit https://github.com/tuya/tuya-openclaw-skills for announcements and troubleshooting
- Never log or display the
value in outputTUYA_API_KEY - CLI exits with code
for usage/validation errors, and2
for runtime/API/network errors1
Supported and Unsupported Operations
Supported Property Types for Control
Only basic data type properties are currently supported for device control:
| Type | Description | Example |
|---|---|---|
| bool | Boolean on/off | Turn light on/off, turn AC on/off, turn plug on/off |
| enum | Enumeration selection | Switch AC mode (auto/cold/hot), set fan speed (low/mid/high) |
| value (Integer) | Numeric value | Adjust brightness (0-1000), set temperature (16-30) |
| string | String value | Set device display text |
Unsupported Operations
The following operations involve sensitive actions or complex data types and are NOT supported:
- Lock control — Unlock doors, lock/unlock smart locks (security-sensitive)
- Live video streaming — Pull real-time video streams or view camera live footage (cloud snapshot/short video capture IS supported — see Workflow 8)
- Image operations — Retrieve or push images from/to devices
- Complex data type control — Properties with
,raw
,bitmap
, orstruct
typeSpec are not supported for issuing commandsarray - Firmware upgrades — OTA firmware update operations
- Device pairing/removal — Adding new devices or removing existing devices
If the user requests any of these unsupported operations, clearly inform them that the operation is not available through this skill and suggest using the Tuya App directly.
Data Egress Statement
This skill sends data to the Tuya Open Platform:
| Data Type | Sent To | Purpose | Required |
|---|---|---|---|
| Api-key | User-configured base_url | API authentication | Required |
| Device ID | User-configured base_url | Device query and control | Required |
| Control commands | User-configured base_url | Device property issuance | Required |
| Api-key | Auto-detected WebSocket URI | Real-time event subscription authentication | Required for message subscription |