I work with many creative individuals who frequently come up with interesting projects, and occasionally, their ideas intersect with my own.

In my apartment, I’ve set up several Ubiquiti cameras along with a variety of Hue lights. This particular project began with a simple goal: to automatically enable camera recordings and turn off the lights when both my partner and I leave the apartment and to revert those actions when either of us returns.

Since I use Home Assistant, I experimented with several approaches before receiving a recommendation to try ESPresense, but more on that later. I tend to explore multiple solutions independently before seeking out advice, so let’s start by looking at the methods I initially attempted.

Unify WiFi Route

If you’re familiar with Home Assistant, you’ll know it supports integration with a wide range of devices and systems. One of those is Ubiquiti’s UniFi network equipment, which I happen to use specifically, UniFi access points that I had already integrated into Home Assistant.

With that in place, I created an automation that monitored whether our mobile devices were marked as seen or offline on the network. Initially, this setup worked as expected. However, I soon realized there was a noticeable delay in how quickly devices were marked as offline after leaving the apartment, and similarly, a lag before they were recognized as seen upon returning.

For example, I could walk nearly 50 meters away before my device was flagged as offline, and it might take up to a minute after re-entering the apartment for the automation to trigger depending on when the phone reconnected to WiFi. This made the solution less than ideal for my use case.

Apple iCloud Route

The next approach I tried was integrating Home Assistant with iCloud, since iCloud has access to the real-time location of our phones. While this method somewhat worked, it came with a few drawbacks. Home Assistant repeatedly triggered iCloud login attempts, which resulted in a constant stream of email alerts from Apple. Additionally, the location accuracy wasn’t particularly reliable, making this solution less effective than I had hoped.

Home Assistant App Route

Since I don’t expose Home Assistant to the public internet and didn’t want to rely on a VPN on both of our devices, I quickly ruled out this option.

ESPresense to the Rescue

Now for the fun part after laying out the backstory. A colleague of mine, who previously worked for an IoT company specializing in smart home solutions, shared what both he and his father use for presence detection in similar scenarios. Namely ESPresense.

ESPresense in itself

ESPresense is an ESP32-based presence detection node designed for localized device tracking.

Reasons to use this over other solutions:

  • ESP32 nodes are cheaper and easier to use than Raspberry Pis
  • Extensive fingerprint-based IDs instead of MAC addresses for tracking or counting devices others can’t
  • IRK-based enrollment of Apple devices to passively locate them uniquely, even with private random addresses
  • Home Assistant MQTT Discovery for easy HA configuration
  • Auto-updates by downloading GitHub-released binaries (optional, with a preference to disable if desired)
  • Filters measured distance with both a median pre-filter and a 1Euro filter (reduces jitter for greater accuracy)
  • Companion allows for full multilateration

XIAO ESP32C3

Following his recommendation, I ordered a pack of XIAO ESP32C3 boards from a local vendor. Given how affordable they are, I ended up purchasing 20 units.

Image Description

Here is some of the specs:

  • Flexible MCU Board: Incorporate the ESP32-C3 32-bit RISC-V chip, operating up to 160 MHz, mounted multiple development ports,
  • Developer Friendly: Compatible with Arduino IDE, MicroPython, CircuitPython, PlatformIO, ESP IDF, Zephyr, Matter, ESPNow, Meshtastic, WLED, ESPHome, Home Assistant, Ubidots
  • Outstanding RF performance: Complete Wi-Fi functions and Bluetooth Low Energy, while supporting communication over 100m with anFL antenna
  • Elaborate Power Design: 4 working modes as low as 44 μA in deep sleep mode, while supporting lithium battery charge management
  • Thumb-sized Design: 21 x 17.5mm, Seeed Studio XIAO series classic form factor
  • Perfect for Production: Breadboard-friendly with elegant Surface-Mounted SMD design, no components on the back

The most important feature of these boards is their support for Bluetooth Low Energy (BLE).

These compact devices are ideal for discreet placement, thanks to their small size. They also come equipped with an external antenna and 3M adhesive, making it easy to mount them wherever needed without drawing attention.

Image Description

Flashing ESPresense on the device

Installing the firmware on the device was quite easy affair as the website had a ESP Web Tool on it.

