Hakierspejs Łódź 🦄

Budujemy Hackerspace w Łodzi. Szukamy ludzi.

Freeing a Tuya mmWave motion sensor

23 lutego 2025

Tagi: DIY Smart home English

During the build of my open source smart home, the electricians - and then the interior contractors - made a couple small mistakes. One of them was installing an alarm wire in my boiler room that was just a couple centimeters too short - which was then covered by a suspended ceiling, messing up my plans to use 12/24V DC alarm wiring for all motion sensing.

Ceiling before the install
Believe it or not - this wasn't long enough!

Luckily, I’ve also made a second mistake, during the design of the lights in the room - equipped with absolutely zero information of how the equipment will be installed, I thought a second lamp above the equipment would be of any use. Surprise! Half of the room’s volume is piping, boilers, water tanks and so on - so I just have a lone 230V cable hanging from the ceiling. Perfect for powering a hard wired motion sensor!

Ceiling with all the equipment installed
I've never felt happier about a random live wire hanging from the ceiling.

Selecting the sensor, I had three requirements:

Despite having built custom mmWave ESPHome sensors (which will be covered in another post), I didn’t feel like putting a “custom” 230V to 5V power supply in the ceiling. After some research, I selected the $20-ish Tuya ZY-M100-5.8G series - it both satisfied the Zigbee support requirement (albeit with some caveats) and in theory it was easy to replace the Tuya Zigbee module with an ESP-12F.

Marketing photos of the Tuya motion sensor
What a cute little guy, I sure hope it doesn't break my Zigbee network!

After receiving the unit, I immediately opened it up to evaluate the second option - replacing the Zigbee module with an unlocked Wi-Fi one. Despite the apparent existence of some firmware fixing spammy Zigbee behaviour, I didn’t want to risk my 100+ devices becoming unreliable - and since I’m using ZHA, the ZY-M100 quirks were not implemented at the time I made the decision to flash it.

Motion sensor PCB sandwich
The nutritious PCB sandwich.

A quick Google search got me onto the ESPHome devices repository, where someone claimed that replacing the Tuya WiFi module with an ESP-12F-compatible chip (ESP32-C3 one per their writeup) would be trivial, other than having to connect EN to 3V3. I ended up buying the following items on Amazon:

For any further disassembly, I applied flux and leaded solder to the joints I wanted to remove, diluting the RoHS-compliant lead-free solder. This gets the melting temperature low enough that I can safely apply hot air to the module without damaging the rest of the components’ joints.

The 230V-5V PSU PCB is attached to the MCU board with four 2 pin headers:

I used the Engineer SS-02 solder sucker to clean the through holes. Despite removing most of the solder, I then had to use a tiny, flat screwdriver to detach the pins from the vias. Do not worry if the non-connected PSU vias split from the PCB’s substrate - there is no copper pour holding them in place, so they are very fragile. Make sure to not bend the pins when pulling the PCBs apart.

Tuya MCU PCB with the Wi-Fi module already removed
Oops, the Wi-Fi (not Zigbee?) module is already removed here.

The MCU board has a simple layout with a 5V to 3.3V LDO and what appears to be a reset circuit implemented through cutting off power. For any further testing, I replaced the PSU board with a lab power supply feeding 5V into DuPont female jumper cables, so as not to have to deal with 230V sitting on my test bench.

The stock WBR3 module… wasn’t a Zigbee one? Lovely. Yet another reason to replace it! Per the spec sheet:

There are no further requirements to get the module running. As you will soon see, this is most likely different to the module you’d want to replace it with.

Removing the Wi-Fi module is very simple with a hot air station. Apply thick flux, leaded solder to the pins to decrease their melting point, and heat up both rows of the castellated pins by alternating application of the hot air between them. Eventually you should be able to slide the module off the MCU board. Do not use any significant amounts of force, or you’ll strip the pads off the board. After you remove the module, use a solder wick to clean up the pads.

Afterwards, I used the breakout board, put the ESP-12F module against the pins and flashed the ESPHome firmware. I had to make a couple changes compared to the ESPHome device repository config - namely switch to the ESP8266 platform, disable UART logging and change the pins. Here’s my config:

substitutions:
  device_ssid: "Tuya Motion Sensor"
  device_name: boiler-motion-sensor
  device_description: "Tuya ZY-M100 Human Prescence Sensor ESP-12F ESP-IDF"
  friendly_name: "Boiler Motion Sensor"
  main_device_id: "boiler-motion-sensor" # Put the name that you want to see in Home Assistant.
  project_name: "tuya.zy-m100-wifi-esp-idf"
  project_version: "1.0"

esphome:
  name: ${device_name}
  comment: ${device_description}
  platformio_options:
    board_build.flash_mode: dio
  project:
    name: "${project_name}"
    version: "${project_version}"

esp8266:
  board: d1_mini

api:
  password: "your-ha-api-password"

logger:
  baud_rate: 0

web_server:
  port: 80

ota:
  - platform: esphome
    password: ""

wifi:
  ssid: "your-wifi-network"
  password: "your-wifi-password"
  power_save_mode: none
  ap:
    ssid: ${device_ssid}
    password: "your-fallback-ap-password"

uart:
  rx_pin: GPIO3
  tx_pin: GPIO1
  baud_rate: 115200

# Register the Tuya MCU connection
tuya:

