tinytuya


Nametinytuya JSON
Version 1.16.0 PyPI version JSON
download
home_pagehttps://github.com/jasonacox/tinytuya
SummaryPython module to interface with Tuya WiFi smart devices
upload_time2025-01-06 05:46:54
maintainerNone
docs_urlNone
authorJason Cox
requires_pythonNone
licenseNone
keywords
VCS
bugtrack_url
requirements cryptography requests colorama
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # TinyTuya

[![License](https://img.shields.io/github/license/jasonacox/tinytuya)](https://img.shields.io/github/license/jasonacox/tinytuya)
[![PyPI version](https://badge.fury.io/py/tinytuya.svg)](https://badge.fury.io/py/tinytuya)
[![Build Test](https://github.com/jasonacox/tinytuya/actions/workflows/test.yml/badge.svg)](https://github.com/jasonacox/tinytuya/actions/workflows/test.yml)
[![Contrib Test](https://github.com/jasonacox/tinytuya/actions/workflows/contrib.yml/badge.svg)](https://github.com/jasonacox/tinytuya/actions/workflows/contrib.yml)
[![PyPI - Python Version](https://img.shields.io/pypi/pyversions/tinytuya)](https://img.shields.io/pypi/pyversions/tinytuya)
[![PyPI Downloads](https://static.pepy.tech/badge/tinytuya/month)](https://static.pepy.tech/badge/tinytuya/month)

Python module to interface with Tuya WiFi smart devices

## Description

This python module controls and reads state of [Tuya](https://en.tuya.com/) compatible WiFi Smart Devices (Plugs, Switches, Lights, Window Covers, etc.) using the local area network (LAN) or the cloud (TuyaCloud API).  This is a compatible replacement for the `pytuya` PyPI module and currently supports Tuya Protocols 3.1, 3.2, 3.3, 3.4 and 3.5.

[Tuya](https://en.tuya.com/) devices are designed to communicate with the TuyaCloud but most also expose a local area network API.  This allows us to directly control the devices without using the cloud. This python module provides a way to poll status and issue commands to these devices.

TinyTuya can also connect to the Tuya Cloud to poll status and issue commands to Tuya devices.

![TinyTuya Diagram](https://raw.githubusercontent.com/jasonacox/tinytuya/master/docs/TinyTuya-diagram.svg)

```python
# Example Usage of TinyTuya
import tinytuya

d = tinytuya.Device('DEVICE_ID_HERE', 'IP_ADDRESS_HERE', 'LOCAL_KEY_HERE', version=3.3)
data = d.status() 
print('Device status: %r' % data)
```

NOTE: Devices need to be **activated** by Smart Life App.

## TinyTuya Installation  

```bash
# Install TinyTuya Library
python -m pip install tinytuya

# Optional: Install Command Line Tool
pipx install tinytuya
```

Pip will attempt to install `cryptography`, `requests` and `colorama` if not already installed.

## Tuya Device Preparation

Controlling and monitoring Tuya devices on your network requires the following:
* *Address* - Network address (IPv4) of the device e.g. 10.0.1.100
* *Device ID* - Unique identifier for the Tuya device
* *Version* - Tuya protocol version used (3.1, 3.2, 3.3, 3.4 or 3.5)
* *Local_Key* - Security key needed to access the Tuya device. See [Setup Wizard](https://github.com/jasonacox/tinytuya#setup-wizard---getting-local-keys) to get these keys.

### Network Scanner

TinyTuya has a built in network scanner that can be used to find Tuya Devices on your local network. It will show *Address*, *Device ID* and *Version* for each device. Your LAN and firewall will need to allow UDP (6666, 6667 and 7000) and TCP (6668) traffic.

```bash
python -m tinytuya scan
```

### Setup Wizard - Getting Local Keys

TinyTuya has a built-in setup Wizard that uses the Tuya IoT Cloud Platform to generate a JSON list (devices.json) of all your registered devices, including secret *Local_Key* and *Name* of your devices. Follow the steps below:

1. PAIR - Download the *Smart Life App* or *Tuya Smart App*, available for [iPhone](https://itunes.apple.com/us/app/smart-life-smart-living/id1115101477?mt=8) or [Android](https://play.google.com/store/apps/details?id=com.tuya.smartlife&hl=en). Set up your SmartLife account and pair all of your Tuya devices (this is important as you cannot access a device that has not been paired).  Do not use a 'guest' account, otherwise it will get deleted without confirmation at step 3.

2. SCAN (Optional) - Run the TinyTuya scan to get a list of Tuya devices on your network. It will show device *Address*, *Device ID* and *Version* number (3.x):
    ```bash
    python -m tinytuya scan
    ```
    NOTE: You will need to use one of the displayed *Device IDs* for step 4.

3. TUYA ACCOUNT - Set up a Tuya Account (see [PDF Instructions](https://github.com/jasonacox/tinytuya/files/12836816/Tuya.IoT.API.Setup.v2.pdf)):
    * *NOTE: Tuya often changes their portal and services. Please open an [issue](https://github.com/jasonacox/tinytuya/issues) with screenshots if we need to update these instructions.*
    * Create a Tuya Developer account on [iot.tuya.com](https://iot.tuya.com/). When it asks for the "Account Type", select "Skip this step..." (see [screenshot](https://user-images.githubusercontent.com/836718/213877860-34c39851-5671-4c9f-b4d5-251873f18c77.png)).  
    * Click on "Cloud" icon -> "Create Cloud Project"
      1. Pick the correct Data Center "Region" for your location (check [HERE](https://developer.tuya.com/en/docs/iot/oem-app-data-center-distributed?id=Kafi0ku9l07qb#title-3-Western%20America%20Data%20Center) to find your Region).  This will be used by TinyTuya Wizard ([screenshot](https://user-images.githubusercontent.com/836718/138598647-c9657e49-1a89-4ed6-8105-ceee95d9513f.png)).
      2. Skip the configuration wizard but remember the Authorization Key: *API ID* and *Secret* for below ([screenshot](https://user-images.githubusercontent.com/836718/138598788-f74d2fe8-57fa-439c-8003-18735a44e7e5.png)).
    * Click on "Cloud" icon -> Select your project -> **Devices** -> **Link Tuya App Account** ([see screenshot](https://user-images.githubusercontent.com/836718/155827671-44d5fce4-0119-4d0e-a224-ef3715fafc24.png))
    * Click **Add App Account** ([screenshot](https://user-images.githubusercontent.com/836718/273364035-0fd133b7-4e9e-4a6d-900e-efe63d69f1a0.png)) and it will pop-up a "Link Tuya App Account" dialog, chose "Automatic" and "Read Only Status" (it will still alow commands). Click OK and it will display a QR code. Scan the QR code with the *Smart Life app* on your Phone (see step 1 above) by going to the "Me" tab in the *Smart Life app* and clicking on the QR code button `[..]` in the upper right hand corner of the app. When you scan the QR code, it will link all of the devices registered in your *Smart Life app* into your Tuya IoT project. If the QR code will not scan then make sure to disable any browser theming plug-ins (such as Dark Reader) and try again.
    * **NO DEVICES?** If no devices show up after scanning the QR code, you will need to select a different data center and edit your project (or create a new one) until you see your paired devices from the *Smart Life App* show up. ([screenshot](https://user-images.githubusercontent.com/35581194/148679597-391adecb-a271-453b-90c0-c64cdfad42e4.png)). The data center may not be the most logical. As an example, some in the UK have reported needing to select "Central Europe" instead of "Western Europe".
    * **SERVICE API:** Under "Service API" ensure these APIs are listed: `IoT Core` and `Authorization`. To be sure, click subscribe again on every service.  Very important: **disable popup blockers** otherwise subscribing won't work without providing any indication of a failure. Make sure you authorize your Project to use those APIs:
        - Click "Service API" tab
        - Click "**Go to Authorize**" button
        - Select the API Groups from the dropdown and click `Subscribe` ([screenshot](https://user-images.githubusercontent.com/38729644/128742724-9ed42673-7765-4e21-94c8-76022de8937a.png))
    * **RENEWAL:** The subscription to the `IoT Core` service expires after some time. By default, when you subscribe to it for the first time, it will last for one month. Once expired, the setup wizard won't be able to communicate with the Tuya account anymore, so it needs to be renewed. As of November 12th 2024, it can be renewed for a duration of 1, 3 or 6 months by simply filling in a form with some basic information (e.g. purpose of the project, type of developer).

5. WIZARD - Run Setup Wizard:
    * From your Linux/Mac/Win PC run the TinyTuya Setup **Wizard** to fetch the *Local_Keys* for all of your registered devices:
      ```bash
      python -m tinytuya wizard   # use -nocolor for non-ANSI-color terminals
      ```
    * The **Wizard** will prompt you for the *API ID* key, API *Secret*, API *Region* (cn, us, us-e, eu, eu-w, or in) from your Tuya IoT project as set in Step 3 above.
        * To find those again, go to [iot.tuya.com](https://iot.tuya.com/), choose your project and click `Overview`
            * API Key: Access ID/Client ID
            * API Secret: Access Secret/Client Secret
    * It will also ask for a sample *Device ID*.  You can have the wizard scan for one (enter `scan`), use one from step 2 above or in the Device List on your Tuya IoT project.
    * The **Wizard** will poll the Tuya IoT Cloud Platform and print a JSON list of all your registered devices with the "name", "id" and "key" of your registered device(s). The "key"s in this list are the Devices' *Local_Key* you will use to access your device. 
    * In addition to displaying the list of devices, **Wizard** will create a local file `devices.json` that TinyTuya will use to provide additional details for scan results from `tinytuya.deviceScan()` or when running `python -m tinytuya scan`. The wizard also creates a local file `tuya-raw.json` that contains the entire payload from Tuya Cloud.
    * The **Wizard** will ask if you want to poll all the devices. If you do, it will display the status of all devices on record and create a `snapshot.json` file with these results. Make sure your LAN and firewall permit UDP (6666, 6667 and 7000) and TCP (6668) traffic.

Notes:
* If you ever reset or re-pair your smart devices, the *Local_Key* will be reset and you will need to repeat the steps above.
* The TinyTuya *Wizard* was inspired by the TuyAPI CLI which is an alternative way to fetch the *Local_Keys*: `npm i @tuyapi/cli -g` and run `tuya-cli wizard`  

## Programming with TinyTuya

After importing tinytuya, you create a device handle for the device you want to read or control.  Here is an example for a Tuya smart switch or plug:

```python
import tinytuya

# Connect to Device
d = tinytuya.OutletDevice(
    dev_id='DEVICE_ID_HERE',
    address='IP_ADDRESS_HERE',      # Or set to 'Auto' to auto-discover IP address
    local_key='LOCAL_KEY_HERE', 
    version=3.3)

# Get Status
data = d.status() 
print('set_status() result %r' % data)

# Turn On
d.turn_on()

# Turn Off
d.turn_off()
```

### TinyTuya Module Classes and Functions 
```
Classes

  AESCipher - Cryptography Helpers
  XenonDevice(args...) - Base Class
    Device(args...) - Tuya Class for Devices
      OutletDevice(args...)
      CoverDevice(args...)
      BulbDevice(args...)
        Where args:
          dev_id (str): Device ID e.g. 01234567891234567890
          address (str): Device Network IP Address e.g. 10.0.1.99 or "Auto" to auto-find
          local_key (str): The encryption key
          dev_type (str): Device type for payload options (see below)
          version = 3.1 (float): Tuya Protocol (e.g. 3.1, 3.2, 3.3, 3.4, 3.5)
          persist = False (bool): Keep TCP link open
          cid = None (str): Optional sub device id
          node_id = None (str): Alias for cid
          parent = None (object): Gateway device object this is a child of
          port = TCPPORT (int): The port to connect to device
          connection_timeout = 5 (int): Timeout in seconds
          connection_retry_limit = 5 (int)
          connection_retry_delay = 5 (int)

          Total timeout = (connection_timeout * connection_retry_limit) + 
                          (connection_retry_delay * (connection_retry_limit - 1))
                          Defaults: (5 * 5) + (5 * (5 - 1)) = 45 seconds
                        
  Cloud(apiRegion, apiKey, apiSecret, apiDeviceID, new_sign_algorithm)

TinyTuya Base Functions
    devices = deviceScan()                        # Returns dictionary of devices found on local network
    scan()                                        # Interactive scan of local network
    wizard()                                      # Interactive setup wizard
    set_debug(toggle, color)                      # Activate verbose debugging output
    pack_message(msg, hmac_key)                   # Packs a TuyaMessage(), encrypting or adding a CRC if required 
    unpack_message(data, hmac_key, header, 
                    no_retcode)                   # Unpacks a TuyaMessage() 
    parse_header(data)                            # Unpacks just the header part of a message into a TuyaHeader()
    find_device(dev_id, address)                  # Scans network for Tuya devices with either ID = dev_id or IP = address
    device_info(dev_id)                           # Searches DEVICEFILE (usually devices.json) for device with ID
    assign_dp_mappings(tuyadevices, mappings)     # Adds mappings to all the devices in the tuyadevices list
    decrypt_udp(msg)                              # Decrypts a UDP network broadcast packet

 Device Functions (All Devices)
    json = status()                               # returns json payload
    subdev_query(nowait)                          # query sub-device status (only for gateway devices)
    set_version(version)                          # 3.1 [default], 3.2, 3.3 or 3.4
    set_socketPersistent(False/True)              # False [default] or True
    set_socketNODELAY(False/True)                 # False or True [default]
    set_socketRetryLimit(integer)                 # retry count limit [default 5]
    set_socketRetryDelay(integer)                 # retry delay [default 5]
    set_socketTimeout(timeout)                    # set connection timeout in seconds [default 5]
    set_dpsUsed(dps_to_request)                   # add data points (DPS) to request
    add_dps_to_request(index)                     # add data point (DPS) index set to None
    set_retry(retry=True)                         # retry if response payload is truncated
    set_status(on, switch=1, nowait)              # Set status of switch to 'on' or 'off' (bool)
    set_value(index, value, nowait)               # Set int value of any index.
    set_multiple_values(index_value_dict, nowait) # Set multiple values with a single request
    heartbeat(nowait)                             # Send heartbeat to device
    updatedps(index=[1], nowait)                  # Send updatedps command to device
    turn_on(switch=1, nowait)                     # Turn on device / switch #
    turn_off(switch=1, nowait)                    # Turn off
    set_timer(num_secs, nowait)                   # Set timer for num_secs
    set_sendWait(num_secs)                        # Time to wait after sending commands before pulling response
    detect_available_dps()                        # Return list of DPS available from device
    generate_payload(command, data,...            # Generate TuyaMessage payload for command with data
    send(payload)                                 # Send payload to device (do not wait for response)
    receive()                                     # Receive payload from device

OutletDevice Additional Functions
    set_dimmer(percentage):

BulbDevice Additional Functions
    set_colour(r, g, b, nowait):
    set_hsv(h, s, v, nowait):
    set_white(brightness, colourtemp, nowait):
    set_white_percentage(brightness=100, colourtemp=0, nowait):
    set_brightness(brightness, nowait):
    set_brightness_percentage(brightness=100, nowait):
    set_colourtemp(colourtemp, nowait):
    set_colourtemp_percentage(colourtemp=100, nowait):
    set_scene(scene, nowait):                     # 1=nature, 3=rave, 4=rainbow
    set_mode(mode='white', nowait):               # white, colour, scene, music
    result = brightness():
    result = colourtemp():
    (r, g, b) = colour_rgb():
    (h,s,v) = colour_hsv()
    result = state():

CoverDevice Additional Functions
    open_cover(switch=1):
    close_cover(switch=1):
    stop_cover(switch=1):

Cloud Functions
    setregion(apiRegion)
    cloudrequest(url, action=[POST if post else GET], post={}, query={})
    getdevices(verbose=False)
    getstatus(deviceid)
    getfunctions(deviceid)
    getproperties(deviceid)
    getdps(deviceid)
    sendcommand(deviceid, commands [, uri])
    getconnectstatus(deviceid)
    getdevicelog(deviceid, start=[now - 1 day], end=[now], evtype="1,2,3,4,5,6,7,8,9,10", size=100, params={})
      -> when start or end are negative, they are the number of days before "right now"
          i.e. "start=-1" is 1 day ago, "start=-7" is 7 days ago
          
```

### TinyTuya Error Codes

Starting with v1.2.0 TinyTuya functions will return error details in the JSON data responses instead of raising exceptions.  The format for this response:

```json
{ "Error":"Invalid JSON Payload", "Err":"900", "Payload":"{Tuya Message}" }
```

The "Err" number will be one of these:

* 900 (ERR_JSON) - Invalid JSON Response from Device
* 901 (ERR_CONNECT) - Network Error: Unable to Connect
* 902 (ERR_TIMEOUT) - Timeout Waiting for Device
* 903 (ERR_RANGE) - Specified Value Out of Range
* 904 (ERR_PAYLOAD) - Unexpected Payload from Device
* 905 (ERR_OFFLINE) - Network Error: Device Unreachable
* 906 (ERR_STATE) - Device in Unknown State
* 907 (ERR_FUNCTION) - Function Not Supported by Device
* 908 (ERR_DEVTYPE) - Device22 Detected: Retry Command
* 909 (ERR_CLOUDKEY) - Missing Tuya Cloud Key and Secret
* 910 (ERR_CLOUDRESP) - Invalid JSON Response from Cloud
* 911 (ERR_CLOUDTOKEN) - Unable to Get Cloud Token
* 912 (ERR_PARAMS) - Missing Function Parameters
* 913 (ERR_CLOUD) - Error Response from Tuya Cloud
* 914 (ERR_KEY_OR_VER) - Check device key or version

### Example Usage

See the sample python script [test.py](test.py) for an OutletDevice example or look in the [examples](examples) directory for other scripts.

```python
import tinytuya

"""
OUTLET Device
"""
d = tinytuya.Device('DEVICE_ID_HERE', 'IP_ADDRESS_HERE', 'LOCAL_KEY_HERE', version=3.3)
data = d.status()  

# Show status and state of first controlled switch on device
print('Dictionary %r' % data)
print('State (bool, true is ON) %r' % data['dps']['1'])  

# Toggle switch state
switch_state = data['dps']['1']
data = d.set_status(not switch_state)  # This requires a valid key
if data:
    print('set_status() result %r' % data)

# On a switch that has 4 controllable ports, turn the fourth OFF (1 is the first)
data = d.set_status(False, 4)
if data:
    print('set_status() result %r' % data)
    print('set_status() extra %r' % data[20:-8])

"""
RGB Bulb Device
"""
import time

d = tinytuya.BulbDevice('DEVICE_ID_HERE', 'IP_ADDRESS_HERE', 'LOCAL_KEY_HERE')
d.set_version(3.3)  # IMPORTANT to set this regardless of version
d.set_socketPersistent(True)  # Optional: Keep socket open for multiple commands
data = d.status()

# Show status of first controlled switch on device
print('Dictionary %r' % data)

# Set to RED Color - set_colour(r, g, b):
d.set_colour(255,0,0)  

# Cycle through the Rainbow
rainbow = {"red": [255, 0, 0], "orange": [255, 127, 0], "yellow": [255, 200, 0],
          "green": [0, 255, 0], "blue": [0, 0, 255], "indigo": [46, 43, 95],
          "violet": [139, 0, 255]}
for color in rainbow:
    [r, g, b] = rainbow[color]
    d.set_colour(r, g, b, nowait=True)  # nowait = Go fast don't wait for response
    time.sleep(0.25)

# Brightness: Type A devices range = 25-255 and Type B = 10-1000
d.set_brightness(1000)

# Set to White - set_white(brightness, colourtemp):
#    colourtemp: Type A devices range = 0-255 and Type B = 0-1000
d.set_white(1000,10)

# Set Bulb to Scene Mode
d.set_mode('scene')

# Scene Example: Set Color Rotation Scene
d.set_value(25, '07464602000003e803e800000000464602007803e803e80000000046460200f003e803e800000000464602003d03e803e80000000046460200ae03e803e800000000464602011303e803e800000000')

```
### Example Device Monitor

You can set up a persistent connection to a device and then monitor the state changes with a continual loop. This is helpful for troubleshooting and discovering DPS values.

```python
import tinytuya

d = tinytuya.OutletDevice('DEVICEID', 'DEVICEIP', 'DEVICEKEY', version=3.3, persist=True)

print(" > Send Request for Status < ")
d.status(nowait=True)

print(" > Begin Monitor Loop <")
while(True):
    # See if any data is available
    data = d.receive()
    print('Received Payload: %r' % data)

    # Send keep-alive heartbeat
    if not data:
        print(" > Send Heartbeat Ping < ")
    	d.heartbeat()

    # NOTE If you are not seeing updates, you can force them - uncomment:
    # print(" > Send Request for Status < ")
    # d.status(nowait=True)

    # NOTE Some smart plugs require an UPDATEDPS command to update power data
    # print(" > Send DPS Update Request < ")
    # payload = d.generate_payload(tinytuya.UPDATEDPS)
    # d.send(payload)    

```

### Tuya Cloud Access

You can poll and manage Tuya devices using the `Cloud` class and functions.

_CAUTION: The free Tuya IoT Developer (Trial) account allows a very limited number of Cloud API calls. Be aware of the restrictions before enabling any automation that makes frequent calls._

```python
import tinytuya

# Connect to Tuya Cloud
# c = tinytuya.Cloud()  # uses tinytuya.json 
c = tinytuya.Cloud(
        apiRegion="us", 
        apiKey="xxxxxxxxxxxxxxxxxxxx", 
        apiSecret="xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx", 
        apiDeviceID="xxxxxxxxxxxxxxxxxxID")

# Display list of devices
devices = c.getdevices()
print("Device List: %r" % devices)

# Select a Device ID to Test
id = "xxxxxxxxxxxxxxxxxxID"

# Display Properties of Device
result = c.getproperties(id)
print("Properties of device:\n", result)

# Display Status of Device
result = c.getstatus(id)
print("Status of device:\n", result)

# Send Command - Turn on switch
commands = {
    "commands": [
        {"code": "switch_1", "value": True},
        {"code": "countdown_1", "value": 0},
    ]
}
print("Sending command...")
result = c.sendcommand(id,commands)
print("Results\n:", result)
```

Up to one week of device logs can also be pulled from the Cloud.  By default getdevicelog() will pull 1 day of logs or 5000 log entries, whichever comes first.  The returned timestamps are unixtime*1000, and event_id 7 (data report) will probably be the most useful.

```python
import tinytuya
import json

c = tinytuya.Cloud()
#r = c.getdevicelog( '00112233445566778899', start=-1, end=0, size=0, max_fetches=50 )
#r = c.getdevicelog( '00112233445566778899', start=1669990000, end=1669990300, size=20 )
r = c.getdevicelog( '00112233445566778899' )
print( json.dumps(r, indent=2) )
```

### Encryption Notes

Tuya devices use AES encryption which is not available in the Python standard library. **PyCA/cryptography** is recommended and installed by default. Other options include **PyCryptodome** , **PyCrypto** and **pyaes**.

* Deprecation notice for pyaes: The pyaes library works for Tuya Protocol <= 3.4 but will not work for 3.5 devices. This is because pyaes does not support GCM which is required for v3.5 devices.

### Command Line

TinyTuya provides a built-in command line interface to get Local key, scan and poll devices.

Installation

```bash
# Option-1: pip install tinytuya
python -m tinytuya

# Option-2: pipx install tinytuya
tinytuya 
```

Command Line Usage

```
tinytuya <command> [-debug] [-nocolor] [-h] [-yes] [-no-poll] [-device-file FILE] [-snapshot-file FILE]

  wizard         Launch Setup Wizard to get Tuya Local KEYs.
  scan           Scan local network for Tuya devices.
  devices        Scan all devices listed in devices.json file.
  snapshot       Scan devices listed in snapshot.json file.
  json           Scan devices listed in snapshot.json file [JSON].

  Wizard
      tinytuya wizard [-h] [-debug] [-force [0.0.0.0/24 ...]] [-no-broadcasts] [-nocolor] [-yes] [-no-poll]
                [-device-file FILE] [-raw-response-file FILE] [-snapshot-file FILE] [-credentials-file FILE]
                [-key KEY] [-secret SECRET] [-region {cn,eu,eu-w,in,us,us-e}] [-device DEVICE [DEVICE ...]]
                [-dry-run] [max_time]

        Common Options
        max_time             Maximum time to find Tuya devices [Default: 18]
        -no-broadcasts       Ignore broadcast packets when force scanning
        -nocolor             Disable color text output.
        -debug               Activate debug mode.
        -h, -help            Show usage help for command.
        -yes, -y             Answer "yes" to all questions
        -no-poll, -no        Answer "no" to "Poll?" (overrides -yes)
        -device-file FILE    JSON file to load devices from [Default: devices.json]
        -snapshot-file FILE  JSON file to load/save snapshot from/to [Default: snapshot.json]
        -force [0.0.0.0/24 ...], -f [0.0.0.0/24 ...]
                             Force network scan of device IP addresses [Default: Auto-detects net/mask]
        -no-broadcasts       Ignore broadcast packets when force scanning
        -raw-response-file   JSON file to save the raw server response to [Default: tuya-raw.json]

        Wizard Cloud API Options
        -dry-run             Do not actually connect to the Cloud
        -credentials-file    JSON file to load/save Cloud credentials from/to [Default: tinytuya.json]
        -key KEY             Cloud API Key to use
        -secret SECRET       Cloud API Secret to use
        -region              Cloud API Region to use {cn,eu,eu-w,in,us,us-e}
        -device DEVICE(S)    One or more Device ID(s) to use

  Scan
      tinytuya scan [-h] [-debug] [-force [0.0.0.0/24 ...]] [-no-broadcasts] [-nocolor] [-yes] 
                [-device-file FILE] [-snapshot-file FILE] [max_time]

  Devices
      tinytuya devices [-h] [-debug] [-force [0.0.0.0/24 ...]] [-no-broadcasts] [-nocolor] [-yes] 
                [-no-poll] [-device-file FILE] [-snapshot-file FILE] [max_time]

  Snapshot
      tinytuya snapshot [-h] [-debug] [-nocolor] [-yes] [-no-poll] [-device-file FILE] [-snapshot-file FILE]

  JSON
      tinytuya json [-h] [-debug] [-device-file FILE] [-snapshot-file FILE]

```

### Scan Tool 
The function `tinytuya.scan()` will listen to your local network (UDP 6666 and 6667) and identify Tuya devices broadcasting their Address, Device ID, Product ID and Version and will print that and their stats to stdout.  This can help you get a list of compatible devices on your network. The `tinytuya.deviceScan()` function returns all found devices and their stats (via dictionary result).

You can run the scanner from the command line using these interactive commands:
  ```bash
  # Listen for Tuya Devices and match to devices.json if available
  python -m tinytuya scan

  # The above creates a snapshot.json file with IP addresses for devices
  # You can use this command to get a rapid poll of status of all devices
  python -m tinytuya snapshot

  # The sames thing as above but with a non-interactive JSON response
  python -m tinytuya json

  # List all register devices discovered from Wizard and poll them
  python -m tinytuya devices

  ```

By default, the scan functions will retry 15 times to find new devices. If you are not seeing all your devices, you can increase max_retries by passing an optional arguments (eg. 50 retries):

  ```bash
  # command line
  python -m tinytuya scan 50
  ```

  ```python
  # invoke verbose interactive scan
  tinytuya.scan(50)

  # return payload of devices
  devices = tinytuya.deviceScan(false, 50)
  ```

## Troubleshooting

* Tuya devices only allow one TCP connection at a time.  Make sure you close the TuyaSmart or SmartLife app before using *TinyTuya* to connect.
* Some devices ship with older firmware that may not work with *TinyTuya*. If you're experiencing issues, please try updating the device's firmware in the official app.
* The LOCAL KEY for Tuya devices will change every time a device is removed and re-added to the TuyaSmart app. If you're getting decrypt errors, try getting the key again as it might have changed. 
* Devices running protocol version 3.1 (e.g. below Firmware 1.0.5) do not require a device *Local_Key* to read the status. All devices will require a device *Local_Key* to control the device.
* Some devices with 22 character IDs will require additional setting to poll correctly. TinyTuya will attempt to detect and accomodate for this, but it can be specified directly:
  ```python
  a = tinytuya.OutletDevice('here_is_my_key', '192.168.x.x', 'secret_key_here', 'device22')
  a.set_version(3.3)
  a.set_dpsUsed({"1": None})  # This needs to be a datapoint available on the device
  data =  a.status()
  print(data)
  ```
* Windows 10 Users - TinyTuya `wizard` and `scan` interactive tools use ANSI color. This will work correctly in PowerShell but will show cryptic escape codes when run in Windows `CMD`.  You can fix this by using the `-nocolor` option on tinytuya, or by changing the Windows `CMD` console registry to process ANSI escape codes by doing something like this:
  ```
  reg add HKEY_CURRENT_USER\Console /v VirtualTerminalLevel /t REG_DWORD /d 0x00000001 /f
  ```

## User Contributed Device Modules

In addition to the built-in `OutletDevice`, `BulbDevice` and `CoverDevice` device support, the community is encourage to submit additional device modules which are available here: [Contrib Library](https://github.com/jasonacox/tinytuya/tree/master/tinytuya/Contrib):

```python
# Example usage of community contributed device modules
from tinytuya.Contrib import ThermostatDevice

thermo = ThermostatDevice( 'abcdefghijklmnop123456', '172.28.321.475', '1234567890123abc' )
```

## Tuya Data Points - DPS Table

The Tuya devices send back data points (DPS) also called device function points, in a json string.  The DPS attributes define the state of the device.  Each key in the DPS dictionary refers to key value pair, the key is the DP ID and its value is the dpValue. You can refer to the [Tuya developer platform](https://iot.tuya.com/index/) for definition of function points for the products.

The following table represents several of the standard Tuya DPS values and their properties. It represents data compiled from Tuya documentation and self-discovery. Devices may vary. Feedback or additional data would would be appreciated.  Please submit a Issue or Pull Request if you have additional data that would be helpful for others.

To find Tuya DPS for devices not listed below, you can discover the DPS values using the Tuya IoT platform. See this help here: [Find your Data Point](https://www.zigbee2mqtt.io/advanced/support-new-devices/03_find_tuya_data_points.html#_8-display-device-logs).

DPS Read and Set Example:
```python
# Read Value of DPS 25
data = d.status()  
print("Value of DPS 25 is ", data['dps']['25'])

# Set Value of DPS 25
d.set_value(25, '010e0d0000000000000003e803e8')
```

### Version 3.1 Devices

#### Version 3.1 (and some 3.3) - Plug or Switch Type
| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
|1|Switch|bool|True/False||
|2|Countdown?|integer|0-86400|s|
|4|Current|integer|0-30000|mA|
|5|Power|integer|0-50000|W|
|6|Voltage|integer|0-5000|V|

#### Version 3.1 - Light Type (RGB)
| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
| 1|Switch|bool|True/False||
| 2|Mode|enum|white,colour,scene,music||
| 3|Bright|integer|10-1000*||
| 4|Color Temp|integer|0-1000*||
| 5|Color|hexstring|r:0-255,g:0-255,b:0-255,h:0-360,s:0-255,v:0-255|rgb+hsv|

### Version 3.3 Devices

#### Version 3.3 - Plug, Switch, Power Strip Type
| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
|1|Switch 1|bool|True/False||
|2|Switch 2|bool|True/False||
|3|Switch 3|bool|True/False||
|4|Switch 4|bool|True/False||
|5|Switch 5|bool|True/False||
|6|Switch 6|bool|True/False||
|7|Switch 7/usb|bool|True/False||
|9|Countdown 1|integer|0-86400|s|
|10|Countdown 2|integer|0-86400|s|
|11|Countdown 3|integer|0-86400|s|
|12|Countdown 4|integer|0-86400|s|
|13|Countdown 5|integer|0-86400|s|
|14|Countdown 6|integer|0-86400|s|
|15|Countdown 7|integer|0-86400|s|
|17|Add Electricity|integer|0-50000|kwh|
|18|Current|integer|0-30000|mA|
|19|Power|integer|0-50000|W|
|20|Voltage|integer|0-5000|V|
|21|Test Bit|integer|0-5|n/a|
|22|Voltage coeff.|integer|0-1000000||
|23|Current coeff.|integer|0-1000000||
|24|Power coeff.|integer|0-1000000||
|25|Electricity coeff.|integer|0-1000000||
|26|Fault|fault|ov_cr||
|38|Power-on state setting|enum|off, on, memory||
|39|Overcharge Switch|bool|True/False||
|40|Indicator status setting|enum|none, on, relay, pos||
|41|Child Lock|bool|True/False||
|42|UNKNOWN||||
|43|UNKNOWN||||
|44|UNKNOWN||||

Note: Some 3.3 energy management plugs use the DPS values of the 3.1 plug above.

#### Version 3.3 - Dimmer Switch
| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
| 1|Switch|bool|True/False||
| 2|Brightness|integer|10-1000*||
| 3|Minimum of Brightness|integer|10-1000*||
| 4|Type of light source1|enum|LED,incandescent,halogen||
| 5|Mode|enum|white||

#### Version 3.3 - Light Type (RGB)
| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
| 20|Switch|bool|True/False||
| 21|Mode|enum|white,colour,scene,music||
| 22|Bright|integer|10-1000*||
| 23|Color Temp|integer|0-1000||
| 24|Color|hexstring|h:0-360,s:0-1000,v:0-1000|hsv|
| 25|Scene|string|n/a||
| 26|Left time|integer|0-86400|s|
| 27|Music|string|n/a||
| 28|Debugger|string|n/a||
| 29|Debug|string|n/a||
| 30|Rhythms|n/a|n/a||
| 31|Go To Sleep|n/a|n/a||
| 32|Wake Up|n/a|n/a||
| 33|Power Off Memory|n/a|n/a||
| 34|Do not Disturb|n/a|n/a||
| 41|Remote Control Switch|n/a|n/a||
| 209|Cycle Timing|n/a|n/a||
| 210|Vaction Timing|n/a|n/a||

#### Version 3.3 - Automated Curtain Type
| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
|1|Curtain Switch 1|enum|open, stop, close, continue||
|2|Percent control 1|integer|0-100|%|
|3|Accurate Calibration 1|enum|start, end||
|4|Curtain Switch 2|enum|open, stop, close, continue||
|5|Percent control 2|integer|0-100||
|6|Accurate Calibration 2|enum|start, end||
|8|Motor Steer 1|enum|forward, back||
|9|Motor steer 2|enum|forward, back||
|10|Quick Calibration 1|integer|1-180|s|
|11|Quick Calibration 2|integer|1-180|s|
|12|Motor Mode 1|enum|strong_power, dry_contact||
|13|Motor Mode 2|enum|strong_power, dry_contact||
|14|Light mode|enum|relay, pos, none||

#### Version 3.3 - Fan Switch Type
| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
|1|Fan switch|bool|True/False|n/a|
|2|Fan countdown|integer|0-86400|s|
|3|Fan speed|enum|level_1, level_2, level_3, level_4, level_5||
|4|Fan speed|integer|1-100|%|
|5|Fan light switch|bool|True/False||
|6|Brightness integer|integer|10-1000||
|7|Fan light countdown|integer|0-86400||
|8|Minimum brightness|integer|10-1000||
|9|Maximum brightness|integer|10-1000||
|10|Mode|enum|white||
|11|Power-on state setting|enum|off, on, memory||
|12|Indicator status setting|enum|none, relay, pos||
|13|Backlight switch|bool|True/False||

#### Version 3.3 - Universal IR Controller with optional Temp/Humidity
| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
|101|Current Temperature|integer|0-600|10x Celsius|
|102|Current Humidity|integer|0-100|%|
|201|IR Commands (set only)|JSON*|n/a|n/a|

  ```python
  # The IR Commands JSON has the following format:
  command = {
      "control": "send_ir",
      "head": "",
      "key1": "[[TO_BE_REPLACED]]",
      "type": 0,
      "delay": 300,
  }
  # Sending the IR command:
  payload = d.generate_payload(tinytuya.CONTROL, {"201": json.dumps(command)})
  d.send(payload)
  ```

The `key1` attribute is a base64 string that contains the IR signal. You can extract it using this procedure:

1. Register a new IR device on Tuya Smart / Smart Life app (if not registered already) and map, setup or import your buttons.
2. Tap multiple times on the button you wish to control.
3. Go to [Tuya IoT Platform](https://iot.tuya.com/) and select your app under Cloud > Development section.
4. Go to to the Device tab and select "Debug Device" on the parent device. Browse Device Logs section and retrieve the `key1` attribute that matches your tapping timestamp from step 2 above. Use that `key1` attribute in the payload example above.

You need to repeat these steps for each button (cloud logging is not always sequential).

#### Version 3.3 - Sensor Type

_Important Note:_
Battery-powered Tuya sensors are usually designed to stay in sleep mode until a state change (eg.open or close alert). This means you will not be able to poll these devices except in the brief moment they awake, connect to the WiFi and send their state update payload the the Tuya Cloud. Keep in mind that if you manage to poll the device enough to keep it awake, you will likely quickly drain the battery.

| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
|1|Door Sensor|bool|True/False||
|2|Battery level state|enum|low, middle, high||
|3|Battery level|integer|0-100|%|
|4|Temper alarm|bool|True/False||
|5|Flooding Detection State|enum|alarm, normal||
|6|Luminance detection state|enum|low, middle, high, strong||
|7|Current Luminance|integer|0-100|%|
|8|Current Temperature|integer|400-2000||
|9|Current Humidity|integer|0-100|%|
|10|Shake State|enum|normal, vibration, drop, tilt||
|11|Pressure State|enum|alarm, normal||
|12|PIR state|enum|pir, none||
|13|Smoke Detection State|enum|alarm, normal||
|14|Smoke value|integer|0-1000||
|15|Alarm Volume|enum|low, middle, high, mute||
|16|Alarm Ringtone|enum|1, 2, 3, 4, 5||
|17|Alarm Time|integer|0-60|s|
|18|Auto-Detect|bool|True/False||
|19|Auto-Detect Result|enum|checking, check_success, check_failure, others||
|20|Preheat|bool|True/False||
|21|Fault Alarm|fault|fault, serious_fault, sensor_fault, probe_fault, power_fault|Barrier|
|22|Lifecycle|bool|True/False||
|23|Alarm Switch|bool|True/False||
|24|Silence|bool|True/False||
|25|Gas Detection State|enum|alarm, normal||
|26|Detected Gas|integer|0-1000||
|27|CH4 Detection State|enum|alarm, normal||
|28|CH4 value|integer|0-1000||
|29|Alarm state|enum|alarm_sound, alarm_light, alarm_sound_light, normal||
|30|VOC Detection State|enum|alarm, normal||
|31|VOC value|integer|0-999||
|32|PM2.5 state|enum|alarm, normal||
|33|PM2.5 value|integer|0-999||
|34|CO state|enum|alarm, normal||
|35|CO value|integer|0-1000||
|36|CO2 Detection State|enum|alarm, normal||
|37|CO2 value|integer|0-1000||
|38|Formaldehyde Detection State|enum|alarm, normal||
|39|CH2O value|integer|0-1000||
|40|Master mode|enum|disarmed, arm, home, sos||
|41|Air quality index|enum|level_1, level_2, level_3, level_4, level_5, level_6||

NOTE (*) - The range can vary depending on the device. As an example, for dimmers, it may be 10-1000 or 25-255.

#### Version 3.3 - WiFi Air Quality Detector PM2.5/Formaldehyde/VOC/CO2/Temperature/Humidity

| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
| 2 | PM2.5 value | integer | 0 - 999 | ug/m3 |
| 18 | Current Temperature | integer | 0 - 850 | ˚C (multiplied by 10) |
| 19 | Current Humidity | integer | 0 - 1000 | % (multiplied by 10) |
| 20 | CH2O (Formaldehyde) value | integer | 0 - 1000 | ppm |
| 21 | VOC (Volatile organic compound) value | integer | 0 - 2000 | ppm |
| 22 | CO2 value | integer | 350 - 2000 | ppm |

Example device: https://www.aliexpress.com/item/1005005034880204.html

#### Version 3.3 - Robot Mower Type

| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
| 6    | Battery | integer |  0-100 | % |
| 101  | Machine Status | enum | <ul><li>STANDBY MOWING</li><li>CHARGING</li><li>EMERGENCY</li><li>LOCKED</li><li>PAUSED</li><li>PARK</li><li>CHARGING_WITH_TASK_SUSPEND</li><li>FIXED_MOWING</li></ul> ||
| 102  | Machine error | integer | 0, ? ||
| 103  | Machine warning | enum | <ul><li>MOWER_LEAN</li><li>MOWER_EMERGENCY</li><li>MOWER_UI_LOCKED</li><ul> ||
| 104  | Rain mode | boolean | True/False ||
| 105  | Work time | interger | 1-99 | hours |
| 106  | Machine password | byte str | ? ||
| 107  | Clear machine appointment | boolean | True/False ||
| 108  | Query machine reservation | boolean | True/False ||
| 109  | Query partition parameters | boolean | True/False ||
| 110  | Report machine reservation | byte str |||
| 111  | Error log | byte str |||
| 112  | Work log | byte str |||
| 113  | Partition parameters | byte str |||
| 114  | Work mode | enum | AutoMode/?? ||                                                                                                                           | 115  | Machine control CMD | enum | <ul><li>StartMowing</li><li>StartFixedMowing</li><li>PauseWork</li><li>CancelWork</li><li>StartReturnStation</li><ul> ||

Reference [pymoebot](https://github.com/Whytey/pymoebot) for further definition.

#### Version 3.3 - 24v Thermostat (i.e. PCT513-TY)
| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
|2|System Mode|enum|[ 'auto' 'cool' 'heat' 'off' others? ]||
|16|Center of Setpoint, High-Resolution °C|integer|500-3200|°C x 100 in steps of 50|
|17|Center of Setpoint, °F|integer|20-102|°F|
|18*|Cooling Setpoint, Low-Resolution °F|integer|20-102|°F|
|19*|Cooling Setpoint, Low-Resolution °C|integer|500-3200|°C|
|20*|Heating Setpoint, Low-Resolution °F|integer|20-102|°F|
|23|Display Units|enum|[ 'f' 'c' ]||
|24|Current Temperature, High-Resolution °C|integer|500-3200|°C x 100 in steps of 50|
|26*|Heating Setpoint, Low-Resolution °C|integer|5-32|°C|
|27*|Temperature Correction|integer|-10 - +10||
|29|Current Temperature, °F|integer|20-102|°F|
|34|Current Humidity|integer|0-100|%|
|45|Fault Flags|bitmask|[ e1 e2 e3 ]||
|107|System Type|integer-as-string|?|?|
|108*|Cooling Setpoint, High-Resolution °C|integer|500-3200|°C x 100 in steps of 50|
|109*|Heating Setpoint, High-Resolution °C|integer|500-3200|°C x 100 in steps of 50|
|110*|Cooling Setpoint, °F|integer|20-102|°F|
|111*|Heating Setpoint, °F|integer|20-102|°F|
|115|Fan Mode|enum|[ 'auto' 'cycle' 'on' ]||
|116|"at home/away from home"|integer-as-string|?|?|
|118|Schedule Data|base64|binary blob||
|119|Schedule Enabled|bool|True/False||
|120|Hold/Schedule|enum|[ 'permhold' 'temphold' 'followschedule' ]||
|121|Vacation Data|base64|binary blob||
|122|[Sensor Data, list 1](https://github.com/jasonacox/tinytuya/discussions/139)|base64|binary blob||
|123|Minimum Fan Run Time|integer|0-55|minutes per hour|
|125|[Sensor Data, list 2](https://github.com/jasonacox/tinytuya/discussions/139)|base64|binary blob||
|126|[Sensor Data, list 3](https://github.com/jasonacox/tinytuya/discussions/139)|base64|binary blob||
|127|[Sensor Data, list 4](https://github.com/jasonacox/tinytuya/discussions/139)|base64|binary blob||
|128|[Sensor Data, list 5](https://github.com/jasonacox/tinytuya/discussions/139)|base64|binary blob||
|129|System State|enum|[ 'fanon' 'coolfanon' 'alloff' others? ]||
|130|Weather Forcast|?|?|?|

NOTE (*) - Depending on the firmware, either 18/19/20/26/27 or 108/109/110/111/x are used, not both

A user contributed module is available for this device in the [Contrib library](https://github.com/jasonacox/tinytuya/tree/master/tinytuya/Contrib):

```python
from tinytuya.Contrib import ThermostatDevice

thermo = ThermostatDevice( 'abcdefghijklmnop123456', '172.28.321.475', '1234567890123abc' )
```

For info on the Sensor Data lists, see https://github.com/jasonacox/tinytuya/discussions/139

#### Version 3.4 - WIFI Dual Meter (i.e. PJ1103A)
| DP ID        | Function Point | Type        | Range       | Units |
| ------------- | ------------- | ------------- | ------------- |------------- |
|1|Forward energy total|integer||kWh x 100|
|2|Reverse energy total|integer||kWh x 100|
|101|Power (A)|integer||W x 10|
|102|Direction of current flow (A)| enum | <ul><li>FORWARD</li><li>REVERSE</li></ul> ||
|104|Direction of current flow (B)| enum | <ul><li>FORWARD</li><li>REVERSE</li></ul> ||
|105|Power (B)|integer||W x 10|
|106|Energy forward (A)|integer||kWh x 100|
|107|Energy reverse (A)|integer||kWh x 100|
|108|Energy forward (B)|integer||kWh x 100|
|109|Energy reverse (B)|integer||kWh x 100|
|110|Power Factor (A)|integer||value x 100|
|111|AC frequency|integer||Hz x 100|
|112|AC voltage|integer||V x 10|
|113|Current (A)|integer||mA|
|114|Current (B)|integer||mA|
|115|Total power|integer||W x 10|
|116|Voltage Calibration|integer|800-1200|value * 1000|
|117|Current Calibration (A)|integer|800-1200|value * 1000|
|118|Power Calibration (A)|integer|800-1200|value * 1000|
|119|Energy Calibration forward (A)|integer|800-1200|value * 1000|
|121|Power Factor (B)|integer||value x 100|
|122|Frequency Calibration|integer|800-1200|value * 1000|
|123|Current Calibration (B)|integer|800-1200|value * 1000|
|124|Power Calibration (B)|integer|800-1200|value * 1000|
|125|Energy Calibration forward (B)|integer|800-1200|value * 1000|
|127|Energy Calibration reverse (A)|integer|800-1200|value * 1000|
|128|Energy Calibration reverse (B)|integer|800-1200|value * 1000|
|129|Report Rate Control|integer|3-60|s|

Note: (A) or (B) means channel A or channel B

A user contributed module is available for this device in the [Contrib library](https://github.com/jasonacox/tinytuya/tree/master/tinytuya/Contrib):

```python                                                                                                                                                                                                         from tinytuya.Contrib import WiFiDualMeterDevice

wdm = WiFiDualMeterDevice.WiFiDualMeterDevice(
    dev_id='abcdefghijklmnop123456',
    address='192.168.0.29',      # Or set to 'Auto' to auto-discover IP address
    local_key='1234567890123abc',
    version=3.4)
```

### Tuya References

* Tuya Hardware Development - Protocol: https://developer.tuya.com/en/docs/iot/device-development/embedded-software-development/mcu-development-access/wifi-mcu-sdk-solution/tuya-cloud-universal-serial-port-access-protocol?id=K9hhi0xxtn9cb
* TuyaMCU listing of Tuya DP IDs: https://tasmota.github.io/docs/TuyaMCU/#switches-or-plugspower-strips

## Credits

  * TuyAPI https://github.com/codetheweb/tuyapi by codetheweb and blackrozes.
    Protocol reverse engineering from jepsonrob and clach04.
  * PyTuya https://github.com/clach04/python-tuya by clach04. 
    The origin of this python module (now abandoned). Thanks to nijave for pycryptodome support and testing, Exilit for unittests and docstrings, mike-gracia for improved Python version support, samuscherer for RGB Bulb support, magneticflux for improved Python version support, sean6541 for initial PyPi package and Home Assistant support <https://github.com/sean6541/tuya-homeassistant>, ziirish - for resolving a dependency problem related to version numbers at install time
  * https://github.com/rospogrigio/localtuya-homeassistant by rospogrigio. 
    Updated pytuya to support devices with Device IDs of 22 characters
  * Thanks to [@uzlonewolf](https://github.com/uzlonewolf), our top contributor and resident wizard, for expanding the Outlet/Cover/Bulb/Cloud modules into separate files, introducing Contrib structure for user generated device modules, making enhancements to TuyaMessage logic for multi-payload messages, rewriting the scanner and adding [Tuya Protocol 3.2, 3.4 & 3.5](https://github.com/jasonacox/tinytuya/discussions/260) support to TinyTuya!
  * Finally, thanks to the entire TinyTuya community for the great engagement, contributions and encouragement! See [RELEASE notes](https://github.com/jasonacox/tinytuya/blob/master/RELEASE.md#release-notes) for the ever growing journal of improvements and the incredible list of talent making this project possible.

## Related Projects

  * https://github.com/sean6541/tuyaapi Python API to the web api
  * https://github.com/codetheweb/tuyapi node.js
  * https://github.com/Marcus-L/m4rcus.TuyaCore - .NET
  * https://github.com/SDNick484/rectec_status/ - RecTec pellet smokers control (with Alexa skill)
  * https://github.com/TradeFace/tuyaface - Python Async Tuya API

## TinyTuya Powered Projects

Please feel free to submit a PR or open an issue to add your project.

* https://github.com/mafrosis/tinytuya2mqtt - A bridge between TinyTuya and Home Assistant via MQTT
* https://github.com/Whytey/pymoebot - A Python library intended to monitor and control the MoeBot robotic lawn mowers.
* https://github.com/make-all/tuya-local - Local support for Tuya devices in Home Assistant
* https://github.com/teejo75/mtghs - Provides an HTTP service for Moonraker to control Tuya outlets.
* https://github.com/Xenomes/Domoticz-TinyTUYA-Plugin - Plugin for Domoticz Home Automation

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/jasonacox/tinytuya",
    "name": "tinytuya",
    "maintainer": null,
    "docs_url": null,
    "requires_python": null,
    "maintainer_email": null,
    "keywords": null,
    "author": "Jason Cox",
    "author_email": "jason@jasonacox.com",
    "download_url": null,
    "platform": null,
    "description": "# TinyTuya\n\n[![License](https://img.shields.io/github/license/jasonacox/tinytuya)](https://img.shields.io/github/license/jasonacox/tinytuya)\n[![PyPI version](https://badge.fury.io/py/tinytuya.svg)](https://badge.fury.io/py/tinytuya)\n[![Build Test](https://github.com/jasonacox/tinytuya/actions/workflows/test.yml/badge.svg)](https://github.com/jasonacox/tinytuya/actions/workflows/test.yml)\n[![Contrib Test](https://github.com/jasonacox/tinytuya/actions/workflows/contrib.yml/badge.svg)](https://github.com/jasonacox/tinytuya/actions/workflows/contrib.yml)\n[![PyPI - Python Version](https://img.shields.io/pypi/pyversions/tinytuya)](https://img.shields.io/pypi/pyversions/tinytuya)\n[![PyPI Downloads](https://static.pepy.tech/badge/tinytuya/month)](https://static.pepy.tech/badge/tinytuya/month)\n\nPython module to interface with Tuya WiFi smart devices\n\n## Description\n\nThis python module controls and reads state of [Tuya](https://en.tuya.com/) compatible WiFi Smart Devices (Plugs, Switches, Lights, Window Covers, etc.) using the local area network (LAN) or the cloud (TuyaCloud API).  This is a compatible replacement for the `pytuya` PyPI module and currently supports Tuya Protocols 3.1, 3.2, 3.3, 3.4 and 3.5.\n\n[Tuya](https://en.tuya.com/) devices are designed to communicate with the TuyaCloud but most also expose a local area network API.  This allows us to directly control the devices without using the cloud. This python module provides a way to poll status and issue commands to these devices.\n\nTinyTuya can also connect to the Tuya Cloud to poll status and issue commands to Tuya devices.\n\n![TinyTuya Diagram](https://raw.githubusercontent.com/jasonacox/tinytuya/master/docs/TinyTuya-diagram.svg)\n\n```python\n# Example Usage of TinyTuya\nimport tinytuya\n\nd = tinytuya.Device('DEVICE_ID_HERE', 'IP_ADDRESS_HERE', 'LOCAL_KEY_HERE', version=3.3)\ndata = d.status() \nprint('Device status: %r' % data)\n```\n\nNOTE: Devices need to be **activated** by Smart Life App.\n\n## TinyTuya Installation  \n\n```bash\n# Install TinyTuya Library\npython -m pip install tinytuya\n\n# Optional: Install Command Line Tool\npipx install tinytuya\n```\n\nPip will attempt to install `cryptography`, `requests` and `colorama` if not already installed.\n\n## Tuya Device Preparation\n\nControlling and monitoring Tuya devices on your network requires the following:\n* *Address* - Network address (IPv4) of the device e.g. 10.0.1.100\n* *Device ID* - Unique identifier for the Tuya device\n* *Version* - Tuya protocol version used (3.1, 3.2, 3.3, 3.4 or 3.5)\n* *Local_Key* - Security key needed to access the Tuya device. See [Setup Wizard](https://github.com/jasonacox/tinytuya#setup-wizard---getting-local-keys) to get these keys.\n\n### Network Scanner\n\nTinyTuya has a built in network scanner that can be used to find Tuya Devices on your local network. It will show *Address*, *Device ID* and *Version* for each device. Your LAN and firewall will need to allow UDP (6666, 6667 and 7000) and TCP (6668) traffic.\n\n```bash\npython -m tinytuya scan\n```\n\n### Setup Wizard - Getting Local Keys\n\nTinyTuya has a built-in setup Wizard that uses the Tuya IoT Cloud Platform to generate a JSON list (devices.json) of all your registered devices, including secret *Local_Key* and *Name* of your devices. Follow the steps below:\n\n1. PAIR - Download the *Smart Life App* or *Tuya Smart App*, available for [iPhone](https://itunes.apple.com/us/app/smart-life-smart-living/id1115101477?mt=8) or [Android](https://play.google.com/store/apps/details?id=com.tuya.smartlife&hl=en). Set up your SmartLife account and pair all of your Tuya devices (this is important as you cannot access a device that has not been paired).  Do not use a 'guest' account, otherwise it will get deleted without confirmation at step 3.\n\n2. SCAN (Optional) - Run the TinyTuya scan to get a list of Tuya devices on your network. It will show device *Address*, *Device ID* and *Version* number (3.x):\n    ```bash\n    python -m tinytuya scan\n    ```\n    NOTE: You will need to use one of the displayed *Device IDs* for step 4.\n\n3. TUYA ACCOUNT - Set up a Tuya Account (see [PDF Instructions](https://github.com/jasonacox/tinytuya/files/12836816/Tuya.IoT.API.Setup.v2.pdf)):\n    * *NOTE: Tuya often changes their portal and services. Please open an [issue](https://github.com/jasonacox/tinytuya/issues) with screenshots if we need to update these instructions.*\n    * Create a Tuya Developer account on [iot.tuya.com](https://iot.tuya.com/). When it asks for the \"Account Type\", select \"Skip this step...\" (see [screenshot](https://user-images.githubusercontent.com/836718/213877860-34c39851-5671-4c9f-b4d5-251873f18c77.png)).  \n    * Click on \"Cloud\" icon -> \"Create Cloud Project\"\n      1. Pick the correct Data Center \"Region\" for your location (check [HERE](https://developer.tuya.com/en/docs/iot/oem-app-data-center-distributed?id=Kafi0ku9l07qb#title-3-Western%20America%20Data%20Center) to find your Region).  This will be used by TinyTuya Wizard ([screenshot](https://user-images.githubusercontent.com/836718/138598647-c9657e49-1a89-4ed6-8105-ceee95d9513f.png)).\n      2. Skip the configuration wizard but remember the Authorization Key: *API ID* and *Secret* for below ([screenshot](https://user-images.githubusercontent.com/836718/138598788-f74d2fe8-57fa-439c-8003-18735a44e7e5.png)).\n    * Click on \"Cloud\" icon -> Select your project -> **Devices** -> **Link Tuya App Account** ([see screenshot](https://user-images.githubusercontent.com/836718/155827671-44d5fce4-0119-4d0e-a224-ef3715fafc24.png))\n    * Click **Add App Account** ([screenshot](https://user-images.githubusercontent.com/836718/273364035-0fd133b7-4e9e-4a6d-900e-efe63d69f1a0.png)) and it will pop-up a \"Link Tuya App Account\" dialog, chose \"Automatic\" and \"Read Only Status\" (it will still alow commands). Click OK and it will display a QR code. Scan the QR code with the *Smart Life app* on your Phone (see step 1 above) by going to the \"Me\" tab in the *Smart Life app* and clicking on the QR code button `[..]` in the upper right hand corner of the app. When you scan the QR code, it will link all of the devices registered in your *Smart Life app* into your Tuya IoT project. If the QR code will not scan then make sure to disable any browser theming plug-ins (such as Dark Reader) and try again.\n    * **NO DEVICES?** If no devices show up after scanning the QR code, you will need to select a different data center and edit your project (or create a new one) until you see your paired devices from the *Smart Life App* show up. ([screenshot](https://user-images.githubusercontent.com/35581194/148679597-391adecb-a271-453b-90c0-c64cdfad42e4.png)). The data center may not be the most logical. As an example, some in the UK have reported needing to select \"Central Europe\" instead of \"Western Europe\".\n    * **SERVICE API:** Under \"Service API\" ensure these APIs are listed: `IoT Core` and `Authorization`. To be sure, click subscribe again on every service.  Very important: **disable popup blockers** otherwise subscribing won't work without providing any indication of a failure. Make sure you authorize your Project to use those APIs:\n        - Click \"Service API\" tab\n        - Click \"**Go to Authorize**\" button\n        - Select the API Groups from the dropdown and click `Subscribe` ([screenshot](https://user-images.githubusercontent.com/38729644/128742724-9ed42673-7765-4e21-94c8-76022de8937a.png))\n    * **RENEWAL:** The subscription to the `IoT Core` service expires after some time. By default, when you subscribe to it for the first time, it will last for one month. Once expired, the setup wizard won't be able to communicate with the Tuya account anymore, so it needs to be renewed. As of November 12th 2024, it can be renewed for a duration of 1, 3 or 6 months by simply filling in a form with some basic information (e.g. purpose of the project, type of developer).\n\n5. WIZARD - Run Setup Wizard:\n    * From your Linux/Mac/Win PC run the TinyTuya Setup **Wizard** to fetch the *Local_Keys* for all of your registered devices:\n      ```bash\n      python -m tinytuya wizard   # use -nocolor for non-ANSI-color terminals\n      ```\n    * The **Wizard** will prompt you for the *API ID* key, API *Secret*, API *Region* (cn, us, us-e, eu, eu-w, or in) from your Tuya IoT project as set in Step 3 above.\n        * To find those again, go to [iot.tuya.com](https://iot.tuya.com/), choose your project and click `Overview`\n            * API Key: Access ID/Client ID\n            * API Secret: Access Secret/Client Secret\n    * It will also ask for a sample *Device ID*.  You can have the wizard scan for one (enter `scan`), use one from step 2 above or in the Device List on your Tuya IoT project.\n    * The **Wizard** will poll the Tuya IoT Cloud Platform and print a JSON list of all your registered devices with the \"name\", \"id\" and \"key\" of your registered device(s). The \"key\"s in this list are the Devices' *Local_Key* you will use to access your device. \n    * In addition to displaying the list of devices, **Wizard** will create a local file `devices.json` that TinyTuya will use to provide additional details for scan results from `tinytuya.deviceScan()` or when running `python -m tinytuya scan`. The wizard also creates a local file `tuya-raw.json` that contains the entire payload from Tuya Cloud.\n    * The **Wizard** will ask if you want to poll all the devices. If you do, it will display the status of all devices on record and create a `snapshot.json` file with these results. Make sure your LAN and firewall permit UDP (6666, 6667 and 7000) and TCP (6668) traffic.\n\nNotes:\n* If you ever reset or re-pair your smart devices, the *Local_Key* will be reset and you will need to repeat the steps above.\n* The TinyTuya *Wizard* was inspired by the TuyAPI CLI which is an alternative way to fetch the *Local_Keys*: `npm i @tuyapi/cli -g` and run `tuya-cli wizard`  \n\n## Programming with TinyTuya\n\nAfter importing tinytuya, you create a device handle for the device you want to read or control.  Here is an example for a Tuya smart switch or plug:\n\n```python\nimport tinytuya\n\n# Connect to Device\nd = tinytuya.OutletDevice(\n    dev_id='DEVICE_ID_HERE',\n    address='IP_ADDRESS_HERE',      # Or set to 'Auto' to auto-discover IP address\n    local_key='LOCAL_KEY_HERE', \n    version=3.3)\n\n# Get Status\ndata = d.status() \nprint('set_status() result %r' % data)\n\n# Turn On\nd.turn_on()\n\n# Turn Off\nd.turn_off()\n```\n\n### TinyTuya Module Classes and Functions \n```\nClasses\n\n  AESCipher - Cryptography Helpers\n  XenonDevice(args...) - Base Class\n    Device(args...) - Tuya Class for Devices\n      OutletDevice(args...)\n      CoverDevice(args...)\n      BulbDevice(args...)\n        Where args:\n          dev_id (str): Device ID e.g. 01234567891234567890\n          address (str): Device Network IP Address e.g. 10.0.1.99 or \"Auto\" to auto-find\n          local_key (str): The encryption key\n          dev_type (str): Device type for payload options (see below)\n          version = 3.1 (float): Tuya Protocol (e.g. 3.1, 3.2, 3.3, 3.4, 3.5)\n          persist = False (bool): Keep TCP link open\n          cid = None (str): Optional sub device id\n          node_id = None (str): Alias for cid\n          parent = None (object): Gateway device object this is a child of\n          port = TCPPORT (int): The port to connect to device\n          connection_timeout = 5 (int): Timeout in seconds\n          connection_retry_limit = 5 (int)\n          connection_retry_delay = 5 (int)\n\n          Total timeout = (connection_timeout * connection_retry_limit) + \n                          (connection_retry_delay * (connection_retry_limit - 1))\n                          Defaults: (5 * 5) + (5 * (5 - 1)) = 45 seconds\n                        \n  Cloud(apiRegion, apiKey, apiSecret, apiDeviceID, new_sign_algorithm)\n\nTinyTuya Base Functions\n    devices = deviceScan()                        # Returns dictionary of devices found on local network\n    scan()                                        # Interactive scan of local network\n    wizard()                                      # Interactive setup wizard\n    set_debug(toggle, color)                      # Activate verbose debugging output\n    pack_message(msg, hmac_key)                   # Packs a TuyaMessage(), encrypting or adding a CRC if required \n    unpack_message(data, hmac_key, header, \n                    no_retcode)                   # Unpacks a TuyaMessage() \n    parse_header(data)                            # Unpacks just the header part of a message into a TuyaHeader()\n    find_device(dev_id, address)                  # Scans network for Tuya devices with either ID = dev_id or IP = address\n    device_info(dev_id)                           # Searches DEVICEFILE (usually devices.json) for device with ID\n    assign_dp_mappings(tuyadevices, mappings)     # Adds mappings to all the devices in the tuyadevices list\n    decrypt_udp(msg)                              # Decrypts a UDP network broadcast packet\n\n Device Functions (All Devices)\n    json = status()                               # returns json payload\n    subdev_query(nowait)                          # query sub-device status (only for gateway devices)\n    set_version(version)                          # 3.1 [default], 3.2, 3.3 or 3.4\n    set_socketPersistent(False/True)              # False [default] or True\n    set_socketNODELAY(False/True)                 # False or True [default]\n    set_socketRetryLimit(integer)                 # retry count limit [default 5]\n    set_socketRetryDelay(integer)                 # retry delay [default 5]\n    set_socketTimeout(timeout)                    # set connection timeout in seconds [default 5]\n    set_dpsUsed(dps_to_request)                   # add data points (DPS) to request\n    add_dps_to_request(index)                     # add data point (DPS) index set to None\n    set_retry(retry=True)                         # retry if response payload is truncated\n    set_status(on, switch=1, nowait)              # Set status of switch to 'on' or 'off' (bool)\n    set_value(index, value, nowait)               # Set int value of any index.\n    set_multiple_values(index_value_dict, nowait) # Set multiple values with a single request\n    heartbeat(nowait)                             # Send heartbeat to device\n    updatedps(index=[1], nowait)                  # Send updatedps command to device\n    turn_on(switch=1, nowait)                     # Turn on device / switch #\n    turn_off(switch=1, nowait)                    # Turn off\n    set_timer(num_secs, nowait)                   # Set timer for num_secs\n    set_sendWait(num_secs)                        # Time to wait after sending commands before pulling response\n    detect_available_dps()                        # Return list of DPS available from device\n    generate_payload(command, data,...            # Generate TuyaMessage payload for command with data\n    send(payload)                                 # Send payload to device (do not wait for response)\n    receive()                                     # Receive payload from device\n\nOutletDevice Additional Functions\n    set_dimmer(percentage):\n\nBulbDevice Additional Functions\n    set_colour(r, g, b, nowait):\n    set_hsv(h, s, v, nowait):\n    set_white(brightness, colourtemp, nowait):\n    set_white_percentage(brightness=100, colourtemp=0, nowait):\n    set_brightness(brightness, nowait):\n    set_brightness_percentage(brightness=100, nowait):\n    set_colourtemp(colourtemp, nowait):\n    set_colourtemp_percentage(colourtemp=100, nowait):\n    set_scene(scene, nowait):                     # 1=nature, 3=rave, 4=rainbow\n    set_mode(mode='white', nowait):               # white, colour, scene, music\n    result = brightness():\n    result = colourtemp():\n    (r, g, b) = colour_rgb():\n    (h,s,v) = colour_hsv()\n    result = state():\n\nCoverDevice Additional Functions\n    open_cover(switch=1):\n    close_cover(switch=1):\n    stop_cover(switch=1):\n\nCloud Functions\n    setregion(apiRegion)\n    cloudrequest(url, action=[POST if post else GET], post={}, query={})\n    getdevices(verbose=False)\n    getstatus(deviceid)\n    getfunctions(deviceid)\n    getproperties(deviceid)\n    getdps(deviceid)\n    sendcommand(deviceid, commands [, uri])\n    getconnectstatus(deviceid)\n    getdevicelog(deviceid, start=[now - 1 day], end=[now], evtype=\"1,2,3,4,5,6,7,8,9,10\", size=100, params={})\n      -> when start or end are negative, they are the number of days before \"right now\"\n          i.e. \"start=-1\" is 1 day ago, \"start=-7\" is 7 days ago\n          \n```\n\n### TinyTuya Error Codes\n\nStarting with v1.2.0 TinyTuya functions will return error details in the JSON data responses instead of raising exceptions.  The format for this response:\n\n```json\n{ \"Error\":\"Invalid JSON Payload\", \"Err\":\"900\", \"Payload\":\"{Tuya Message}\" }\n```\n\nThe \"Err\" number will be one of these:\n\n* 900 (ERR_JSON) - Invalid JSON Response from Device\n* 901 (ERR_CONNECT) - Network Error: Unable to Connect\n* 902 (ERR_TIMEOUT) - Timeout Waiting for Device\n* 903 (ERR_RANGE) - Specified Value Out of Range\n* 904 (ERR_PAYLOAD) - Unexpected Payload from Device\n* 905 (ERR_OFFLINE) - Network Error: Device Unreachable\n* 906 (ERR_STATE) - Device in Unknown State\n* 907 (ERR_FUNCTION) - Function Not Supported by Device\n* 908 (ERR_DEVTYPE) - Device22 Detected: Retry Command\n* 909 (ERR_CLOUDKEY) - Missing Tuya Cloud Key and Secret\n* 910 (ERR_CLOUDRESP) - Invalid JSON Response from Cloud\n* 911 (ERR_CLOUDTOKEN) - Unable to Get Cloud Token\n* 912 (ERR_PARAMS) - Missing Function Parameters\n* 913 (ERR_CLOUD) - Error Response from Tuya Cloud\n* 914 (ERR_KEY_OR_VER) - Check device key or version\n\n### Example Usage\n\nSee the sample python script [test.py](test.py) for an OutletDevice example or look in the [examples](examples) directory for other scripts.\n\n```python\nimport tinytuya\n\n\"\"\"\nOUTLET Device\n\"\"\"\nd = tinytuya.Device('DEVICE_ID_HERE', 'IP_ADDRESS_HERE', 'LOCAL_KEY_HERE', version=3.3)\ndata = d.status()  \n\n# Show status and state of first controlled switch on device\nprint('Dictionary %r' % data)\nprint('State (bool, true is ON) %r' % data['dps']['1'])  \n\n# Toggle switch state\nswitch_state = data['dps']['1']\ndata = d.set_status(not switch_state)  # This requires a valid key\nif data:\n    print('set_status() result %r' % data)\n\n# On a switch that has 4 controllable ports, turn the fourth OFF (1 is the first)\ndata = d.set_status(False, 4)\nif data:\n    print('set_status() result %r' % data)\n    print('set_status() extra %r' % data[20:-8])\n\n\"\"\"\nRGB Bulb Device\n\"\"\"\nimport time\n\nd = tinytuya.BulbDevice('DEVICE_ID_HERE', 'IP_ADDRESS_HERE', 'LOCAL_KEY_HERE')\nd.set_version(3.3)  # IMPORTANT to set this regardless of version\nd.set_socketPersistent(True)  # Optional: Keep socket open for multiple commands\ndata = d.status()\n\n# Show status of first controlled switch on device\nprint('Dictionary %r' % data)\n\n# Set to RED Color - set_colour(r, g, b):\nd.set_colour(255,0,0)  \n\n# Cycle through the Rainbow\nrainbow = {\"red\": [255, 0, 0], \"orange\": [255, 127, 0], \"yellow\": [255, 200, 0],\n          \"green\": [0, 255, 0], \"blue\": [0, 0, 255], \"indigo\": [46, 43, 95],\n          \"violet\": [139, 0, 255]}\nfor color in rainbow:\n    [r, g, b] = rainbow[color]\n    d.set_colour(r, g, b, nowait=True)  # nowait = Go fast don't wait for response\n    time.sleep(0.25)\n\n# Brightness: Type A devices range = 25-255 and Type B = 10-1000\nd.set_brightness(1000)\n\n# Set to White - set_white(brightness, colourtemp):\n#    colourtemp: Type A devices range = 0-255 and Type B = 0-1000\nd.set_white(1000,10)\n\n# Set Bulb to Scene Mode\nd.set_mode('scene')\n\n# Scene Example: Set Color Rotation Scene\nd.set_value(25, '07464602000003e803e800000000464602007803e803e80000000046460200f003e803e800000000464602003d03e803e80000000046460200ae03e803e800000000464602011303e803e800000000')\n\n```\n### Example Device Monitor\n\nYou can set up a persistent connection to a device and then monitor the state changes with a continual loop. This is helpful for troubleshooting and discovering DPS values.\n\n```python\nimport tinytuya\n\nd = tinytuya.OutletDevice('DEVICEID', 'DEVICEIP', 'DEVICEKEY', version=3.3, persist=True)\n\nprint(\" > Send Request for Status < \")\nd.status(nowait=True)\n\nprint(\" > Begin Monitor Loop <\")\nwhile(True):\n    # See if any data is available\n    data = d.receive()\n    print('Received Payload: %r' % data)\n\n    # Send keep-alive heartbeat\n    if not data:\n        print(\" > Send Heartbeat Ping < \")\n    \td.heartbeat()\n\n    # NOTE If you are not seeing updates, you can force them - uncomment:\n    # print(\" > Send Request for Status < \")\n    # d.status(nowait=True)\n\n    # NOTE Some smart plugs require an UPDATEDPS command to update power data\n    # print(\" > Send DPS Update Request < \")\n    # payload = d.generate_payload(tinytuya.UPDATEDPS)\n    # d.send(payload)    \n\n```\n\n### Tuya Cloud Access\n\nYou can poll and manage Tuya devices using the `Cloud` class and functions.\n\n_CAUTION: The free Tuya IoT Developer (Trial) account allows a very limited number of Cloud API calls. Be aware of the restrictions before enabling any automation that makes frequent calls._\n\n```python\nimport tinytuya\n\n# Connect to Tuya Cloud\n# c = tinytuya.Cloud()  # uses tinytuya.json \nc = tinytuya.Cloud(\n        apiRegion=\"us\", \n        apiKey=\"xxxxxxxxxxxxxxxxxxxx\", \n        apiSecret=\"xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\", \n        apiDeviceID=\"xxxxxxxxxxxxxxxxxxID\")\n\n# Display list of devices\ndevices = c.getdevices()\nprint(\"Device List: %r\" % devices)\n\n# Select a Device ID to Test\nid = \"xxxxxxxxxxxxxxxxxxID\"\n\n# Display Properties of Device\nresult = c.getproperties(id)\nprint(\"Properties of device:\\n\", result)\n\n# Display Status of Device\nresult = c.getstatus(id)\nprint(\"Status of device:\\n\", result)\n\n# Send Command - Turn on switch\ncommands = {\n    \"commands\": [\n        {\"code\": \"switch_1\", \"value\": True},\n        {\"code\": \"countdown_1\", \"value\": 0},\n    ]\n}\nprint(\"Sending command...\")\nresult = c.sendcommand(id,commands)\nprint(\"Results\\n:\", result)\n```\n\nUp to one week of device logs can also be pulled from the Cloud.  By default getdevicelog() will pull 1 day of logs or 5000 log entries, whichever comes first.  The returned timestamps are unixtime*1000, and event_id 7 (data report) will probably be the most useful.\n\n```python\nimport tinytuya\nimport json\n\nc = tinytuya.Cloud()\n#r = c.getdevicelog( '00112233445566778899', start=-1, end=0, size=0, max_fetches=50 )\n#r = c.getdevicelog( '00112233445566778899', start=1669990000, end=1669990300, size=20 )\nr = c.getdevicelog( '00112233445566778899' )\nprint( json.dumps(r, indent=2) )\n```\n\n### Encryption Notes\n\nTuya devices use AES encryption which is not available in the Python standard library. **PyCA/cryptography** is recommended and installed by default. Other options include **PyCryptodome** , **PyCrypto** and **pyaes**.\n\n* Deprecation notice for pyaes: The pyaes library works for Tuya Protocol <= 3.4 but will not work for 3.5 devices. This is because pyaes does not support GCM which is required for v3.5 devices.\n\n### Command Line\n\nTinyTuya provides a built-in command line interface to get Local key, scan and poll devices.\n\nInstallation\n\n```bash\n# Option-1: pip install tinytuya\npython -m tinytuya\n\n# Option-2: pipx install tinytuya\ntinytuya \n```\n\nCommand Line Usage\n\n```\ntinytuya <command> [-debug] [-nocolor] [-h] [-yes] [-no-poll] [-device-file FILE] [-snapshot-file FILE]\n\n  wizard         Launch Setup Wizard to get Tuya Local KEYs.\n  scan           Scan local network for Tuya devices.\n  devices        Scan all devices listed in devices.json file.\n  snapshot       Scan devices listed in snapshot.json file.\n  json           Scan devices listed in snapshot.json file [JSON].\n\n  Wizard\n      tinytuya wizard [-h] [-debug] [-force [0.0.0.0/24 ...]] [-no-broadcasts] [-nocolor] [-yes] [-no-poll]\n                [-device-file FILE] [-raw-response-file FILE] [-snapshot-file FILE] [-credentials-file FILE]\n                [-key KEY] [-secret SECRET] [-region {cn,eu,eu-w,in,us,us-e}] [-device DEVICE [DEVICE ...]]\n                [-dry-run] [max_time]\n\n        Common Options\n        max_time             Maximum time to find Tuya devices [Default: 18]\n        -no-broadcasts       Ignore broadcast packets when force scanning\n        -nocolor             Disable color text output.\n        -debug               Activate debug mode.\n        -h, -help            Show usage help for command.\n        -yes, -y             Answer \"yes\" to all questions\n        -no-poll, -no        Answer \"no\" to \"Poll?\" (overrides -yes)\n        -device-file FILE    JSON file to load devices from [Default: devices.json]\n        -snapshot-file FILE  JSON file to load/save snapshot from/to [Default: snapshot.json]\n        -force [0.0.0.0/24 ...], -f [0.0.0.0/24 ...]\n                             Force network scan of device IP addresses [Default: Auto-detects net/mask]\n        -no-broadcasts       Ignore broadcast packets when force scanning\n        -raw-response-file   JSON file to save the raw server response to [Default: tuya-raw.json]\n\n        Wizard Cloud API Options\n        -dry-run             Do not actually connect to the Cloud\n        -credentials-file    JSON file to load/save Cloud credentials from/to [Default: tinytuya.json]\n        -key KEY             Cloud API Key to use\n        -secret SECRET       Cloud API Secret to use\n        -region              Cloud API Region to use {cn,eu,eu-w,in,us,us-e}\n        -device DEVICE(S)    One or more Device ID(s) to use\n\n  Scan\n      tinytuya scan [-h] [-debug] [-force [0.0.0.0/24 ...]] [-no-broadcasts] [-nocolor] [-yes] \n                [-device-file FILE] [-snapshot-file FILE] [max_time]\n\n  Devices\n      tinytuya devices [-h] [-debug] [-force [0.0.0.0/24 ...]] [-no-broadcasts] [-nocolor] [-yes] \n                [-no-poll] [-device-file FILE] [-snapshot-file FILE] [max_time]\n\n  Snapshot\n      tinytuya snapshot [-h] [-debug] [-nocolor] [-yes] [-no-poll] [-device-file FILE] [-snapshot-file FILE]\n\n  JSON\n      tinytuya json [-h] [-debug] [-device-file FILE] [-snapshot-file FILE]\n\n```\n\n### Scan Tool \nThe function `tinytuya.scan()` will listen to your local network (UDP 6666 and 6667) and identify Tuya devices broadcasting their Address, Device ID, Product ID and Version and will print that and their stats to stdout.  This can help you get a list of compatible devices on your network. The `tinytuya.deviceScan()` function returns all found devices and their stats (via dictionary result).\n\nYou can run the scanner from the command line using these interactive commands:\n  ```bash\n  # Listen for Tuya Devices and match to devices.json if available\n  python -m tinytuya scan\n\n  # The above creates a snapshot.json file with IP addresses for devices\n  # You can use this command to get a rapid poll of status of all devices\n  python -m tinytuya snapshot\n\n  # The sames thing as above but with a non-interactive JSON response\n  python -m tinytuya json\n\n  # List all register devices discovered from Wizard and poll them\n  python -m tinytuya devices\n\n  ```\n\nBy default, the scan functions will retry 15 times to find new devices. If you are not seeing all your devices, you can increase max_retries by passing an optional arguments (eg. 50 retries):\n\n  ```bash\n  # command line\n  python -m tinytuya scan 50\n  ```\n\n  ```python\n  # invoke verbose interactive scan\n  tinytuya.scan(50)\n\n  # return payload of devices\n  devices = tinytuya.deviceScan(false, 50)\n  ```\n\n## Troubleshooting\n\n* Tuya devices only allow one TCP connection at a time.  Make sure you close the TuyaSmart or SmartLife app before using *TinyTuya* to connect.\n* Some devices ship with older firmware that may not work with *TinyTuya*. If you're experiencing issues, please try updating the device's firmware in the official app.\n* The LOCAL KEY for Tuya devices will change every time a device is removed and re-added to the TuyaSmart app. If you're getting decrypt errors, try getting the key again as it might have changed. \n* Devices running protocol version 3.1 (e.g. below Firmware 1.0.5) do not require a device *Local_Key* to read the status. All devices will require a device *Local_Key* to control the device.\n* Some devices with 22 character IDs will require additional setting to poll correctly. TinyTuya will attempt to detect and accomodate for this, but it can be specified directly:\n  ```python\n  a = tinytuya.OutletDevice('here_is_my_key', '192.168.x.x', 'secret_key_here', 'device22')\n  a.set_version(3.3)\n  a.set_dpsUsed({\"1\": None})  # This needs to be a datapoint available on the device\n  data =  a.status()\n  print(data)\n  ```\n* Windows 10 Users - TinyTuya `wizard` and `scan` interactive tools use ANSI color. This will work correctly in PowerShell but will show cryptic escape codes when run in Windows `CMD`.  You can fix this by using the `-nocolor` option on tinytuya, or by changing the Windows `CMD` console registry to process ANSI escape codes by doing something like this:\n  ```\n  reg add HKEY_CURRENT_USER\\Console /v VirtualTerminalLevel /t REG_DWORD /d 0x00000001 /f\n  ```\n\n## User Contributed Device Modules\n\nIn addition to the built-in `OutletDevice`, `BulbDevice` and `CoverDevice` device support, the community is encourage to submit additional device modules which are available here: [Contrib Library](https://github.com/jasonacox/tinytuya/tree/master/tinytuya/Contrib):\n\n```python\n# Example usage of community contributed device modules\nfrom tinytuya.Contrib import ThermostatDevice\n\nthermo = ThermostatDevice( 'abcdefghijklmnop123456', '172.28.321.475', '1234567890123abc' )\n```\n\n## Tuya Data Points - DPS Table\n\nThe Tuya devices send back data points (DPS) also called device function points, in a json string.  The DPS attributes define the state of the device.  Each key in the DPS dictionary refers to key value pair, the key is the DP ID and its value is the dpValue. You can refer to the [Tuya developer platform](https://iot.tuya.com/index/) for definition of function points for the products.\n\nThe following table represents several of the standard Tuya DPS values and their properties. It represents data compiled from Tuya documentation and self-discovery. Devices may vary. Feedback or additional data would would be appreciated.  Please submit a Issue or Pull Request if you have additional data that would be helpful for others.\n\nTo find Tuya DPS for devices not listed below, you can discover the DPS values using the Tuya IoT platform. See this help here: [Find your Data Point](https://www.zigbee2mqtt.io/advanced/support-new-devices/03_find_tuya_data_points.html#_8-display-device-logs).\n\nDPS Read and Set Example:\n```python\n# Read Value of DPS 25\ndata = d.status()  \nprint(\"Value of DPS 25 is \", data['dps']['25'])\n\n# Set Value of DPS 25\nd.set_value(25, '010e0d0000000000000003e803e8')\n```\n\n### Version 3.1 Devices\n\n#### Version 3.1 (and some 3.3) - Plug or Switch Type\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n|1|Switch|bool|True/False||\n|2|Countdown?|integer|0-86400|s|\n|4|Current|integer|0-30000|mA|\n|5|Power|integer|0-50000|W|\n|6|Voltage|integer|0-5000|V|\n\n#### Version 3.1 - Light Type (RGB)\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n| 1|Switch|bool|True/False||\n| 2|Mode|enum|white,colour,scene,music||\n| 3|Bright|integer|10-1000*||\n| 4|Color Temp|integer|0-1000*||\n| 5|Color|hexstring|r:0-255,g:0-255,b:0-255,h:0-360,s:0-255,v:0-255|rgb+hsv|\n\n### Version 3.3 Devices\n\n#### Version 3.3 - Plug, Switch, Power Strip Type\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n|1|Switch 1|bool|True/False||\n|2|Switch 2|bool|True/False||\n|3|Switch 3|bool|True/False||\n|4|Switch 4|bool|True/False||\n|5|Switch 5|bool|True/False||\n|6|Switch 6|bool|True/False||\n|7|Switch 7/usb|bool|True/False||\n|9|Countdown 1|integer|0-86400|s|\n|10|Countdown 2|integer|0-86400|s|\n|11|Countdown 3|integer|0-86400|s|\n|12|Countdown 4|integer|0-86400|s|\n|13|Countdown 5|integer|0-86400|s|\n|14|Countdown 6|integer|0-86400|s|\n|15|Countdown 7|integer|0-86400|s|\n|17|Add Electricity|integer|0-50000|kwh|\n|18|Current|integer|0-30000|mA|\n|19|Power|integer|0-50000|W|\n|20|Voltage|integer|0-5000|V|\n|21|Test Bit|integer|0-5|n/a|\n|22|Voltage coeff.|integer|0-1000000||\n|23|Current coeff.|integer|0-1000000||\n|24|Power coeff.|integer|0-1000000||\n|25|Electricity coeff.|integer|0-1000000||\n|26|Fault|fault|ov_cr||\n|38|Power-on state setting|enum|off, on, memory||\n|39|Overcharge Switch|bool|True/False||\n|40|Indicator status setting|enum|none, on, relay, pos||\n|41|Child Lock|bool|True/False||\n|42|UNKNOWN||||\n|43|UNKNOWN||||\n|44|UNKNOWN||||\n\nNote: Some 3.3 energy management plugs use the DPS values of the 3.1 plug above.\n\n#### Version 3.3 - Dimmer Switch\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n| 1|Switch|bool|True/False||\n| 2|Brightness|integer|10-1000*||\n| 3|Minimum of Brightness|integer|10-1000*||\n| 4|Type of light source1|enum|LED,incandescent,halogen||\n| 5|Mode|enum|white||\n\n#### Version 3.3 - Light Type (RGB)\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n| 20|Switch|bool|True/False||\n| 21|Mode|enum|white,colour,scene,music||\n| 22|Bright|integer|10-1000*||\n| 23|Color Temp|integer|0-1000||\n| 24|Color|hexstring|h:0-360,s:0-1000,v:0-1000|hsv|\n| 25|Scene|string|n/a||\n| 26|Left time|integer|0-86400|s|\n| 27|Music|string|n/a||\n| 28|Debugger|string|n/a||\n| 29|Debug|string|n/a||\n| 30|Rhythms|n/a|n/a||\n| 31|Go To Sleep|n/a|n/a||\n| 32|Wake Up|n/a|n/a||\n| 33|Power Off Memory|n/a|n/a||\n| 34|Do not Disturb|n/a|n/a||\n| 41|Remote Control Switch|n/a|n/a||\n| 209|Cycle Timing|n/a|n/a||\n| 210|Vaction Timing|n/a|n/a||\n\n#### Version 3.3 - Automated Curtain Type\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n|1|Curtain Switch 1|enum|open, stop, close, continue||\n|2|Percent control 1|integer|0-100|%|\n|3|Accurate Calibration 1|enum|start, end||\n|4|Curtain Switch 2|enum|open, stop, close, continue||\n|5|Percent control 2|integer|0-100||\n|6|Accurate Calibration 2|enum|start, end||\n|8|Motor Steer 1|enum|forward, back||\n|9|Motor steer 2|enum|forward, back||\n|10|Quick Calibration 1|integer|1-180|s|\n|11|Quick Calibration 2|integer|1-180|s|\n|12|Motor Mode 1|enum|strong_power, dry_contact||\n|13|Motor Mode 2|enum|strong_power, dry_contact||\n|14|Light mode|enum|relay, pos, none||\n\n#### Version 3.3 - Fan Switch Type\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n|1|Fan switch|bool|True/False|n/a|\n|2|Fan countdown|integer|0-86400|s|\n|3|Fan speed|enum|level_1, level_2, level_3, level_4, level_5||\n|4|Fan speed|integer|1-100|%|\n|5|Fan light switch|bool|True/False||\n|6|Brightness integer|integer|10-1000||\n|7|Fan light countdown|integer|0-86400||\n|8|Minimum brightness|integer|10-1000||\n|9|Maximum brightness|integer|10-1000||\n|10|Mode|enum|white||\n|11|Power-on state setting|enum|off, on, memory||\n|12|Indicator status setting|enum|none, relay, pos||\n|13|Backlight switch|bool|True/False||\n\n#### Version 3.3 - Universal IR Controller with optional Temp/Humidity\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n|101|Current Temperature|integer|0-600|10x Celsius|\n|102|Current Humidity|integer|0-100|%|\n|201|IR Commands (set only)|JSON*|n/a|n/a|\n\n  ```python\n  # The IR Commands JSON has the following format:\n  command = {\n      \"control\": \"send_ir\",\n      \"head\": \"\",\n      \"key1\": \"[[TO_BE_REPLACED]]\",\n      \"type\": 0,\n      \"delay\": 300,\n  }\n  # Sending the IR command:\n  payload = d.generate_payload(tinytuya.CONTROL, {\"201\": json.dumps(command)})\n  d.send(payload)\n  ```\n\nThe `key1` attribute is a base64 string that contains the IR signal. You can extract it using this procedure:\n\n1. Register a new IR device on Tuya Smart / Smart Life app (if not registered already) and map, setup or import your buttons.\n2. Tap multiple times on the button you wish to control.\n3. Go to [Tuya IoT Platform](https://iot.tuya.com/) and select your app under Cloud > Development section.\n4. Go to to the Device tab and select \"Debug Device\" on the parent device. Browse Device Logs section and retrieve the `key1` attribute that matches your tapping timestamp from step 2 above. Use that `key1` attribute in the payload example above.\n\nYou need to repeat these steps for each button (cloud logging is not always sequential).\n\n#### Version 3.3 - Sensor Type\n\n_Important Note:_\nBattery-powered Tuya sensors are usually designed to stay in sleep mode until a state change (eg.open or close alert). This means you will not be able to poll these devices except in the brief moment they awake, connect to the WiFi and send their state update payload the the Tuya Cloud. Keep in mind that if you manage to poll the device enough to keep it awake, you will likely quickly drain the battery.\n\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n|1|Door Sensor|bool|True/False||\n|2|Battery level state|enum|low, middle, high||\n|3|Battery level|integer|0-100|%|\n|4|Temper alarm|bool|True/False||\n|5|Flooding Detection State|enum|alarm, normal||\n|6|Luminance detection state|enum|low, middle, high, strong||\n|7|Current Luminance|integer|0-100|%|\n|8|Current Temperature|integer|400-2000||\n|9|Current Humidity|integer|0-100|%|\n|10|Shake State|enum|normal, vibration, drop, tilt||\n|11|Pressure State|enum|alarm, normal||\n|12|PIR state|enum|pir, none||\n|13|Smoke Detection State|enum|alarm, normal||\n|14|Smoke value|integer|0-1000||\n|15|Alarm Volume|enum|low, middle, high, mute||\n|16|Alarm Ringtone|enum|1, 2, 3, 4, 5||\n|17|Alarm Time|integer|0-60|s|\n|18|Auto-Detect|bool|True/False||\n|19|Auto-Detect Result|enum|checking, check_success, check_failure, others||\n|20|Preheat|bool|True/False||\n|21|Fault Alarm|fault|fault, serious_fault, sensor_fault, probe_fault, power_fault|Barrier|\n|22|Lifecycle|bool|True/False||\n|23|Alarm Switch|bool|True/False||\n|24|Silence|bool|True/False||\n|25|Gas Detection State|enum|alarm, normal||\n|26|Detected Gas|integer|0-1000||\n|27|CH4 Detection State|enum|alarm, normal||\n|28|CH4 value|integer|0-1000||\n|29|Alarm state|enum|alarm_sound, alarm_light, alarm_sound_light, normal||\n|30|VOC Detection State|enum|alarm, normal||\n|31|VOC value|integer|0-999||\n|32|PM2.5 state|enum|alarm, normal||\n|33|PM2.5 value|integer|0-999||\n|34|CO state|enum|alarm, normal||\n|35|CO value|integer|0-1000||\n|36|CO2 Detection State|enum|alarm, normal||\n|37|CO2 value|integer|0-1000||\n|38|Formaldehyde Detection State|enum|alarm, normal||\n|39|CH2O value|integer|0-1000||\n|40|Master mode|enum|disarmed, arm, home, sos||\n|41|Air quality index|enum|level_1, level_2, level_3, level_4, level_5, level_6||\n\nNOTE (*) - The range can vary depending on the device. As an example, for dimmers, it may be 10-1000 or 25-255.\n\n#### Version 3.3 - WiFi Air Quality Detector PM2.5/Formaldehyde/VOC/CO2/Temperature/Humidity\n\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n| 2 | PM2.5 value | integer | 0 - 999 | ug/m3 |\n| 18 | Current Temperature | integer | 0 - 850 | \u02daC (multiplied by 10) |\n| 19 | Current Humidity | integer | 0 - 1000 | % (multiplied by 10) |\n| 20 | CH2O (Formaldehyde) value | integer | 0 - 1000 | ppm |\n| 21 | VOC (Volatile organic compound) value | integer | 0 - 2000 | ppm |\n| 22 | CO2 value | integer | 350 - 2000 | ppm |\n\nExample device: https://www.aliexpress.com/item/1005005034880204.html\n\n#### Version 3.3 - Robot Mower Type\n\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n| 6    | Battery | integer |  0-100 | % |\n| 101  | Machine Status | enum | <ul><li>STANDBY MOWING</li><li>CHARGING</li><li>EMERGENCY</li><li>LOCKED</li><li>PAUSED</li><li>PARK</li><li>CHARGING_WITH_TASK_SUSPEND</li><li>FIXED_MOWING</li></ul> ||\n| 102  | Machine error | integer | 0, ? ||\n| 103  | Machine warning | enum | <ul><li>MOWER_LEAN</li><li>MOWER_EMERGENCY</li><li>MOWER_UI_LOCKED</li><ul> ||\n| 104  | Rain mode | boolean | True/False ||\n| 105  | Work time | interger | 1-99 | hours |\n| 106  | Machine password | byte str | ? ||\n| 107  | Clear machine appointment | boolean | True/False ||\n| 108  | Query machine reservation | boolean | True/False ||\n| 109  | Query partition parameters | boolean | True/False ||\n| 110  | Report machine reservation | byte str |||\n| 111  | Error log | byte str |||\n| 112  | Work log | byte str |||\n| 113  | Partition parameters | byte str |||\n| 114  | Work mode | enum | AutoMode/?? ||                                                                                                                           | 115  | Machine control CMD | enum | <ul><li>StartMowing</li><li>StartFixedMowing</li><li>PauseWork</li><li>CancelWork</li><li>StartReturnStation</li><ul> ||\n\nReference [pymoebot](https://github.com/Whytey/pymoebot) for further definition.\n\n#### Version 3.3 - 24v Thermostat (i.e. PCT513-TY)\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n|2|System Mode|enum|[ 'auto' 'cool' 'heat' 'off' others? ]||\n|16|Center of Setpoint, High-Resolution \u00b0C|integer|500-3200|\u00b0C x 100 in steps of 50|\n|17|Center of Setpoint, \u00b0F|integer|20-102|\u00b0F|\n|18*|Cooling Setpoint, Low-Resolution \u00b0F|integer|20-102|\u00b0F|\n|19*|Cooling Setpoint, Low-Resolution \u00b0C|integer|500-3200|\u00b0C|\n|20*|Heating Setpoint, Low-Resolution \u00b0F|integer|20-102|\u00b0F|\n|23|Display Units|enum|[ 'f' 'c' ]||\n|24|Current Temperature, High-Resolution \u00b0C|integer|500-3200|\u00b0C x 100 in steps of 50|\n|26*|Heating Setpoint, Low-Resolution \u00b0C|integer|5-32|\u00b0C|\n|27*|Temperature Correction|integer|-10 - +10||\n|29|Current Temperature, \u00b0F|integer|20-102|\u00b0F|\n|34|Current Humidity|integer|0-100|%|\n|45|Fault Flags|bitmask|[ e1 e2 e3 ]||\n|107|System Type|integer-as-string|?|?|\n|108*|Cooling Setpoint, High-Resolution \u00b0C|integer|500-3200|\u00b0C x 100 in steps of 50|\n|109*|Heating Setpoint, High-Resolution \u00b0C|integer|500-3200|\u00b0C x 100 in steps of 50|\n|110*|Cooling Setpoint, \u00b0F|integer|20-102|\u00b0F|\n|111*|Heating Setpoint, \u00b0F|integer|20-102|\u00b0F|\n|115|Fan Mode|enum|[ 'auto' 'cycle' 'on' ]||\n|116|\"at home/away from home\"|integer-as-string|?|?|\n|118|Schedule Data|base64|binary blob||\n|119|Schedule Enabled|bool|True/False||\n|120|Hold/Schedule|enum|[ 'permhold' 'temphold' 'followschedule' ]||\n|121|Vacation Data|base64|binary blob||\n|122|[Sensor Data, list 1](https://github.com/jasonacox/tinytuya/discussions/139)|base64|binary blob||\n|123|Minimum Fan Run Time|integer|0-55|minutes per hour|\n|125|[Sensor Data, list 2](https://github.com/jasonacox/tinytuya/discussions/139)|base64|binary blob||\n|126|[Sensor Data, list 3](https://github.com/jasonacox/tinytuya/discussions/139)|base64|binary blob||\n|127|[Sensor Data, list 4](https://github.com/jasonacox/tinytuya/discussions/139)|base64|binary blob||\n|128|[Sensor Data, list 5](https://github.com/jasonacox/tinytuya/discussions/139)|base64|binary blob||\n|129|System State|enum|[ 'fanon' 'coolfanon' 'alloff' others? ]||\n|130|Weather Forcast|?|?|?|\n\nNOTE (*) - Depending on the firmware, either 18/19/20/26/27 or 108/109/110/111/x are used, not both\n\nA user contributed module is available for this device in the [Contrib library](https://github.com/jasonacox/tinytuya/tree/master/tinytuya/Contrib):\n\n```python\nfrom tinytuya.Contrib import ThermostatDevice\n\nthermo = ThermostatDevice( 'abcdefghijklmnop123456', '172.28.321.475', '1234567890123abc' )\n```\n\nFor info on the Sensor Data lists, see https://github.com/jasonacox/tinytuya/discussions/139\n\n#### Version 3.4 - WIFI Dual Meter (i.e. PJ1103A)\n| DP ID        | Function Point | Type        | Range       | Units |\n| ------------- | ------------- | ------------- | ------------- |------------- |\n|1|Forward energy total|integer||kWh x 100|\n|2|Reverse energy total|integer||kWh x 100|\n|101|Power (A)|integer||W x 10|\n|102|Direction of current flow (A)| enum | <ul><li>FORWARD</li><li>REVERSE</li></ul> ||\n|104|Direction of current flow (B)| enum | <ul><li>FORWARD</li><li>REVERSE</li></ul> ||\n|105|Power (B)|integer||W x 10|\n|106|Energy forward (A)|integer||kWh x 100|\n|107|Energy reverse (A)|integer||kWh x 100|\n|108|Energy forward (B)|integer||kWh x 100|\n|109|Energy reverse (B)|integer||kWh x 100|\n|110|Power Factor (A)|integer||value x 100|\n|111|AC frequency|integer||Hz x 100|\n|112|AC voltage|integer||V x 10|\n|113|Current (A)|integer||mA|\n|114|Current (B)|integer||mA|\n|115|Total power|integer||W x 10|\n|116|Voltage Calibration|integer|800-1200|value * 1000|\n|117|Current Calibration (A)|integer|800-1200|value * 1000|\n|118|Power Calibration (A)|integer|800-1200|value * 1000|\n|119|Energy Calibration forward (A)|integer|800-1200|value * 1000|\n|121|Power Factor (B)|integer||value x 100|\n|122|Frequency Calibration|integer|800-1200|value * 1000|\n|123|Current Calibration (B)|integer|800-1200|value * 1000|\n|124|Power Calibration (B)|integer|800-1200|value * 1000|\n|125|Energy Calibration forward (B)|integer|800-1200|value * 1000|\n|127|Energy Calibration reverse (A)|integer|800-1200|value * 1000|\n|128|Energy Calibration reverse (B)|integer|800-1200|value * 1000|\n|129|Report Rate Control|integer|3-60|s|\n\nNote: (A) or (B) means channel A or channel B\n\nA user contributed module is available for this device in the [Contrib library](https://github.com/jasonacox/tinytuya/tree/master/tinytuya/Contrib):\n\n```python                                                                                                                                                                                                         from tinytuya.Contrib import WiFiDualMeterDevice\n\nwdm = WiFiDualMeterDevice.WiFiDualMeterDevice(\n    dev_id='abcdefghijklmnop123456',\n    address='192.168.0.29',      # Or set to 'Auto' to auto-discover IP address\n    local_key='1234567890123abc',\n    version=3.4)\n```\n\n### Tuya References\n\n* Tuya Hardware Development - Protocol: https://developer.tuya.com/en/docs/iot/device-development/embedded-software-development/mcu-development-access/wifi-mcu-sdk-solution/tuya-cloud-universal-serial-port-access-protocol?id=K9hhi0xxtn9cb\n* TuyaMCU listing of Tuya DP IDs: https://tasmota.github.io/docs/TuyaMCU/#switches-or-plugspower-strips\n\n## Credits\n\n  * TuyAPI https://github.com/codetheweb/tuyapi by codetheweb and blackrozes.\n    Protocol reverse engineering from jepsonrob and clach04.\n  * PyTuya https://github.com/clach04/python-tuya by clach04. \n    The origin of this python module (now abandoned). Thanks to nijave for pycryptodome support and testing, Exilit for unittests and docstrings, mike-gracia for improved Python version support, samuscherer for RGB Bulb support, magneticflux for improved Python version support, sean6541 for initial PyPi package and Home Assistant support <https://github.com/sean6541/tuya-homeassistant>, ziirish - for resolving a dependency problem related to version numbers at install time\n  * https://github.com/rospogrigio/localtuya-homeassistant by rospogrigio. \n    Updated pytuya to support devices with Device IDs of 22 characters\n  * Thanks to [@uzlonewolf](https://github.com/uzlonewolf), our top contributor and resident wizard, for expanding the Outlet/Cover/Bulb/Cloud modules into separate files, introducing Contrib structure for user generated device modules, making enhancements to TuyaMessage logic for multi-payload messages, rewriting the scanner and adding [Tuya Protocol 3.2, 3.4 & 3.5](https://github.com/jasonacox/tinytuya/discussions/260) support to TinyTuya!\n  * Finally, thanks to the entire TinyTuya community for the great engagement, contributions and encouragement! See [RELEASE notes](https://github.com/jasonacox/tinytuya/blob/master/RELEASE.md#release-notes) for the ever growing journal of improvements and the incredible list of talent making this project possible.\n\n## Related Projects\n\n  * https://github.com/sean6541/tuyaapi Python API to the web api\n  * https://github.com/codetheweb/tuyapi node.js\n  * https://github.com/Marcus-L/m4rcus.TuyaCore - .NET\n  * https://github.com/SDNick484/rectec_status/ - RecTec pellet smokers control (with Alexa skill)\n  * https://github.com/TradeFace/tuyaface - Python Async Tuya API\n\n## TinyTuya Powered Projects\n\nPlease feel free to submit a PR or open an issue to add your project.\n\n* https://github.com/mafrosis/tinytuya2mqtt - A bridge between TinyTuya and Home Assistant via MQTT\n* https://github.com/Whytey/pymoebot - A Python library intended to monitor and control the MoeBot robotic lawn mowers.\n* https://github.com/make-all/tuya-local - Local support for Tuya devices in Home Assistant\n* https://github.com/teejo75/mtghs - Provides an HTTP service for Moonraker to control Tuya outlets.\n* https://github.com/Xenomes/Domoticz-TinyTUYA-Plugin - Plugin for Domoticz Home Automation\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "Python module to interface with Tuya WiFi smart devices",
    "version": "1.16.0",
    "project_urls": {
        "Homepage": "https://github.com/jasonacox/tinytuya"
    },
    "split_keywords": [],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "13a0ebc447b775dbd7d9accbce48c10ea75d9dff7f69516dfd1ec55c4510b721",
                "md5": "c0f02802bf9f055a619ad1f13267537d",
                "sha256": "30c0e84e5982a637d63ea9c528e26953f280dd0c691f50191f6be9262c143543"
            },
            "downloads": -1,
            "filename": "tinytuya-1.16.0-py2.py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "c0f02802bf9f055a619ad1f13267537d",
            "packagetype": "bdist_wheel",
            "python_version": "py2.py3",
            "requires_python": null,
            "size": 134280,
            "upload_time": "2025-01-06T05:46:54",
            "upload_time_iso_8601": "2025-01-06T05:46:54.935361Z",
            "url": "https://files.pythonhosted.org/packages/13/a0/ebc447b775dbd7d9accbce48c10ea75d9dff7f69516dfd1ec55c4510b721/tinytuya-1.16.0-py2.py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2025-01-06 05:46:54",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "jasonacox",
    "github_project": "tinytuya",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "requirements": [
        {
            "name": "cryptography",
            "specs": [
                [
                    ">=",
                    "3.1"
                ]
            ]
        },
        {
            "name": "requests",
            "specs": []
        },
        {
            "name": "colorama",
            "specs": []
        }
    ],
    "lcname": "tinytuya"
}
        
Elapsed time: 0.50082s