Image Description

For this kind of boars you only need to pick the latest version and select standard as flavor.

Image Description

The tool will prompt you to select the serial port to which the device is connected typically labeled as USB JTAG/serial debug unit. After that, you’ll be presented with two options: Install ESPresense or Logs & Console. In this case, we want to proceed with the installation.

Next, it will ask whether you’d like to erase the device before installing. Once confirmed, the tool will flash the ESPresense firmware onto the device.

Image Description

Image Description

Once the installation is complete, the device will automatically reboot and broadcast its own WiFi network for initial configuration. In my case, it appeared as espresense-e8df86.

Image Description

When you connect to the device, it will present a captive portal similar to what you might see when connecting to hotel WiFi for the first time. Here, you simply need to enter your WiFi SSID, WiFi password, and assign a room name to indicate where the device will be located. Once saved, restart the device to complete the setup.

Image Description

Image Description

It will now be available in your network for rest of the configuration later on.

Home Assistant setup

Head over to Home Assistant

MQTT

In Home Assistant you need to add MQTT and MQTT Room Presence integrations.

Image Description

Once the installation is complete, you’ll need to retrieve the MQTT password used by the integration. Since Home Assistant OS (HAOS) runs on Docker, and this integration operates within its own container, I inspected the container to extract the randomly generated MQTT password.

docker exec -it addon_core_mosquitto cat /data/system_user.json
{"homeassistant":{"password":"THIS IS A LONG F STRING"},"addons":{"password":"THIS IS NOT HE PASSWORD WE NEED, THE HOMEASSISTANT ONE IS"}}

Copy the password associated with homeassistant, we will be needing it later.

Configuring MQTT on ESPresence

Now we can connect to the ESPresence again over HTTP to do configurations to it. In my case it had the IP 192.168.1.173

Image Description

Next, you’ll need to configure the ESPresense device to connect to your Home Assistant instance. Enter the following details:

  • MQTT Server IP: 192.168.0.161 (your Home Assistant IP address)
  • MQTT Port: 1883
  • Username: homeassistant
  • Password: the one retrieved from the Docker container

Once that’s saved and the device connects, you’ll see it appear in Home Assistant under Devices, typically named something like espresense-hallway.

At this point, the device is successfully connected to Home Assistant, but it still doesn’t know what to detect or report on.

To configure this, go to the ESPresense device page in Home Assistant. At the bottom, you’ll find a link labeled “Click here to edit other settings!”. Clicking this opens the advanced ESPresense configuration interface.

Here, you can view detected devices and examine fingerprints which are the Bluetooth signatures of nearby devices the ESPresense unit can see.

Enroll bluetooth device

Under the Devices section in the ESPresense interface, there’s an Enroll button that allows you to register Bluetooth devices for tracking. However, during my setup, I had some issues on enrollment of our Apple devices while running ESPresense version 3.3.5.

Based on my research, there appears to be a known issue related to enrolling iPhones with S3/C3 boards, which affects consistent detection.

Fortunately, since I had access to a MacBook, I was able to manually retrieve the Identity Resolving Keys (IRKs) for both my iPhone and my partner’s by digging through the iCloud Keychain.

I later came across a comment on the GitHub issue suggesting an alternative approach: connecting the ESPresense device via USB and using the debug console to capture the IRK while attempting to enroll an iPhone. However, I haven’t tested this method myself, as the comment was posted after I had already completed my setup.

If you connect a serial monitoring application to the USB port of the Espresense device, during enrollment the irk:value is clearly displayed, though the alias never shows up on the fingerprints page.

Getting IRK from Keychain due to bug

Method for doing that is as follow’s:

  • On MacOS, ensure you are logged in with the iCloud ID associated with your device.
  • Launch the Keychain Access application.
  • In the left sidebar, click on iCloud.
  • In the upper right search bar, type bluetooth.
  • A series of entries will appear with the application password type. Image Description
  • On your Apple Watch device, go to Settings > About, scroll down to find the Bluetooth address, in the format: XX:XX:XX:XX:XX:XX
  • Open each entry to find the one associated with your Apple Watch. The Account field should match the Bluetooth address of your watch, in the format: Public: XX:XX:XX:XX:XX:XX.
  • Click on Show passwordType your macOS password twice and copy the contents. Image Description