sensor:
  # WiFi Signal sensor.
  - platform: wifi_signal
    name: ${friendly_name} Signal strength
    update_interval: 60s
    internal: true
  # Uptime Sensor
  - platform: uptime
    name: "${friendly_name} Uptime"
    id: uptime_sensor
    update_interval: 360s
    on_raw_value:
      then:
        - text_sensor.template.publish:
            id: uptime_human
            state: !lambda |-
              int seconds = round(id(uptime_sensor).raw_state);
              int days = seconds / (24 * 3600);
              seconds = seconds % (24 * 3600);
              int hours = seconds / 3600;
              seconds = seconds % 3600;
              int minutes = seconds /  60;
              seconds = seconds % 60;
              return (
                (days ? to_string(days) + "d " : "") +
                (hours ? to_string(hours) + "h " : "") +
                (minutes ? to_string(minutes) + "m " : "") +
                (to_string(seconds) + "s")
              ).c_str();
    # Light Sensor
  - platform: tuya
    name: "${friendly_name} Light Intensity"
    id: light_intensity
    sensor_datapoint: 104
    unit_of_measurement: "lux"
    icon: "mdi:brightness-5"
    device_class: "illuminance"
    state_class: "measurement"
    # Distance from Detected Object
  - platform: "tuya"
    name: "${friendly_name} Target Distance"
    id: target_distance
    sensor_datapoint: 9
    unit_of_measurement: "cm"
    icon: "mdi:eye"
    device_class: "distance"
    state_class: "measurement"

text_sensor:
  # Expose WiFi information as sensors.
  - platform: wifi_info
    ip_address:
      name: ${friendly_name} IP
    ssid:
      name: ${friendly_name} SSID
    bssid:
      name: ${friendly_name} BSSID
  # Expose Uptime
  - platform: template
    name: ${friendly_name} Uptime Human Readable
    id: uptime_human
    icon: mdi:clock-start

# Restart Buttons
button:
  - platform: restart
    id: "restart_device"
    name: "${friendly_name} Restart"
    entity_category: "diagnostic"
  - platform: safe_mode
    id: "restart_device_safe_mode"
    name: "${friendly_name} Restart (Safe Mode)"
    entity_category: "diagnostic"
number:
    # Sensitivity
  - platform: "tuya"
    name: "${friendly_name} Sensitivity"
    number_datapoint: 2
    min_value: 0
    max_value: 9
    step: 1
    icon: "mdi:ray-vertex"
    # Min Detection Distance
  - platform: "tuya"
    name: "${friendly_name} Near Detection"
    number_datapoint: 3
    min_value: 0
    max_value: 1000
    step: 1
    mode: slider
    unit_of_measurement: "cm"
    icon: "mdi:signal-distance-variant"
    # Max Detection Distance
  - platform: "tuya"
    name: "${friendly_name} Far Detection"
    number_datapoint: 4
    min_value: 0
    max_value: 1000
    step: 1
    mode: slider
    unit_of_measurement: "cm"
    icon: "mdi:signal-distance-variant"
    # Detection Delay
  - platform: "tuya"
    name: "${friendly_name} Detection Delay"
    number_datapoint: 101
    min_value: 0
    max_value: 100
    step: 1
    unit_of_measurement: "s"
    mode: slider
    icon: "mdi:clock"
    # Fading Time - Cool Down Period
  - platform: "tuya"
    name: "${friendly_name} Fading Time"
    number_datapoint: 102
    min_value: 0
    max_value: 1500
    step: 1
    unit_of_measurement: "s"
    mode: slider
    icon: "mdi:clock"

select:
    # Self Check Enum
  - platform: "tuya"
    name: "${friendly_name} Self Check Result"
    icon: mdi:eye
    enum_datapoint: 6
    options:
      0: Checking
      1: Check Success
      2: Check Failure
      3: Others
      4: Comm Fault
      5: Radar Fault

binary_sensor:
    # Status
  - platform: status
    name: "${friendly_name} Status"
    # Occupancy Binary Sensor
  - platform: "tuya"
    name: "${friendly_name} Presence State"
    sensor_datapoint: 1
    device_class: occupancy

I confirmed that the firmware was working by watching the DHCP IP tables in my router and connecting to the web UI. Afterwards I applied more flux to the MCU board and soldered the new module in place. This was trivial as the backing ground pour is not present in the 12F.

Having missed the note about connecting EN to VCC, I ran into an issue - the PCB still didn’t boot, despite technically having the same layout. Since the activity LED was not blinking, it was quite obvious that it wasn’t enabled. I went to the ESP-12F’s spec sheet and l found out the following tables:

By going by each pin from the boot mode table, probing it and analyzing the schematic we can conclude that:

I skipped TXD0, as it was connected to other circuitry on the Tuya MCU board - it worked fine without it. The PCB with all the required 10K resistors looks like this:

After connecting 5V and GND to the MCU PCB the module successfully booted! I reassembled the sensor, mounted it in the desired place, fine tuned it in the desired place.

Automating it in Home Assistant was as simple as adding the device and linking the light’s state to the presence state entity.

- id: 'motion_boiiler_room'
  alias: 'Motion in the boiler room turns on the ceiling light'
  description: ''
  use_blueprint:
    path: homeassistant/motion_light.yaml
    input:
      motion_entity: binary_sensor.boiler_motion_sensor_presence_state
      light_target: 
        entity_id:
          - light.boiler_room_light
      no_motion_wait: 5

Happy automating!

Author: @pzduniak