The output will contain a Base64-encoded string, which needs to be decoded to extract the Identity Resolving Key (IRK).

Example of a Base64-encoded string: WWVzIHRoaXMgaXMgYSBiYXNlNjQgZW5jb2RlZCBzdHJpbmc=

Decode on Linux/Mac:

echo "WWVzIHRoaXMgaXMgYSBiYXNlNjQgZW5jb2RlZCBzdHJpbmc=" | base64 --decode

Decode on Windows:

[System.Text.Encoding]::UTF8.GetString([System.Convert]::FromBase64String("WWVzIHRoaXMgaXMgYSBiYXNlNjQgZW5jb2RlZCBzdHJpbmc="))

Once decoded, you should have a 32-character IRK. Be sure to save this key for future use.

For the purposes of the examples that follow, I’ll use the placeholder xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx to represent the IRK.

Continue with the configuration in Home Assistant

In Home Assistant, navigate to Settings → Devices & Services → MQTT, then click Configure under core-mosquitto.

From there, use the Publish a packet section to send configuration data directly to your ESPresense device. This feature is especially useful when managing multiple ESPresense units across different rooms, allowing you to centrally configure and update them as needed.

Under Topic you add the following and replace xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx with your IRK:

espresense/settings/irk:xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx/config

And with the payload you will need to add a id that we will be referencing later and a name:

{"id":"irk:mikael_ble", "name":"Mikael BLE"}

Once you have done that hit Publish.

Once the configuration has been successfully published, you’ll see the new entry appear at the top of the ESPresense device’s settings page, as well as under Devices.

Next, we need to create a sensor in Home Assistant that can receive and reflect updates from ESPresense. To do this, open your configuration.yaml file and add the following snippet of code:

sensor:
  - platform: "mqtt_room"
    device_id: "irk:mikael_ble"
    name: "Mikael iPhone BLE"
    state_topic: "espresense/devices/irk:mikael_ble"
    away_timeout: 60
    timeout: 10
    unique_id: "mikael_iphone_BLE"

What we’re doing here is creating a sensor to store presence data, using the reference we specified in the Publish command. We’re assigning it a human-readable name and a unique_id so it can be easily identified and managed within Home Assistant.

Next, we’ll create an automation to convert this sensor data into device_tracker entities. These will report values of either home or not_home, depending on whether the device is currently detected (seen) or not.

alias: ESPresense - Convert to device_tracker
description: Convert ESPresence sensors to device_trackers
triggers:
  - entity_id:
      - sensor.mikael_iphone_ble
    trigger: state
conditions: []
actions:
  - variables:
      dev_id: "{{ trigger.to_state.entity_id.split('.')[1] }}"
      previous_state: "{{ states('device_tracker.' ~ dev_id) }}"
  - data:
      dev_id: "{{ dev_id }}"
      source_type: bluetooth_le
      location_name: |-
        {% if not is_state('sensor.' ~ dev_id, 'not_home') %}
          home
        {% else %}
          not_home
        {% endif %}
    action: device_tracker.see
mode: parallel

Since this automation can be quite active and generate frequent state changes, it may end up cluttering the logs. To avoid this, we can return to the configuration.yaml file and adjust the logging settings to suppress unnecessary log entries.

recorder:
  exclude:
    entities:
      - automation.espresense_convert_to_device_tracker

Since we’ve made changes to the configuration.yaml file, we need to reload the configuration. To do this, go to Settings → System, then click the power icon in the top-right corner and select Quick reload. This will apply the changes without requiring a full Home Assistant restart.

The final step is to link the newly created device_tracker to a person in Home Assistant. Navigate to Settings → People, select the appropriate person, and under Select the devices that belong to this person, add the device tracker for example, mikael_iphone_ble.

Image Description

Image Description

Image Description

Success

Thanks to its use of Bluetooth Low Energy (BLE), ESPresense is capable of detecting presence with millisecond-level responsiveness allowing near-instant recognition when a device enters or leaves the area.

Below is an example of the automation I created to trigger actions based on whether we’re home or away:

alias: Nobody Home
description: ""
triggers:
  - trigger: state
    entity_id:
      - zone.home
    from: null
    to: "0"
conditions: []
actions:
  - action: input_boolean.turn_on
    metadata: {}
    data: {}
    target:
      entity_id: input_boolean.camera_record
  - action: light.turn_off
    metadata: {}
    data: {}
    target:
      area_id:
        - living_room
        - bedroom
        - guest_room
      device_id:
        - 6622ef2f97a996d30c75cab328520f93
        - cffb87d6e88692ecbe7f554fc4a2e89a
mode: single
alias: Someone Arrived Home
description: ""
triggers:
  - trigger: state
    entity_id:
      - zone.home
    from: "0"
    to: null
conditions: []
actions:
  - action: input_boolean.turn_off
    metadata: {}
    data: {}
    target:
      entity_id: input_boolean.camera_record
  - action: light.turn_on
    metadata: {}
    data:
      brightness_pct: 50
      kelvin: 2652
    target:
      area_id:
        - living_room
        - kitchen
mode: single

By placing an ESPresense device in the bedroom, you can also detect when someone has gone to bed, enabling additional automation based on presence in that specific room.

alias: Detect that Mikael and The better half has gone to bed!
description: ""
triggers:
  - trigger: state
    entity_id:
      - sensor.wife_iphone_ble
    to: bedroom
    for:
      hours: 0
      minutes: 1
      seconds: 0
    alias: When The better half changed to Bedroom for 1 min
  - alias: When Mikael changed to Bedroom for 1 min
    trigger: state
    entity_id:
      - sensor.mikael_iphone_ble
    to: bedroom
    for:
      hours: 0
      minutes: 1
      seconds: 0
conditions:
  - alias: If The better half and Mikael is in the bedroom and time is between 23:00 and 08:00
    condition: and
    conditions:
      - condition: state
        entity_id: sensor.wife_iphone_ble
        state: bedroom
      - condition: state
        entity_id: sensor.mikael_iphone_ble
        state: bedroom
      - condition: time
        after: "23:00:00"
        before: "08:00:00"
actions:
  - action: light.turn_off
    metadata: {}
    data: {}
    target:
      area_id:
        - living_room
      device_id:
        - 6622ef2f97a996d30c75cab328520f93
        - cffb87d6e88692ecbe7f554fc4a2e89a
    alias: Turn off all Light except under bench in kitchen
  - type: turn_on
    device_id: 73218151408795c2dcc5f7ba6c65aa2e
    entity_id: 17bae889032cfd2c7ed14c7d8862056e
    domain: light
    brightness_pct: 10
    alias: Set Benklight 1 to 10%
  - type: turn_on
    device_id: e5671e11169722cd9539ce8eef105b82
    entity_id: fdb242e9d98cd86edba64edd7e4b84d0
    domain: light
    brightness_pct: 10
    alias: Set Benklight 2 to 10%
mode: single

As a bonus, if you need a 3D model for printing of a case for the device I created one and uploaded to Makerworld

Image Description

Bugs

Strange behavior on Apple Device tracking

For Bluetooth Low Energy (BLE) presence detection to remain active on an iPhone—even when the screen is off—you typically need to have a companion device that supports Apple’s Continuity features, such as an Apple Watch. Without such a device, iOS tends to suspend BLE advertising shortly after the screen turns off.

This limitation has been a known issue for some time. I first noticed the behavior when my partner stopped using her Apple Watch due to battery issues, which led to inconsistent presence detection and unexpected automation triggers in Home Assistant.

You can find more information on this issue here.

Enrollment issues with apple devices and ESP32S3 / ESP32C3

Based on my research, there appears to be a known issue related to enrolling iPhones with S3/C3 boards, which affects consistent detection.

Fortunately, since I had access to a MacBook, I was able to manually retrieve the Identity Resolving Keys (IRKs) for both my iPhone and my partner’s by digging through the iCloud Keychain.

I later came across a comment on the GitHub issue suggesting an alternative approach: connecting the ESPresense device via USB and using the debug console to capture the IRK while attempting to enroll an iPhone. However, I haven’t tested this method myself, as the comment was posted after I had already completed my setup.

I wrote about it in-dept on this page.