mirror of
				https://github.com/esphome/esphome.git
				synced 2025-11-03 08:31:47 +00:00 
			
		
		
		
	Compare commits
	
		
			100 Commits
		
	
	
		
			modbus_hea
			...
			2025.7.0b3
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					16292a9f13 | ||
| 
						 | 
					90f0ebb22b | ||
| 
						 | 
					4153380f99 | ||
| 
						 | 
					740c0ef9d7 | ||
| 
						 | 
					b4521e1d8c | ||
| 
						 | 
					10ca7ed85b | ||
| 
						 | 
					e43efdaaec | ||
| 
						 | 
					9207bf97f3 | ||
| 
						 | 
					c13317f807 | ||
| 
						 | 
					77d1d0414d | ||
| 
						 | 
					8f42bc6aac | ||
| 
						 | 
					9beb4e2cd4 | ||
| 
						 | 
					097aac2183 | ||
| 
						 | 
					18787b0be0 | ||
| 
						 | 
					39e01c42e1 | ||
| 
						 | 
					c760f89e46 | ||
| 
						 | 
					01b4e214b9 | ||
| 
						 | 
					bc7cfeb9cd | ||
| 
						 | 
					36dd203e74 | ||
| 
						 | 
					8605994cc6 | ||
| 
						 | 
					80fbe28088 | ||
| 
						 | 
					1d9f17a57c | ||
| 
						 | 
					42947bcf56 | ||
| 
						 | 
					3c864b2bca | ||
| 
						 | 
					35d88fc0d6 | ||
| 
						 | 
					7a6894e087 | ||
| 
						 | 
					1b222ceca3 | ||
| 
						 | 
					bab3deee1b | ||
| 
						 | 
					ccd30110b1 | ||
| 
						 | 
					904c7b8a3a | ||
| 
						 | 
					fa262673e4 | ||
| 
						 | 
					0ef5f1fd65 | ||
| 
						 | 
					23dd2d648e | ||
| 
						 | 
					5ba493acc3 | ||
| 
						 | 
					a5055094d0 | ||
| 
						 | 
					92d03dd196 | ||
| 
						 | 
					bd75f0dfea | ||
| 
						 | 
					6178ab7513 | ||
| 
						 | 
					267574f24c | ||
| 
						 | 
					5235c80781 | ||
| 
						 | 
					0ccc5e340e | ||
| 
						 | 
					86c6e4da2a | ||
| 
						 | 
					5c8b330eaa | ||
| 
						 | 
					4158a5c2a3 | ||
| 
						 | 
					05c5364490 | ||
| 
						 | 
					78eb236a4a | ||
| 
						 | 
					691cc5f7dc | ||
| 
						 | 
					b3d7f001af | ||
| 
						 | 
					3f8b691c32 | ||
| 
						 | 
					a30f01d668 | ||
| 
						 | 
					4648804db6 | ||
| 
						 | 
					51377b2625 | ||
| 
						 | 
					256f9f9943 | ||
| 
						 | 
					a72905191a | ||
| 
						 | 
					7150f2806f | ||
| 
						 | 
					ee8ee4e646 | ||
| 
						 | 
					fb357b8965 | ||
| 
						 | 
					c4fac1a2ae | ||
| 
						 | 
					42a1f6922f | ||
| 
						 | 
					206659ddb8 | ||
| 
						 | 
					440de12e3f | ||
| 
						 | 
					b122112d58 | ||
| 
						 | 
					fe258e1007 | ||
| 
						 | 
					3976fd02ea | ||
| 
						 | 
					e58c793da2 | ||
| 
						 | 
					90fb3680d4 | ||
| 
						 | 
					832a787271 | ||
| 
						 | 
					29747fc730 | ||
| 
						 | 
					e2de6ee29d | ||
| 
						 | 
					053feb5e3b | ||
| 
						 | 
					31f36df4ba | ||
| 
						 | 
					3ef392d433 | ||
| 
						 | 
					138ff749f3 | ||
| 
						 | 
					e88b8d10ec | ||
| 
						 | 
					8147d117a0 | ||
| 
						 | 
					c6f7e84256 | ||
| 
						 | 
					db877e688a | ||
| 
						 | 
					4e25b6da7b | ||
| 
						 | 
					83512b88c4 | ||
| 
						 | 
					fde5f88192 | ||
| 
						 | 
					2510b5ffb5 | ||
| 
						 | 
					364b6ca8d0 | ||
| 
						 | 
					e49b89a051 | ||
| 
						 | 
					bdd52dbaa4 | ||
| 
						 | 
					765793505d | ||
| 
						 | 
					a303f93236 | ||
| 
						 | 
					492580edc3 | ||
| 
						 | 
					1368139f4d | ||
| 
						 | 
					b6fade7339 | ||
| 
						 | 
					8da322fe9e | ||
| 
						 | 
					e5a699a004 | ||
| 
						 | 
					e061b6dc55 | ||
| 
						 | 
					4673a5b48c | ||
| 
						 | 
					0bc18a8281 | ||
| 
						 | 
					20ba035e3b | ||
| 
						 | 
					f7019a4ed7 | ||
| 
						 | 
					a1291c2730 | ||
| 
						 | 
					b0f8922056 | ||
| 
						 | 
					4e9e48e2e7 | ||
| 
						 | 
					86e7013f40 | 
							
								
								
									
										39
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										39
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							@@ -214,17 +214,51 @@ jobs:
 | 
			
		||||
        if: matrix.os == 'windows-latest'
 | 
			
		||||
        run: |
 | 
			
		||||
          ./venv/Scripts/activate
 | 
			
		||||
          pytest -vv --cov-report=xml --tb=native -n auto tests
 | 
			
		||||
          pytest -vv --cov-report=xml --tb=native -n auto tests --ignore=tests/integration/
 | 
			
		||||
      - name: Run pytest
 | 
			
		||||
        if: matrix.os == 'ubuntu-latest' || matrix.os == 'macOS-latest'
 | 
			
		||||
        run: |
 | 
			
		||||
          . venv/bin/activate
 | 
			
		||||
          pytest -vv --cov-report=xml --tb=native -n auto tests
 | 
			
		||||
          pytest -vv --cov-report=xml --tb=native -n auto tests --ignore=tests/integration/
 | 
			
		||||
      - name: Upload coverage to Codecov
 | 
			
		||||
        uses: codecov/codecov-action@v5.4.3
 | 
			
		||||
        with:
 | 
			
		||||
          token: ${{ secrets.CODECOV_TOKEN }}
 | 
			
		||||
 | 
			
		||||
  integration-tests:
 | 
			
		||||
    name: Run integration tests
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    needs:
 | 
			
		||||
      - common
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Check out code from GitHub
 | 
			
		||||
        uses: actions/checkout@v4.2.2
 | 
			
		||||
      - name: Set up Python 3.13
 | 
			
		||||
        id: python
 | 
			
		||||
        uses: actions/setup-python@v5.6.0
 | 
			
		||||
        with:
 | 
			
		||||
          python-version: "3.13"
 | 
			
		||||
      - name: Restore Python virtual environment
 | 
			
		||||
        id: cache-venv
 | 
			
		||||
        uses: actions/cache@v4.2.3
 | 
			
		||||
        with:
 | 
			
		||||
          path: venv
 | 
			
		||||
          key: ${{ runner.os }}-${{ steps.python.outputs.python-version }}-venv-${{ needs.common.outputs.cache-key }}
 | 
			
		||||
      - name: Create Python virtual environment
 | 
			
		||||
        if: steps.cache-venv.outputs.cache-hit != 'true'
 | 
			
		||||
        run: |
 | 
			
		||||
          python -m venv venv
 | 
			
		||||
          . venv/bin/activate
 | 
			
		||||
          python --version
 | 
			
		||||
          pip install -r requirements.txt -r requirements_test.txt
 | 
			
		||||
          pip install -e .
 | 
			
		||||
      - name: Register matcher
 | 
			
		||||
        run: echo "::add-matcher::.github/workflows/matchers/pytest.json"
 | 
			
		||||
      - name: Run integration tests
 | 
			
		||||
        run: |
 | 
			
		||||
          . venv/bin/activate
 | 
			
		||||
          pytest -vv --no-cov --tb=native -n auto tests/integration/
 | 
			
		||||
 | 
			
		||||
  clang-format:
 | 
			
		||||
    name: Check clang-format
 | 
			
		||||
    runs-on: ubuntu-24.04
 | 
			
		||||
@@ -494,6 +528,7 @@ jobs:
 | 
			
		||||
      - flake8
 | 
			
		||||
      - pylint
 | 
			
		||||
      - pytest
 | 
			
		||||
      - integration-tests
 | 
			
		||||
      - pyupgrade
 | 
			
		||||
      - clang-tidy
 | 
			
		||||
      - list-components
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,14 @@
 | 
			
		||||
---
 | 
			
		||||
# See https://pre-commit.com for more information
 | 
			
		||||
# See https://pre-commit.com/hooks.html for more hooks
 | 
			
		||||
 | 
			
		||||
ci:
 | 
			
		||||
  autoupdate_commit_msg: 'pre-commit: autoupdate'
 | 
			
		||||
  autoupdate_schedule: weekly
 | 
			
		||||
  autofix_prs: false
 | 
			
		||||
  # Skip hooks that have issues in pre-commit CI environment
 | 
			
		||||
  skip: [pylint, yamllint]
 | 
			
		||||
 | 
			
		||||
repos:
 | 
			
		||||
  - repo: https://github.com/astral-sh/ruff-pre-commit
 | 
			
		||||
    # Ruff version.
 | 
			
		||||
 
 | 
			
		||||
@@ -87,6 +87,7 @@ esphome/components/bp1658cj/* @Cossid
 | 
			
		||||
esphome/components/bp5758d/* @Cossid
 | 
			
		||||
esphome/components/button/* @esphome/core
 | 
			
		||||
esphome/components/bytebuffer/* @clydebarrow
 | 
			
		||||
esphome/components/camera/* @DT-art1 @bdraco
 | 
			
		||||
esphome/components/canbus/* @danielschramm @mvturnho
 | 
			
		||||
esphome/components/cap1188/* @mreditor97
 | 
			
		||||
esphome/components/captive_portal/* @OttoWinter
 | 
			
		||||
@@ -169,6 +170,7 @@ esphome/components/ft5x06/* @clydebarrow
 | 
			
		||||
esphome/components/ft63x6/* @gpambrozio
 | 
			
		||||
esphome/components/gcja5/* @gcormier
 | 
			
		||||
esphome/components/gdk101/* @Szewcson
 | 
			
		||||
esphome/components/gl_r01_i2c/* @pkejval
 | 
			
		||||
esphome/components/globals/* @esphome/core
 | 
			
		||||
esphome/components/gp2y1010au0f/* @zry98
 | 
			
		||||
esphome/components/gp8403/* @jesserockz
 | 
			
		||||
@@ -253,6 +255,7 @@ esphome/components/ln882x/* @lamauny
 | 
			
		||||
esphome/components/lock/* @esphome/core
 | 
			
		||||
esphome/components/logger/* @esphome/core
 | 
			
		||||
esphome/components/logger/select/* @clydebarrow
 | 
			
		||||
esphome/components/lps22/* @nagisa
 | 
			
		||||
esphome/components/ltr390/* @latonita @sjtrny
 | 
			
		||||
esphome/components/ltr501/* @latonita
 | 
			
		||||
esphome/components/ltr_als_ps/* @latonita
 | 
			
		||||
@@ -441,6 +444,7 @@ esphome/components/sun/* @OttoWinter
 | 
			
		||||
esphome/components/sun_gtil2/* @Mat931
 | 
			
		||||
esphome/components/switch/* @esphome/core
 | 
			
		||||
esphome/components/switch/binary_sensor/* @ssieb
 | 
			
		||||
esphome/components/sx126x/* @swoboda1337
 | 
			
		||||
esphome/components/sx127x/* @swoboda1337
 | 
			
		||||
esphome/components/syslog/* @clydebarrow
 | 
			
		||||
esphome/components/t6615/* @tylermenezes
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										2
									
								
								Doxyfile
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								Doxyfile
									
									
									
									
									
								
							@@ -48,7 +48,7 @@ PROJECT_NAME           = ESPHome
 | 
			
		||||
# could be handy for archiving the generated documentation or if some version
 | 
			
		||||
# control system is used.
 | 
			
		||||
 | 
			
		||||
PROJECT_NUMBER         = 2025.7.0-dev
 | 
			
		||||
PROJECT_NUMBER         = 2025.7.0b3
 | 
			
		||||
 | 
			
		||||
# Using the PROJECT_BRIEF tag one can provide an optional one line description
 | 
			
		||||
# for a project that appears at the top of each page and should give viewer a
 | 
			
		||||
 
 | 
			
		||||
@@ -10,8 +10,15 @@ from esphome.components.esp32.const import (
 | 
			
		||||
    VARIANT_ESP32S2,
 | 
			
		||||
    VARIANT_ESP32S3,
 | 
			
		||||
)
 | 
			
		||||
from esphome.config_helpers import filter_source_files_from_platform
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import CONF_ANALOG, CONF_INPUT, CONF_NUMBER, PLATFORM_ESP8266
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_ANALOG,
 | 
			
		||||
    CONF_INPUT,
 | 
			
		||||
    CONF_NUMBER,
 | 
			
		||||
    PLATFORM_ESP8266,
 | 
			
		||||
    PlatformFramework,
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import CORE
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@esphome/core"]
 | 
			
		||||
@@ -229,3 +236,20 @@ def validate_adc_pin(value):
 | 
			
		||||
        )(value)
 | 
			
		||||
 | 
			
		||||
    raise NotImplementedError
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FILTER_SOURCE_FILES = filter_source_files_from_platform(
 | 
			
		||||
    {
 | 
			
		||||
        "adc_sensor_esp32.cpp": {
 | 
			
		||||
            PlatformFramework.ESP32_ARDUINO,
 | 
			
		||||
            PlatformFramework.ESP32_IDF,
 | 
			
		||||
        },
 | 
			
		||||
        "adc_sensor_esp8266.cpp": {PlatformFramework.ESP8266_ARDUINO},
 | 
			
		||||
        "adc_sensor_rp2040.cpp": {PlatformFramework.RP2040_ARDUINO},
 | 
			
		||||
        "adc_sensor_libretiny.cpp": {
 | 
			
		||||
            PlatformFramework.BK72XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.RTL87XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.LN882X_ARDUINO,
 | 
			
		||||
        },
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
 
 | 
			
		||||
@@ -23,7 +23,7 @@ void APDS9960::setup() {
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (id != 0xAB && id != 0x9C && id != 0xA8) {  // APDS9960 all should have one of these IDs
 | 
			
		||||
  if (id != 0xAB && id != 0x9C && id != 0xA8 && id != 0x9E) {  // APDS9960 all should have one of these IDs
 | 
			
		||||
    this->error_code_ = WRONG_ID;
 | 
			
		||||
    this->mark_failed();
 | 
			
		||||
    return;
 | 
			
		||||
 
 | 
			
		||||
@@ -3,6 +3,7 @@ import base64
 | 
			
		||||
from esphome import automation
 | 
			
		||||
from esphome.automation import Condition
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.config_helpers import get_logger_level
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_ACTION,
 | 
			
		||||
@@ -23,8 +24,9 @@ from esphome.const import (
 | 
			
		||||
    CONF_TRIGGER_ID,
 | 
			
		||||
    CONF_VARIABLES,
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import coroutine_with_priority
 | 
			
		||||
from esphome.core import CORE, coroutine_with_priority
 | 
			
		||||
 | 
			
		||||
DOMAIN = "api"
 | 
			
		||||
DEPENDENCIES = ["network"]
 | 
			
		||||
AUTO_LOAD = ["socket"]
 | 
			
		||||
CODEOWNERS = ["@OttoWinter"]
 | 
			
		||||
@@ -50,6 +52,7 @@ SERVICE_ARG_NATIVE_TYPES = {
 | 
			
		||||
}
 | 
			
		||||
CONF_ENCRYPTION = "encryption"
 | 
			
		||||
CONF_BATCH_DELAY = "batch_delay"
 | 
			
		||||
CONF_CUSTOM_SERVICES = "custom_services"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def validate_encryption_key(value):
 | 
			
		||||
@@ -114,6 +117,7 @@ CONFIG_SCHEMA = cv.All(
 | 
			
		||||
                cv.positive_time_period_milliseconds,
 | 
			
		||||
                cv.Range(max=cv.TimePeriod(milliseconds=65535)),
 | 
			
		||||
            ),
 | 
			
		||||
            cv.Optional(CONF_CUSTOM_SERVICES, default=False): cv.boolean,
 | 
			
		||||
            cv.Optional(CONF_ON_CLIENT_CONNECTED): automation.validate_automation(
 | 
			
		||||
                single=True
 | 
			
		||||
            ),
 | 
			
		||||
@@ -138,8 +142,11 @@ async def to_code(config):
 | 
			
		||||
    cg.add(var.set_reboot_timeout(config[CONF_REBOOT_TIMEOUT]))
 | 
			
		||||
    cg.add(var.set_batch_delay(config[CONF_BATCH_DELAY]))
 | 
			
		||||
 | 
			
		||||
    # Set USE_API_SERVICES if any services are enabled
 | 
			
		||||
    if config.get(CONF_ACTIONS) or config[CONF_CUSTOM_SERVICES]:
 | 
			
		||||
        cg.add_define("USE_API_SERVICES")
 | 
			
		||||
 | 
			
		||||
    if actions := config.get(CONF_ACTIONS, []):
 | 
			
		||||
        cg.add_define("USE_API_YAML_SERVICES")
 | 
			
		||||
        for conf in actions:
 | 
			
		||||
            template_args = []
 | 
			
		||||
            func_args = []
 | 
			
		||||
@@ -313,3 +320,25 @@ async def homeassistant_tag_scanned_to_code(config, action_id, template_arg, arg
 | 
			
		||||
@automation.register_condition("api.connected", APIConnectedCondition, {})
 | 
			
		||||
async def api_connected_to_code(config, condition_id, template_arg, args):
 | 
			
		||||
    return cg.new_Pvariable(condition_id, template_arg)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def FILTER_SOURCE_FILES() -> list[str]:
 | 
			
		||||
    """Filter out api_pb2_dump.cpp when proto message dumping is not enabled
 | 
			
		||||
    and user_services.cpp when no services are defined."""
 | 
			
		||||
    files_to_filter = []
 | 
			
		||||
 | 
			
		||||
    # api_pb2_dump.cpp is only needed when HAS_PROTO_MESSAGE_DUMP is defined
 | 
			
		||||
    # This is a particularly large file that still needs to be opened and read
 | 
			
		||||
    # all the way to the end even when ifdef'd out
 | 
			
		||||
    #
 | 
			
		||||
    # HAS_PROTO_MESSAGE_DUMP is defined when ESPHOME_LOG_HAS_VERY_VERBOSE is set,
 | 
			
		||||
    # which happens when the logger level is VERY_VERBOSE
 | 
			
		||||
    if get_logger_level() != "VERY_VERBOSE":
 | 
			
		||||
        files_to_filter.append("api_pb2_dump.cpp")
 | 
			
		||||
 | 
			
		||||
    # user_services.cpp is only needed when services are defined
 | 
			
		||||
    config = CORE.config.get(DOMAIN, {})
 | 
			
		||||
    if config and not config.get(CONF_ACTIONS) and not config[CONF_CUSTOM_SERVICES]:
 | 
			
		||||
        files_to_filter.append("user_services.cpp")
 | 
			
		||||
 | 
			
		||||
    return files_to_filter
 | 
			
		||||
 
 | 
			
		||||
@@ -374,6 +374,7 @@ message CoverCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_COVER";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
 | 
			
		||||
@@ -387,6 +388,7 @@ message CoverCommandRequest {
 | 
			
		||||
  bool has_tilt = 6;
 | 
			
		||||
  float tilt = 7;
 | 
			
		||||
  bool stop = 8;
 | 
			
		||||
  uint32 device_id = 9;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== FAN ====================
 | 
			
		||||
@@ -441,6 +443,7 @@ message FanCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_FAN";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  bool has_state = 2;
 | 
			
		||||
@@ -455,6 +458,7 @@ message FanCommandRequest {
 | 
			
		||||
  int32 speed_level = 11;
 | 
			
		||||
  bool has_preset_mode = 12;
 | 
			
		||||
  string preset_mode = 13;
 | 
			
		||||
  uint32 device_id = 14;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== LIGHT ====================
 | 
			
		||||
@@ -523,6 +527,7 @@ message LightCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_LIGHT";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  bool has_state = 2;
 | 
			
		||||
@@ -551,6 +556,7 @@ message LightCommandRequest {
 | 
			
		||||
  uint32 flash_length = 17;
 | 
			
		||||
  bool has_effect = 18;
 | 
			
		||||
  string effect = 19;
 | 
			
		||||
  uint32 device_id = 28;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== SENSOR ====================
 | 
			
		||||
@@ -640,9 +646,11 @@ message SwitchCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_SWITCH";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  bool state = 2;
 | 
			
		||||
  uint32 device_id = 3;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== TEXT SENSOR ====================
 | 
			
		||||
@@ -799,18 +807,21 @@ enum ServiceArgType {
 | 
			
		||||
  SERVICE_ARG_TYPE_STRING_ARRAY = 7;
 | 
			
		||||
}
 | 
			
		||||
message ListEntitiesServicesArgument {
 | 
			
		||||
  option (ifdef) = "USE_API_SERVICES";
 | 
			
		||||
  string name = 1;
 | 
			
		||||
  ServiceArgType type = 2;
 | 
			
		||||
}
 | 
			
		||||
message ListEntitiesServicesResponse {
 | 
			
		||||
  option (id) = 41;
 | 
			
		||||
  option (source) = SOURCE_SERVER;
 | 
			
		||||
  option (ifdef) = "USE_API_SERVICES";
 | 
			
		||||
 | 
			
		||||
  string name = 1;
 | 
			
		||||
  fixed32 key = 2;
 | 
			
		||||
  repeated ListEntitiesServicesArgument args = 3;
 | 
			
		||||
}
 | 
			
		||||
message ExecuteServiceArgument {
 | 
			
		||||
  option (ifdef) = "USE_API_SERVICES";
 | 
			
		||||
  bool bool_ = 1;
 | 
			
		||||
  int32 legacy_int = 2;
 | 
			
		||||
  float float_ = 3;
 | 
			
		||||
@@ -826,6 +837,7 @@ message ExecuteServiceRequest {
 | 
			
		||||
  option (id) = 42;
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (ifdef) = "USE_API_SERVICES";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  repeated ExecuteServiceArgument args = 2;
 | 
			
		||||
@@ -836,7 +848,7 @@ message ListEntitiesCameraResponse {
 | 
			
		||||
  option (id) = 43;
 | 
			
		||||
  option (base_class) = "InfoResponseProtoMessage";
 | 
			
		||||
  option (source) = SOURCE_SERVER;
 | 
			
		||||
  option (ifdef) = "USE_ESP32_CAMERA";
 | 
			
		||||
  option (ifdef) = "USE_CAMERA";
 | 
			
		||||
 | 
			
		||||
  string object_id = 1;
 | 
			
		||||
  fixed32 key = 2;
 | 
			
		||||
@@ -850,17 +862,19 @@ message ListEntitiesCameraResponse {
 | 
			
		||||
 | 
			
		||||
message CameraImageResponse {
 | 
			
		||||
  option (id) = 44;
 | 
			
		||||
  option (base_class) = "StateResponseProtoMessage";
 | 
			
		||||
  option (source) = SOURCE_SERVER;
 | 
			
		||||
  option (ifdef) = "USE_ESP32_CAMERA";
 | 
			
		||||
  option (ifdef) = "USE_CAMERA";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  bytes data = 2;
 | 
			
		||||
  bool done = 3;
 | 
			
		||||
  uint32 device_id = 4;
 | 
			
		||||
}
 | 
			
		||||
message CameraImageRequest {
 | 
			
		||||
  option (id) = 45;
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_ESP32_CAMERA";
 | 
			
		||||
  option (ifdef) = "USE_CAMERA";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
 | 
			
		||||
  bool single = 1;
 | 
			
		||||
@@ -980,6 +994,7 @@ message ClimateCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_CLIMATE";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  bool has_mode = 2;
 | 
			
		||||
@@ -1005,6 +1020,7 @@ message ClimateCommandRequest {
 | 
			
		||||
  string custom_preset = 21;
 | 
			
		||||
  bool has_target_humidity = 22;
 | 
			
		||||
  float target_humidity = 23;
 | 
			
		||||
  uint32 device_id = 24;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== NUMBER ====================
 | 
			
		||||
@@ -1054,9 +1070,11 @@ message NumberCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_NUMBER";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  float state = 2;
 | 
			
		||||
  uint32 device_id = 3;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== SELECT ====================
 | 
			
		||||
@@ -1096,9 +1114,11 @@ message SelectCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_SELECT";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  string state = 2;
 | 
			
		||||
  uint32 device_id = 3;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== SIREN ====================
 | 
			
		||||
@@ -1137,6 +1157,7 @@ message SirenCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_SIREN";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  bool has_state = 2;
 | 
			
		||||
@@ -1147,6 +1168,7 @@ message SirenCommandRequest {
 | 
			
		||||
  uint32 duration = 7;
 | 
			
		||||
  bool has_volume = 8;
 | 
			
		||||
  float volume = 9;
 | 
			
		||||
  uint32 device_id = 10;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== LOCK ====================
 | 
			
		||||
@@ -1201,12 +1223,14 @@ message LockCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_LOCK";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  LockCommand command = 2;
 | 
			
		||||
 | 
			
		||||
  // Not yet implemented:
 | 
			
		||||
  bool has_code = 3;
 | 
			
		||||
  string code = 4;
 | 
			
		||||
  uint32 device_id = 5;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== BUTTON ====================
 | 
			
		||||
@@ -1232,8 +1256,10 @@ message ButtonCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_BUTTON";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  uint32 device_id = 2;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== MEDIA PLAYER ====================
 | 
			
		||||
@@ -1301,6 +1327,7 @@ message MediaPlayerCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_MEDIA_PLAYER";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
 | 
			
		||||
@@ -1315,6 +1342,7 @@ message MediaPlayerCommandRequest {
 | 
			
		||||
 | 
			
		||||
  bool has_announcement = 8;
 | 
			
		||||
  bool announcement = 9;
 | 
			
		||||
  uint32 device_id = 10;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== BLUETOOTH ====================
 | 
			
		||||
@@ -1843,9 +1871,11 @@ message AlarmControlPanelCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_ALARM_CONTROL_PANEL";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  AlarmControlPanelStateCommand command = 2;
 | 
			
		||||
  string code = 3;
 | 
			
		||||
  uint32 device_id = 4;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ===================== TEXT =====================
 | 
			
		||||
@@ -1892,9 +1922,11 @@ message TextCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_TEXT";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  string state = 2;
 | 
			
		||||
  uint32 device_id = 3;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -1936,11 +1968,13 @@ message DateCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_DATETIME_DATE";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  uint32 year = 2;
 | 
			
		||||
  uint32 month = 3;
 | 
			
		||||
  uint32 day = 4;
 | 
			
		||||
  uint32 device_id = 5;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== DATETIME TIME ====================
 | 
			
		||||
@@ -1981,11 +2015,13 @@ message TimeCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_DATETIME_TIME";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  uint32 hour = 2;
 | 
			
		||||
  uint32 minute = 3;
 | 
			
		||||
  uint32 second = 4;
 | 
			
		||||
  uint32 device_id = 5;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== EVENT ====================
 | 
			
		||||
@@ -2065,11 +2101,13 @@ message ValveCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_VALVE";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  bool has_position = 2;
 | 
			
		||||
  float position = 3;
 | 
			
		||||
  bool stop = 4;
 | 
			
		||||
  uint32 device_id = 5;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== DATETIME DATETIME ====================
 | 
			
		||||
@@ -2108,9 +2146,11 @@ message DateTimeCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_DATETIME_DATETIME";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  fixed32 epoch_seconds = 2;
 | 
			
		||||
  uint32 device_id = 3;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ==================== UPDATE ====================
 | 
			
		||||
@@ -2160,7 +2200,9 @@ message UpdateCommandRequest {
 | 
			
		||||
  option (source) = SOURCE_CLIENT;
 | 
			
		||||
  option (ifdef) = "USE_UPDATE";
 | 
			
		||||
  option (no_delay) = true;
 | 
			
		||||
  option (base_class) = "CommandProtoMessage";
 | 
			
		||||
 | 
			
		||||
  fixed32 key = 1;
 | 
			
		||||
  UpdateCommand command = 2;
 | 
			
		||||
  uint32 device_id = 3;
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -38,10 +38,23 @@ static constexpr uint16_t PING_RETRY_INTERVAL = 1000;
 | 
			
		||||
static constexpr uint32_t KEEPALIVE_DISCONNECT_TIMEOUT = (KEEPALIVE_TIMEOUT_MS * 5) / 2;
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "api.connection";
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
static const int ESP32_CAMERA_STOP_STREAM = 5000;
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
static const int CAMERA_STOP_STREAM = 5000;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
// Helper macro for entity command handlers - gets entity by key, returns if not found, and creates call object
 | 
			
		||||
#define ENTITY_COMMAND_MAKE_CALL(entity_type, entity_var, getter_name) \
 | 
			
		||||
  entity_type *entity_var = App.get_##getter_name##_by_key(msg.key); \
 | 
			
		||||
  if ((entity_var) == nullptr) \
 | 
			
		||||
    return; \
 | 
			
		||||
  auto call = (entity_var)->make_call();
 | 
			
		||||
 | 
			
		||||
// Helper macro for entity command handlers that don't use make_call() - gets entity by key and returns if not found
 | 
			
		||||
#define ENTITY_COMMAND_GET(entity_type, entity_var, getter_name) \
 | 
			
		||||
  entity_type *entity_var = App.get_##getter_name##_by_key(msg.key); \
 | 
			
		||||
  if ((entity_var) == nullptr) \
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
APIConnection::APIConnection(std::unique_ptr<socket::Socket> sock, APIServer *parent)
 | 
			
		||||
    : parent_(parent), initial_state_iterator_(this), list_entities_iterator_(this) {
 | 
			
		||||
#if defined(USE_API_PLAINTEXT) && defined(USE_API_NOISE)
 | 
			
		||||
@@ -58,6 +71,11 @@ APIConnection::APIConnection(std::unique_ptr<socket::Socket> sock, APIServer *pa
 | 
			
		||||
#else
 | 
			
		||||
#error "No frame helper defined"
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
  if (camera::Camera::instance() != nullptr) {
 | 
			
		||||
    this->image_reader_ = std::unique_ptr<camera::CameraImageReader>{camera::Camera::instance()->create_image_reader()};
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint32_t APIConnection::get_batch_delay_ms_() const { return this->parent_->get_batch_delay(); }
 | 
			
		||||
@@ -175,15 +193,16 @@ void APIConnection::loop() {
 | 
			
		||||
      // If we can't send the ping request directly (tx_buffer full),
 | 
			
		||||
      // schedule it at the front of the batch so it will be sent with priority
 | 
			
		||||
      ESP_LOGW(TAG, "Buffer full, ping queued");
 | 
			
		||||
      this->schedule_message_front_(nullptr, &APIConnection::try_send_ping_request, PingRequest::MESSAGE_TYPE);
 | 
			
		||||
      this->schedule_message_front_(nullptr, &APIConnection::try_send_ping_request, PingRequest::MESSAGE_TYPE,
 | 
			
		||||
                                    PingRequest::ESTIMATED_SIZE);
 | 
			
		||||
      this->flags_.sent_ping = true;  // Mark as sent to avoid scheduling multiple pings
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
  if (this->image_reader_.available() && this->helper_->can_write_without_blocking()) {
 | 
			
		||||
    uint32_t to_send = std::min((size_t) MAX_PACKET_SIZE, this->image_reader_.available());
 | 
			
		||||
    bool done = this->image_reader_.available() == to_send;
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
  if (this->image_reader_ && this->image_reader_->available() && this->helper_->can_write_without_blocking()) {
 | 
			
		||||
    uint32_t to_send = std::min((size_t) MAX_BATCH_PACKET_SIZE, this->image_reader_->available());
 | 
			
		||||
    bool done = this->image_reader_->available() == to_send;
 | 
			
		||||
    uint32_t msg_size = 0;
 | 
			
		||||
    ProtoSize::add_fixed_field<4>(msg_size, 1, true);
 | 
			
		||||
    // partial message size calculated manually since its a special case
 | 
			
		||||
@@ -193,18 +212,18 @@ void APIConnection::loop() {
 | 
			
		||||
 | 
			
		||||
    auto buffer = this->create_buffer(msg_size);
 | 
			
		||||
    // fixed32 key = 1;
 | 
			
		||||
    buffer.encode_fixed32(1, esp32_camera::global_esp32_camera->get_object_id_hash());
 | 
			
		||||
    buffer.encode_fixed32(1, camera::Camera::instance()->get_object_id_hash());
 | 
			
		||||
    // bytes data = 2;
 | 
			
		||||
    buffer.encode_bytes(2, this->image_reader_.peek_data_buffer(), to_send);
 | 
			
		||||
    buffer.encode_bytes(2, this->image_reader_->peek_data_buffer(), to_send);
 | 
			
		||||
    // bool done = 3;
 | 
			
		||||
    buffer.encode_bool(3, done);
 | 
			
		||||
 | 
			
		||||
    bool success = this->send_buffer(buffer, CameraImageResponse::MESSAGE_TYPE);
 | 
			
		||||
 | 
			
		||||
    if (success) {
 | 
			
		||||
      this->image_reader_.consume_data(to_send);
 | 
			
		||||
      this->image_reader_->consume_data(to_send);
 | 
			
		||||
      if (done) {
 | 
			
		||||
        this->image_reader_.return_image();
 | 
			
		||||
        this->image_reader_->return_image();
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
@@ -247,7 +266,7 @@ void APIConnection::on_disconnect_response(const DisconnectResponse &value) {
 | 
			
		||||
 | 
			
		||||
// Encodes a message to the buffer and returns the total number of bytes used,
 | 
			
		||||
// including header and footer overhead. Returns 0 if the message doesn't fit.
 | 
			
		||||
uint16_t APIConnection::encode_message_to_buffer(ProtoMessage &msg, uint16_t message_type, APIConnection *conn,
 | 
			
		||||
uint16_t APIConnection::encode_message_to_buffer(ProtoMessage &msg, uint8_t message_type, APIConnection *conn,
 | 
			
		||||
                                                 uint32_t remaining_size, bool is_single) {
 | 
			
		||||
#ifdef HAS_PROTO_MESSAGE_DUMP
 | 
			
		||||
  // If in log-only mode, just log and return
 | 
			
		||||
@@ -298,7 +317,7 @@ uint16_t APIConnection::encode_message_to_buffer(ProtoMessage &msg, uint16_t mes
 | 
			
		||||
#ifdef USE_BINARY_SENSOR
 | 
			
		||||
bool APIConnection::send_binary_sensor_state(binary_sensor::BinarySensor *binary_sensor) {
 | 
			
		||||
  return this->send_message_smart_(binary_sensor, &APIConnection::try_send_binary_sensor_state,
 | 
			
		||||
                                   BinarySensorStateResponse::MESSAGE_TYPE);
 | 
			
		||||
                                   BinarySensorStateResponse::MESSAGE_TYPE, BinarySensorStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint16_t APIConnection::try_send_binary_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
@@ -325,7 +344,8 @@ uint16_t APIConnection::try_send_binary_sensor_info(EntityBase *entity, APIConne
 | 
			
		||||
 | 
			
		||||
#ifdef USE_COVER
 | 
			
		||||
bool APIConnection::send_cover_state(cover::Cover *cover) {
 | 
			
		||||
  return this->send_message_smart_(cover, &APIConnection::try_send_cover_state, CoverStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(cover, &APIConnection::try_send_cover_state, CoverStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   CoverStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_cover_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                             bool is_single) {
 | 
			
		||||
@@ -356,11 +376,7 @@ uint16_t APIConnection::try_send_cover_info(EntityBase *entity, APIConnection *c
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesCoverResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::cover_command(const CoverCommandRequest &msg) {
 | 
			
		||||
  cover::Cover *cover = App.get_cover_by_key(msg.key);
 | 
			
		||||
  if (cover == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = cover->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(cover::Cover, cover, cover)
 | 
			
		||||
  if (msg.has_legacy_command) {
 | 
			
		||||
    switch (msg.legacy_command) {
 | 
			
		||||
      case enums::LEGACY_COVER_COMMAND_OPEN:
 | 
			
		||||
@@ -386,7 +402,8 @@ void APIConnection::cover_command(const CoverCommandRequest &msg) {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_FAN
 | 
			
		||||
bool APIConnection::send_fan_state(fan::Fan *fan) {
 | 
			
		||||
  return this->send_message_smart_(fan, &APIConnection::try_send_fan_state, FanStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(fan, &APIConnection::try_send_fan_state, FanStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   FanStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_fan_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                           bool is_single) {
 | 
			
		||||
@@ -422,11 +439,7 @@ uint16_t APIConnection::try_send_fan_info(EntityBase *entity, APIConnection *con
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesFanResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::fan_command(const FanCommandRequest &msg) {
 | 
			
		||||
  fan::Fan *fan = App.get_fan_by_key(msg.key);
 | 
			
		||||
  if (fan == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = fan->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(fan::Fan, fan, fan)
 | 
			
		||||
  if (msg.has_state)
 | 
			
		||||
    call.set_state(msg.state);
 | 
			
		||||
  if (msg.has_oscillating)
 | 
			
		||||
@@ -445,7 +458,8 @@ void APIConnection::fan_command(const FanCommandRequest &msg) {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_LIGHT
 | 
			
		||||
bool APIConnection::send_light_state(light::LightState *light) {
 | 
			
		||||
  return this->send_message_smart_(light, &APIConnection::try_send_light_state, LightStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(light, &APIConnection::try_send_light_state, LightStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   LightStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_light_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                             bool is_single) {
 | 
			
		||||
@@ -499,11 +513,7 @@ uint16_t APIConnection::try_send_light_info(EntityBase *entity, APIConnection *c
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesLightResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::light_command(const LightCommandRequest &msg) {
 | 
			
		||||
  light::LightState *light = App.get_light_by_key(msg.key);
 | 
			
		||||
  if (light == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = light->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(light::LightState, light, light)
 | 
			
		||||
  if (msg.has_state)
 | 
			
		||||
    call.set_state(msg.state);
 | 
			
		||||
  if (msg.has_brightness)
 | 
			
		||||
@@ -537,7 +547,8 @@ void APIConnection::light_command(const LightCommandRequest &msg) {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
bool APIConnection::send_sensor_state(sensor::Sensor *sensor) {
 | 
			
		||||
  return this->send_message_smart_(sensor, &APIConnection::try_send_sensor_state, SensorStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(sensor, &APIConnection::try_send_sensor_state, SensorStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   SensorStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint16_t APIConnection::try_send_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
@@ -569,7 +580,8 @@ uint16_t APIConnection::try_send_sensor_info(EntityBase *entity, APIConnection *
 | 
			
		||||
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
bool APIConnection::send_switch_state(switch_::Switch *a_switch) {
 | 
			
		||||
  return this->send_message_smart_(a_switch, &APIConnection::try_send_switch_state, SwitchStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(a_switch, &APIConnection::try_send_switch_state, SwitchStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   SwitchStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint16_t APIConnection::try_send_switch_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
@@ -592,9 +604,7 @@ uint16_t APIConnection::try_send_switch_info(EntityBase *entity, APIConnection *
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesSwitchResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::switch_command(const SwitchCommandRequest &msg) {
 | 
			
		||||
  switch_::Switch *a_switch = App.get_switch_by_key(msg.key);
 | 
			
		||||
  if (a_switch == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
  ENTITY_COMMAND_GET(switch_::Switch, a_switch, switch)
 | 
			
		||||
 | 
			
		||||
  if (msg.state) {
 | 
			
		||||
    a_switch->turn_on();
 | 
			
		||||
@@ -607,7 +617,7 @@ void APIConnection::switch_command(const SwitchCommandRequest &msg) {
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
bool APIConnection::send_text_sensor_state(text_sensor::TextSensor *text_sensor) {
 | 
			
		||||
  return this->send_message_smart_(text_sensor, &APIConnection::try_send_text_sensor_state,
 | 
			
		||||
                                   TextSensorStateResponse::MESSAGE_TYPE);
 | 
			
		||||
                                   TextSensorStateResponse::MESSAGE_TYPE, TextSensorStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint16_t APIConnection::try_send_text_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
@@ -634,7 +644,8 @@ uint16_t APIConnection::try_send_text_sensor_info(EntityBase *entity, APIConnect
 | 
			
		||||
 | 
			
		||||
#ifdef USE_CLIMATE
 | 
			
		||||
bool APIConnection::send_climate_state(climate::Climate *climate) {
 | 
			
		||||
  return this->send_message_smart_(climate, &APIConnection::try_send_climate_state, ClimateStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(climate, &APIConnection::try_send_climate_state, ClimateStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   ClimateStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_climate_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                               bool is_single) {
 | 
			
		||||
@@ -703,11 +714,7 @@ uint16_t APIConnection::try_send_climate_info(EntityBase *entity, APIConnection
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesClimateResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::climate_command(const ClimateCommandRequest &msg) {
 | 
			
		||||
  climate::Climate *climate = App.get_climate_by_key(msg.key);
 | 
			
		||||
  if (climate == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = climate->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(climate::Climate, climate, climate)
 | 
			
		||||
  if (msg.has_mode)
 | 
			
		||||
    call.set_mode(static_cast<climate::ClimateMode>(msg.mode));
 | 
			
		||||
  if (msg.has_target_temperature)
 | 
			
		||||
@@ -734,7 +741,8 @@ void APIConnection::climate_command(const ClimateCommandRequest &msg) {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
bool APIConnection::send_number_state(number::Number *number) {
 | 
			
		||||
  return this->send_message_smart_(number, &APIConnection::try_send_number_state, NumberStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(number, &APIConnection::try_send_number_state, NumberStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   NumberStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint16_t APIConnection::try_send_number_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
@@ -762,11 +770,7 @@ uint16_t APIConnection::try_send_number_info(EntityBase *entity, APIConnection *
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesNumberResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::number_command(const NumberCommandRequest &msg) {
 | 
			
		||||
  number::Number *number = App.get_number_by_key(msg.key);
 | 
			
		||||
  if (number == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = number->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(number::Number, number, number)
 | 
			
		||||
  call.set_value(msg.state);
 | 
			
		||||
  call.perform();
 | 
			
		||||
}
 | 
			
		||||
@@ -774,7 +778,8 @@ void APIConnection::number_command(const NumberCommandRequest &msg) {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_DATETIME_DATE
 | 
			
		||||
bool APIConnection::send_date_state(datetime::DateEntity *date) {
 | 
			
		||||
  return this->send_message_smart_(date, &APIConnection::try_send_date_state, DateStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(date, &APIConnection::try_send_date_state, DateStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   DateStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_date_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                            bool is_single) {
 | 
			
		||||
@@ -796,11 +801,7 @@ uint16_t APIConnection::try_send_date_info(EntityBase *entity, APIConnection *co
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesDateResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::date_command(const DateCommandRequest &msg) {
 | 
			
		||||
  datetime::DateEntity *date = App.get_date_by_key(msg.key);
 | 
			
		||||
  if (date == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = date->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(datetime::DateEntity, date, date)
 | 
			
		||||
  call.set_date(msg.year, msg.month, msg.day);
 | 
			
		||||
  call.perform();
 | 
			
		||||
}
 | 
			
		||||
@@ -808,7 +809,8 @@ void APIConnection::date_command(const DateCommandRequest &msg) {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_DATETIME_TIME
 | 
			
		||||
bool APIConnection::send_time_state(datetime::TimeEntity *time) {
 | 
			
		||||
  return this->send_message_smart_(time, &APIConnection::try_send_time_state, TimeStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(time, &APIConnection::try_send_time_state, TimeStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   TimeStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_time_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                            bool is_single) {
 | 
			
		||||
@@ -830,11 +832,7 @@ uint16_t APIConnection::try_send_time_info(EntityBase *entity, APIConnection *co
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesTimeResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::time_command(const TimeCommandRequest &msg) {
 | 
			
		||||
  datetime::TimeEntity *time = App.get_time_by_key(msg.key);
 | 
			
		||||
  if (time == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = time->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(datetime::TimeEntity, time, time)
 | 
			
		||||
  call.set_time(msg.hour, msg.minute, msg.second);
 | 
			
		||||
  call.perform();
 | 
			
		||||
}
 | 
			
		||||
@@ -843,7 +841,7 @@ void APIConnection::time_command(const TimeCommandRequest &msg) {
 | 
			
		||||
#ifdef USE_DATETIME_DATETIME
 | 
			
		||||
bool APIConnection::send_datetime_state(datetime::DateTimeEntity *datetime) {
 | 
			
		||||
  return this->send_message_smart_(datetime, &APIConnection::try_send_datetime_state,
 | 
			
		||||
                                   DateTimeStateResponse::MESSAGE_TYPE);
 | 
			
		||||
                                   DateTimeStateResponse::MESSAGE_TYPE, DateTimeStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_datetime_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                                bool is_single) {
 | 
			
		||||
@@ -866,11 +864,7 @@ uint16_t APIConnection::try_send_datetime_info(EntityBase *entity, APIConnection
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesDateTimeResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::datetime_command(const DateTimeCommandRequest &msg) {
 | 
			
		||||
  datetime::DateTimeEntity *datetime = App.get_datetime_by_key(msg.key);
 | 
			
		||||
  if (datetime == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = datetime->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(datetime::DateTimeEntity, datetime, datetime)
 | 
			
		||||
  call.set_datetime(msg.epoch_seconds);
 | 
			
		||||
  call.perform();
 | 
			
		||||
}
 | 
			
		||||
@@ -878,7 +872,8 @@ void APIConnection::datetime_command(const DateTimeCommandRequest &msg) {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_TEXT
 | 
			
		||||
bool APIConnection::send_text_state(text::Text *text) {
 | 
			
		||||
  return this->send_message_smart_(text, &APIConnection::try_send_text_state, TextStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(text, &APIConnection::try_send_text_state, TextStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   TextStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint16_t APIConnection::try_send_text_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
@@ -904,11 +899,7 @@ uint16_t APIConnection::try_send_text_info(EntityBase *entity, APIConnection *co
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesTextResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::text_command(const TextCommandRequest &msg) {
 | 
			
		||||
  text::Text *text = App.get_text_by_key(msg.key);
 | 
			
		||||
  if (text == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = text->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(text::Text, text, text)
 | 
			
		||||
  call.set_value(msg.state);
 | 
			
		||||
  call.perform();
 | 
			
		||||
}
 | 
			
		||||
@@ -916,7 +907,8 @@ void APIConnection::text_command(const TextCommandRequest &msg) {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
bool APIConnection::send_select_state(select::Select *select) {
 | 
			
		||||
  return this->send_message_smart_(select, &APIConnection::try_send_select_state, SelectStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(select, &APIConnection::try_send_select_state, SelectStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   SelectStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint16_t APIConnection::try_send_select_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
@@ -940,11 +932,7 @@ uint16_t APIConnection::try_send_select_info(EntityBase *entity, APIConnection *
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesSelectResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::select_command(const SelectCommandRequest &msg) {
 | 
			
		||||
  select::Select *select = App.get_select_by_key(msg.key);
 | 
			
		||||
  if (select == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = select->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(select::Select, select, select)
 | 
			
		||||
  call.set_option(msg.state);
 | 
			
		||||
  call.perform();
 | 
			
		||||
}
 | 
			
		||||
@@ -961,17 +949,15 @@ uint16_t APIConnection::try_send_button_info(EntityBase *entity, APIConnection *
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesButtonResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void esphome::api::APIConnection::button_command(const ButtonCommandRequest &msg) {
 | 
			
		||||
  button::Button *button = App.get_button_by_key(msg.key);
 | 
			
		||||
  if (button == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  ENTITY_COMMAND_GET(button::Button, button, button)
 | 
			
		||||
  button->press();
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_LOCK
 | 
			
		||||
bool APIConnection::send_lock_state(lock::Lock *a_lock) {
 | 
			
		||||
  return this->send_message_smart_(a_lock, &APIConnection::try_send_lock_state, LockStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(a_lock, &APIConnection::try_send_lock_state, LockStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   LockStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint16_t APIConnection::try_send_lock_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
@@ -995,9 +981,7 @@ uint16_t APIConnection::try_send_lock_info(EntityBase *entity, APIConnection *co
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesLockResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::lock_command(const LockCommandRequest &msg) {
 | 
			
		||||
  lock::Lock *a_lock = App.get_lock_by_key(msg.key);
 | 
			
		||||
  if (a_lock == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
  ENTITY_COMMAND_GET(lock::Lock, a_lock, lock)
 | 
			
		||||
 | 
			
		||||
  switch (msg.command) {
 | 
			
		||||
    case enums::LOCK_UNLOCK:
 | 
			
		||||
@@ -1015,7 +999,8 @@ void APIConnection::lock_command(const LockCommandRequest &msg) {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_VALVE
 | 
			
		||||
bool APIConnection::send_valve_state(valve::Valve *valve) {
 | 
			
		||||
  return this->send_message_smart_(valve, &APIConnection::try_send_valve_state, ValveStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(valve, &APIConnection::try_send_valve_state, ValveStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   ValveStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_valve_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                             bool is_single) {
 | 
			
		||||
@@ -1040,11 +1025,7 @@ uint16_t APIConnection::try_send_valve_info(EntityBase *entity, APIConnection *c
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesValveResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::valve_command(const ValveCommandRequest &msg) {
 | 
			
		||||
  valve::Valve *valve = App.get_valve_by_key(msg.key);
 | 
			
		||||
  if (valve == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = valve->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(valve::Valve, valve, valve)
 | 
			
		||||
  if (msg.has_position)
 | 
			
		||||
    call.set_position(msg.position);
 | 
			
		||||
  if (msg.stop)
 | 
			
		||||
@@ -1056,7 +1037,7 @@ void APIConnection::valve_command(const ValveCommandRequest &msg) {
 | 
			
		||||
#ifdef USE_MEDIA_PLAYER
 | 
			
		||||
bool APIConnection::send_media_player_state(media_player::MediaPlayer *media_player) {
 | 
			
		||||
  return this->send_message_smart_(media_player, &APIConnection::try_send_media_player_state,
 | 
			
		||||
                                   MediaPlayerStateResponse::MESSAGE_TYPE);
 | 
			
		||||
                                   MediaPlayerStateResponse::MESSAGE_TYPE, MediaPlayerStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_media_player_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                                    bool is_single) {
 | 
			
		||||
@@ -1091,11 +1072,7 @@ uint16_t APIConnection::try_send_media_player_info(EntityBase *entity, APIConnec
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesMediaPlayerResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::media_player_command(const MediaPlayerCommandRequest &msg) {
 | 
			
		||||
  media_player::MediaPlayer *media_player = App.get_media_player_by_key(msg.key);
 | 
			
		||||
  if (media_player == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = media_player->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(media_player::MediaPlayer, media_player, media_player)
 | 
			
		||||
  if (msg.has_command) {
 | 
			
		||||
    call.set_command(static_cast<media_player::MediaPlayerCommand>(msg.command));
 | 
			
		||||
  }
 | 
			
		||||
@@ -1112,36 +1089,36 @@ void APIConnection::media_player_command(const MediaPlayerCommandRequest &msg) {
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
void APIConnection::set_camera_state(std::shared_ptr<esp32_camera::CameraImage> image) {
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
void APIConnection::set_camera_state(std::shared_ptr<camera::CameraImage> image) {
 | 
			
		||||
  if (!this->flags_.state_subscription)
 | 
			
		||||
    return;
 | 
			
		||||
  if (this->image_reader_.available())
 | 
			
		||||
  if (!this->image_reader_)
 | 
			
		||||
    return;
 | 
			
		||||
  if (image->was_requested_by(esphome::esp32_camera::API_REQUESTER) ||
 | 
			
		||||
      image->was_requested_by(esphome::esp32_camera::IDLE))
 | 
			
		||||
    this->image_reader_.set_image(std::move(image));
 | 
			
		||||
  if (this->image_reader_->available())
 | 
			
		||||
    return;
 | 
			
		||||
  if (image->was_requested_by(esphome::camera::API_REQUESTER) || image->was_requested_by(esphome::camera::IDLE))
 | 
			
		||||
    this->image_reader_->set_image(std::move(image));
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_camera_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                             bool is_single) {
 | 
			
		||||
  auto *camera = static_cast<esp32_camera::ESP32Camera *>(entity);
 | 
			
		||||
  auto *camera = static_cast<camera::Camera *>(entity);
 | 
			
		||||
  ListEntitiesCameraResponse msg;
 | 
			
		||||
  msg.unique_id = get_default_unique_id("camera", camera);
 | 
			
		||||
  fill_entity_info_base(camera, msg);
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesCameraResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::camera_image(const CameraImageRequest &msg) {
 | 
			
		||||
  if (esp32_camera::global_esp32_camera == nullptr)
 | 
			
		||||
  if (camera::Camera::instance() == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  if (msg.single)
 | 
			
		||||
    esp32_camera::global_esp32_camera->request_image(esphome::esp32_camera::API_REQUESTER);
 | 
			
		||||
    camera::Camera::instance()->request_image(esphome::camera::API_REQUESTER);
 | 
			
		||||
  if (msg.stream) {
 | 
			
		||||
    esp32_camera::global_esp32_camera->start_stream(esphome::esp32_camera::API_REQUESTER);
 | 
			
		||||
    camera::Camera::instance()->start_stream(esphome::camera::API_REQUESTER);
 | 
			
		||||
 | 
			
		||||
    App.scheduler.set_timeout(this->parent_, "api_esp32_camera_stop_stream", ESP32_CAMERA_STOP_STREAM, []() {
 | 
			
		||||
      esp32_camera::global_esp32_camera->stop_stream(esphome::esp32_camera::API_REQUESTER);
 | 
			
		||||
    });
 | 
			
		||||
    App.scheduler.set_timeout(this->parent_, "api_camera_stop_stream", CAMERA_STOP_STREAM,
 | 
			
		||||
                              []() { camera::Camera::instance()->stop_stream(esphome::camera::API_REQUESTER); });
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
@@ -1213,66 +1190,53 @@ void APIConnection::bluetooth_scanner_set_mode(const BluetoothScannerSetModeRequ
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_VOICE_ASSISTANT
 | 
			
		||||
bool APIConnection::check_voice_assistant_api_connection_() const {
 | 
			
		||||
  return voice_assistant::global_voice_assistant != nullptr &&
 | 
			
		||||
         voice_assistant::global_voice_assistant->get_api_connection() == this;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void APIConnection::subscribe_voice_assistant(const SubscribeVoiceAssistantRequest &msg) {
 | 
			
		||||
  if (voice_assistant::global_voice_assistant != nullptr) {
 | 
			
		||||
    voice_assistant::global_voice_assistant->client_subscription(this, msg.subscribe);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::on_voice_assistant_response(const VoiceAssistantResponse &msg) {
 | 
			
		||||
  if (voice_assistant::global_voice_assistant != nullptr) {
 | 
			
		||||
    if (voice_assistant::global_voice_assistant->get_api_connection() != this) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
  if (!this->check_voice_assistant_api_connection_()) {
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
    if (msg.error) {
 | 
			
		||||
      voice_assistant::global_voice_assistant->failed_to_start();
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    if (msg.port == 0) {
 | 
			
		||||
      // Use API Audio
 | 
			
		||||
      voice_assistant::global_voice_assistant->start_streaming();
 | 
			
		||||
    } else {
 | 
			
		||||
      struct sockaddr_storage storage;
 | 
			
		||||
      socklen_t len = sizeof(storage);
 | 
			
		||||
      this->helper_->getpeername((struct sockaddr *) &storage, &len);
 | 
			
		||||
      voice_assistant::global_voice_assistant->start_streaming(&storage, msg.port);
 | 
			
		||||
    }
 | 
			
		||||
  if (msg.error) {
 | 
			
		||||
    voice_assistant::global_voice_assistant->failed_to_start();
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  if (msg.port == 0) {
 | 
			
		||||
    // Use API Audio
 | 
			
		||||
    voice_assistant::global_voice_assistant->start_streaming();
 | 
			
		||||
  } else {
 | 
			
		||||
    struct sockaddr_storage storage;
 | 
			
		||||
    socklen_t len = sizeof(storage);
 | 
			
		||||
    this->helper_->getpeername((struct sockaddr *) &storage, &len);
 | 
			
		||||
    voice_assistant::global_voice_assistant->start_streaming(&storage, msg.port);
 | 
			
		||||
  }
 | 
			
		||||
};
 | 
			
		||||
void APIConnection::on_voice_assistant_event_response(const VoiceAssistantEventResponse &msg) {
 | 
			
		||||
  if (voice_assistant::global_voice_assistant != nullptr) {
 | 
			
		||||
    if (voice_assistant::global_voice_assistant->get_api_connection() != this) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
  if (this->check_voice_assistant_api_connection_()) {
 | 
			
		||||
    voice_assistant::global_voice_assistant->on_event(msg);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::on_voice_assistant_audio(const VoiceAssistantAudio &msg) {
 | 
			
		||||
  if (voice_assistant::global_voice_assistant != nullptr) {
 | 
			
		||||
    if (voice_assistant::global_voice_assistant->get_api_connection() != this) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
  if (this->check_voice_assistant_api_connection_()) {
 | 
			
		||||
    voice_assistant::global_voice_assistant->on_audio(msg);
 | 
			
		||||
  }
 | 
			
		||||
};
 | 
			
		||||
void APIConnection::on_voice_assistant_timer_event_response(const VoiceAssistantTimerEventResponse &msg) {
 | 
			
		||||
  if (voice_assistant::global_voice_assistant != nullptr) {
 | 
			
		||||
    if (voice_assistant::global_voice_assistant->get_api_connection() != this) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
  if (this->check_voice_assistant_api_connection_()) {
 | 
			
		||||
    voice_assistant::global_voice_assistant->on_timer_event(msg);
 | 
			
		||||
  }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
void APIConnection::on_voice_assistant_announce_request(const VoiceAssistantAnnounceRequest &msg) {
 | 
			
		||||
  if (voice_assistant::global_voice_assistant != nullptr) {
 | 
			
		||||
    if (voice_assistant::global_voice_assistant->get_api_connection() != this) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
  if (this->check_voice_assistant_api_connection_()) {
 | 
			
		||||
    voice_assistant::global_voice_assistant->on_announce(msg);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1280,35 +1244,29 @@ void APIConnection::on_voice_assistant_announce_request(const VoiceAssistantAnno
 | 
			
		||||
VoiceAssistantConfigurationResponse APIConnection::voice_assistant_get_configuration(
 | 
			
		||||
    const VoiceAssistantConfigurationRequest &msg) {
 | 
			
		||||
  VoiceAssistantConfigurationResponse resp;
 | 
			
		||||
  if (voice_assistant::global_voice_assistant != nullptr) {
 | 
			
		||||
    if (voice_assistant::global_voice_assistant->get_api_connection() != this) {
 | 
			
		||||
      return resp;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    auto &config = voice_assistant::global_voice_assistant->get_configuration();
 | 
			
		||||
    for (auto &wake_word : config.available_wake_words) {
 | 
			
		||||
      VoiceAssistantWakeWord resp_wake_word;
 | 
			
		||||
      resp_wake_word.id = wake_word.id;
 | 
			
		||||
      resp_wake_word.wake_word = wake_word.wake_word;
 | 
			
		||||
      for (const auto &lang : wake_word.trained_languages) {
 | 
			
		||||
        resp_wake_word.trained_languages.push_back(lang);
 | 
			
		||||
      }
 | 
			
		||||
      resp.available_wake_words.push_back(std::move(resp_wake_word));
 | 
			
		||||
    }
 | 
			
		||||
    for (auto &wake_word_id : config.active_wake_words) {
 | 
			
		||||
      resp.active_wake_words.push_back(wake_word_id);
 | 
			
		||||
    }
 | 
			
		||||
    resp.max_active_wake_words = config.max_active_wake_words;
 | 
			
		||||
  if (!this->check_voice_assistant_api_connection_()) {
 | 
			
		||||
    return resp;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  auto &config = voice_assistant::global_voice_assistant->get_configuration();
 | 
			
		||||
  for (auto &wake_word : config.available_wake_words) {
 | 
			
		||||
    VoiceAssistantWakeWord resp_wake_word;
 | 
			
		||||
    resp_wake_word.id = wake_word.id;
 | 
			
		||||
    resp_wake_word.wake_word = wake_word.wake_word;
 | 
			
		||||
    for (const auto &lang : wake_word.trained_languages) {
 | 
			
		||||
      resp_wake_word.trained_languages.push_back(lang);
 | 
			
		||||
    }
 | 
			
		||||
    resp.available_wake_words.push_back(std::move(resp_wake_word));
 | 
			
		||||
  }
 | 
			
		||||
  for (auto &wake_word_id : config.active_wake_words) {
 | 
			
		||||
    resp.active_wake_words.push_back(wake_word_id);
 | 
			
		||||
  }
 | 
			
		||||
  resp.max_active_wake_words = config.max_active_wake_words;
 | 
			
		||||
  return resp;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void APIConnection::voice_assistant_set_configuration(const VoiceAssistantSetConfiguration &msg) {
 | 
			
		||||
  if (voice_assistant::global_voice_assistant != nullptr) {
 | 
			
		||||
    if (voice_assistant::global_voice_assistant->get_api_connection() != this) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
  if (this->check_voice_assistant_api_connection_()) {
 | 
			
		||||
    voice_assistant::global_voice_assistant->on_set_configuration(msg.active_wake_words);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1318,7 +1276,8 @@ void APIConnection::voice_assistant_set_configuration(const VoiceAssistantSetCon
 | 
			
		||||
#ifdef USE_ALARM_CONTROL_PANEL
 | 
			
		||||
bool APIConnection::send_alarm_control_panel_state(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) {
 | 
			
		||||
  return this->send_message_smart_(a_alarm_control_panel, &APIConnection::try_send_alarm_control_panel_state,
 | 
			
		||||
                                   AlarmControlPanelStateResponse::MESSAGE_TYPE);
 | 
			
		||||
                                   AlarmControlPanelStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   AlarmControlPanelStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_alarm_control_panel_state(EntityBase *entity, APIConnection *conn,
 | 
			
		||||
                                                           uint32_t remaining_size, bool is_single) {
 | 
			
		||||
@@ -1341,11 +1300,7 @@ uint16_t APIConnection::try_send_alarm_control_panel_info(EntityBase *entity, AP
 | 
			
		||||
                                  is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::alarm_control_panel_command(const AlarmControlPanelCommandRequest &msg) {
 | 
			
		||||
  alarm_control_panel::AlarmControlPanel *a_alarm_control_panel = App.get_alarm_control_panel_by_key(msg.key);
 | 
			
		||||
  if (a_alarm_control_panel == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
 | 
			
		||||
  auto call = a_alarm_control_panel->make_call();
 | 
			
		||||
  ENTITY_COMMAND_MAKE_CALL(alarm_control_panel::AlarmControlPanel, a_alarm_control_panel, alarm_control_panel)
 | 
			
		||||
  switch (msg.command) {
 | 
			
		||||
    case enums::ALARM_CONTROL_PANEL_DISARM:
 | 
			
		||||
      call.disarm();
 | 
			
		||||
@@ -1376,7 +1331,8 @@ void APIConnection::alarm_control_panel_command(const AlarmControlPanelCommandRe
 | 
			
		||||
 | 
			
		||||
#ifdef USE_EVENT
 | 
			
		||||
void APIConnection::send_event(event::Event *event, const std::string &event_type) {
 | 
			
		||||
  this->schedule_message_(event, MessageCreator(event_type), EventResponse::MESSAGE_TYPE);
 | 
			
		||||
  this->schedule_message_(event, MessageCreator(event_type), EventResponse::MESSAGE_TYPE,
 | 
			
		||||
                          EventResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_event_response(event::Event *event, const std::string &event_type, APIConnection *conn,
 | 
			
		||||
                                                uint32_t remaining_size, bool is_single) {
 | 
			
		||||
@@ -1401,7 +1357,8 @@ uint16_t APIConnection::try_send_event_info(EntityBase *entity, APIConnection *c
 | 
			
		||||
 | 
			
		||||
#ifdef USE_UPDATE
 | 
			
		||||
bool APIConnection::send_update_state(update::UpdateEntity *update) {
 | 
			
		||||
  return this->send_message_smart_(update, &APIConnection::try_send_update_state, UpdateStateResponse::MESSAGE_TYPE);
 | 
			
		||||
  return this->send_message_smart_(update, &APIConnection::try_send_update_state, UpdateStateResponse::MESSAGE_TYPE,
 | 
			
		||||
                                   UpdateStateResponse::ESTIMATED_SIZE);
 | 
			
		||||
}
 | 
			
		||||
uint16_t APIConnection::try_send_update_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                              bool is_single) {
 | 
			
		||||
@@ -1433,9 +1390,7 @@ uint16_t APIConnection::try_send_update_info(EntityBase *entity, APIConnection *
 | 
			
		||||
  return encode_message_to_buffer(msg, ListEntitiesUpdateResponse::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
void APIConnection::update_command(const UpdateCommandRequest &msg) {
 | 
			
		||||
  update::UpdateEntity *update = App.get_update_by_key(msg.key);
 | 
			
		||||
  if (update == nullptr)
 | 
			
		||||
    return;
 | 
			
		||||
  ENTITY_COMMAND_GET(update::UpdateEntity, update, update)
 | 
			
		||||
 | 
			
		||||
  switch (msg.command) {
 | 
			
		||||
    case enums::UPDATE_COMMAND_UPDATE:
 | 
			
		||||
@@ -1454,12 +1409,11 @@ void APIConnection::update_command(const UpdateCommandRequest &msg) {
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
bool APIConnection::try_send_log_message(int level, const char *tag, const char *line) {
 | 
			
		||||
bool APIConnection::try_send_log_message(int level, const char *tag, const char *line, size_t message_len) {
 | 
			
		||||
  if (this->flags_.log_subscription < level)
 | 
			
		||||
    return false;
 | 
			
		||||
 | 
			
		||||
  // Pre-calculate message size to avoid reallocations
 | 
			
		||||
  const size_t line_length = strlen(line);
 | 
			
		||||
  uint32_t msg_size = 0;
 | 
			
		||||
 | 
			
		||||
  // Add size for level field (field ID 1, varint type)
 | 
			
		||||
@@ -1468,14 +1422,14 @@ bool APIConnection::try_send_log_message(int level, const char *tag, const char
 | 
			
		||||
 | 
			
		||||
  // Add size for string field (field ID 3, string type)
 | 
			
		||||
  // 1 byte for field tag + size of length varint + string length
 | 
			
		||||
  msg_size += 1 + api::ProtoSize::varint(static_cast<uint32_t>(line_length)) + line_length;
 | 
			
		||||
  msg_size += 1 + api::ProtoSize::varint(static_cast<uint32_t>(message_len)) + message_len;
 | 
			
		||||
 | 
			
		||||
  // Create a pre-sized buffer
 | 
			
		||||
  auto buffer = this->create_buffer(msg_size);
 | 
			
		||||
 | 
			
		||||
  // Encode the message (SubscribeLogsResponse)
 | 
			
		||||
  buffer.encode_uint32(1, static_cast<uint32_t>(level));  // LogLevel level = 1
 | 
			
		||||
  buffer.encode_string(3, line, line_length);             // string message = 3
 | 
			
		||||
  buffer.encode_string(3, line, message_len);             // string message = 3
 | 
			
		||||
 | 
			
		||||
  // SubscribeLogsResponse - 29
 | 
			
		||||
  return this->send_buffer(buffer, SubscribeLogsResponse::MESSAGE_TYPE);
 | 
			
		||||
@@ -1597,6 +1551,7 @@ void APIConnection::on_home_assistant_state_response(const HomeAssistantStateRes
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
void APIConnection::execute_service(const ExecuteServiceRequest &msg) {
 | 
			
		||||
  bool found = false;
 | 
			
		||||
  for (auto *service : this->parent_->get_user_services()) {
 | 
			
		||||
@@ -1608,6 +1563,7 @@ void APIConnection::execute_service(const ExecuteServiceRequest &msg) {
 | 
			
		||||
    ESP_LOGV(TAG, "Could not find service");
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_API_NOISE
 | 
			
		||||
NoiseEncryptionSetKeyResponse APIConnection::noise_encryption_set_key(const NoiseEncryptionSetKeyRequest &msg) {
 | 
			
		||||
  psk_t psk{};
 | 
			
		||||
@@ -1651,7 +1607,7 @@ bool APIConnection::try_to_clear_buffer(bool log_out_of_space) {
 | 
			
		||||
  }
 | 
			
		||||
  return false;
 | 
			
		||||
}
 | 
			
		||||
bool APIConnection::send_buffer(ProtoWriteBuffer buffer, uint16_t message_type) {
 | 
			
		||||
bool APIConnection::send_buffer(ProtoWriteBuffer buffer, uint8_t message_type) {
 | 
			
		||||
  if (!this->try_to_clear_buffer(message_type != SubscribeLogsResponse::MESSAGE_TYPE)) {  // SubscribeLogsResponse
 | 
			
		||||
    return false;
 | 
			
		||||
  }
 | 
			
		||||
@@ -1685,7 +1641,8 @@ void APIConnection::on_fatal_error() {
 | 
			
		||||
  this->flags_.remove = true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void APIConnection::DeferredBatch::add_item(EntityBase *entity, MessageCreator creator, uint16_t message_type) {
 | 
			
		||||
void APIConnection::DeferredBatch::add_item(EntityBase *entity, MessageCreator creator, uint8_t message_type,
 | 
			
		||||
                                            uint8_t estimated_size) {
 | 
			
		||||
  // Check if we already have a message of this type for this entity
 | 
			
		||||
  // This provides deduplication per entity/message_type combination
 | 
			
		||||
  // O(n) but optimized for RAM and not performance.
 | 
			
		||||
@@ -1700,12 +1657,13 @@ void APIConnection::DeferredBatch::add_item(EntityBase *entity, MessageCreator c
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // No existing item found, add new one
 | 
			
		||||
  items.emplace_back(entity, std::move(creator), message_type);
 | 
			
		||||
  items.emplace_back(entity, std::move(creator), message_type, estimated_size);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void APIConnection::DeferredBatch::add_item_front(EntityBase *entity, MessageCreator creator, uint16_t message_type) {
 | 
			
		||||
void APIConnection::DeferredBatch::add_item_front(EntityBase *entity, MessageCreator creator, uint8_t message_type,
 | 
			
		||||
                                                  uint8_t estimated_size) {
 | 
			
		||||
  // Insert at front for high priority messages (no deduplication check)
 | 
			
		||||
  items.insert(items.begin(), BatchItem(entity, std::move(creator), message_type));
 | 
			
		||||
  items.insert(items.begin(), BatchItem(entity, std::move(creator), message_type, estimated_size));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool APIConnection::schedule_batch_() {
 | 
			
		||||
@@ -1777,7 +1735,7 @@ void APIConnection::process_batch_() {
 | 
			
		||||
  uint32_t total_estimated_size = 0;
 | 
			
		||||
  for (size_t i = 0; i < this->deferred_batch_.size(); i++) {
 | 
			
		||||
    const auto &item = this->deferred_batch_[i];
 | 
			
		||||
    total_estimated_size += get_estimated_message_size(item.message_type);
 | 
			
		||||
    total_estimated_size += item.estimated_size;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Calculate total overhead for all messages
 | 
			
		||||
@@ -1815,9 +1773,9 @@ void APIConnection::process_batch_() {
 | 
			
		||||
 | 
			
		||||
    // Update tracking variables
 | 
			
		||||
    items_processed++;
 | 
			
		||||
    // After first message, set remaining size to MAX_PACKET_SIZE to avoid fragmentation
 | 
			
		||||
    // After first message, set remaining size to MAX_BATCH_PACKET_SIZE to avoid fragmentation
 | 
			
		||||
    if (items_processed == 1) {
 | 
			
		||||
      remaining_size = MAX_PACKET_SIZE;
 | 
			
		||||
      remaining_size = MAX_BATCH_PACKET_SIZE;
 | 
			
		||||
    }
 | 
			
		||||
    remaining_size -= payload_size;
 | 
			
		||||
    // Calculate where the next message's header padding will start
 | 
			
		||||
@@ -1871,7 +1829,7 @@ void APIConnection::process_batch_() {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint16_t APIConnection::MessageCreator::operator()(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                                   bool is_single, uint16_t message_type) const {
 | 
			
		||||
                                                   bool is_single, uint8_t message_type) const {
 | 
			
		||||
#ifdef USE_EVENT
 | 
			
		||||
  // Special case: EventResponse uses string pointer
 | 
			
		||||
  if (message_type == EventResponse::MESSAGE_TYPE) {
 | 
			
		||||
@@ -1902,149 +1860,6 @@ uint16_t APIConnection::try_send_ping_request(EntityBase *entity, APIConnection
 | 
			
		||||
  return encode_message_to_buffer(req, PingRequest::MESSAGE_TYPE, conn, remaining_size, is_single);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint16_t APIConnection::get_estimated_message_size(uint16_t message_type) {
 | 
			
		||||
  // Use generated ESTIMATED_SIZE constants from each message type
 | 
			
		||||
  switch (message_type) {
 | 
			
		||||
#ifdef USE_BINARY_SENSOR
 | 
			
		||||
    case BinarySensorStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return BinarySensorStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesBinarySensorResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesBinarySensorResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
    case SensorStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return SensorStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesSensorResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesSensorResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
    case SwitchStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return SwitchStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesSwitchResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesSwitchResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
    case TextSensorStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return TextSensorStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesTextSensorResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesTextSensorResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
    case NumberStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return NumberStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesNumberResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesNumberResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_TEXT
 | 
			
		||||
    case TextStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return TextStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesTextResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesTextResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
    case SelectStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return SelectStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesSelectResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesSelectResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_LOCK
 | 
			
		||||
    case LockStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return LockStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesLockResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesLockResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_EVENT
 | 
			
		||||
    case EventResponse::MESSAGE_TYPE:
 | 
			
		||||
      return EventResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesEventResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesEventResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_COVER
 | 
			
		||||
    case CoverStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return CoverStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesCoverResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesCoverResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_FAN
 | 
			
		||||
    case FanStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return FanStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesFanResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesFanResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_LIGHT
 | 
			
		||||
    case LightStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return LightStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesLightResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesLightResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_CLIMATE
 | 
			
		||||
    case ClimateStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ClimateStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesClimateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesClimateResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
    case ListEntitiesCameraResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesCameraResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_BUTTON
 | 
			
		||||
    case ListEntitiesButtonResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesButtonResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_MEDIA_PLAYER
 | 
			
		||||
    case MediaPlayerStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return MediaPlayerStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesMediaPlayerResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesMediaPlayerResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ALARM_CONTROL_PANEL
 | 
			
		||||
    case AlarmControlPanelStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return AlarmControlPanelStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesAlarmControlPanelResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesAlarmControlPanelResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_DATETIME_DATE
 | 
			
		||||
    case DateStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return DateStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesDateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesDateResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_DATETIME_TIME
 | 
			
		||||
    case TimeStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return TimeStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesTimeResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesTimeResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_DATETIME_DATETIME
 | 
			
		||||
    case DateTimeStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return DateTimeStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesDateTimeResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesDateTimeResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_VALVE
 | 
			
		||||
    case ValveStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ValveStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesValveResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesValveResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_UPDATE
 | 
			
		||||
    case UpdateStateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return UpdateStateResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesUpdateResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesUpdateResponse::ESTIMATED_SIZE;
 | 
			
		||||
#endif
 | 
			
		||||
    case ListEntitiesServicesResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesServicesResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case ListEntitiesDoneResponse::MESSAGE_TYPE:
 | 
			
		||||
      return ListEntitiesDoneResponse::ESTIMATED_SIZE;
 | 
			
		||||
    case DisconnectRequest::MESSAGE_TYPE:
 | 
			
		||||
      return DisconnectRequest::ESTIMATED_SIZE;
 | 
			
		||||
    default:
 | 
			
		||||
      // Fallback for unknown message types
 | 
			
		||||
      return 24;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}  // namespace api
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
#endif
 | 
			
		||||
 
 | 
			
		||||
@@ -33,7 +33,7 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
 | 
			
		||||
  bool send_list_info_done() {
 | 
			
		||||
    return this->schedule_message_(nullptr, &APIConnection::try_send_list_info_done,
 | 
			
		||||
                                   ListEntitiesDoneResponse::MESSAGE_TYPE);
 | 
			
		||||
                                   ListEntitiesDoneResponse::MESSAGE_TYPE, ListEntitiesDoneResponse::ESTIMATED_SIZE);
 | 
			
		||||
  }
 | 
			
		||||
#ifdef USE_BINARY_SENSOR
 | 
			
		||||
  bool send_binary_sensor_state(binary_sensor::BinarySensor *binary_sensor);
 | 
			
		||||
@@ -60,8 +60,8 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
  bool send_text_sensor_state(text_sensor::TextSensor *text_sensor);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
  void set_camera_state(std::shared_ptr<esp32_camera::CameraImage> image);
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
  void set_camera_state(std::shared_ptr<camera::CameraImage> image);
 | 
			
		||||
  void camera_image(const CameraImageRequest &msg) override;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_CLIMATE
 | 
			
		||||
@@ -107,7 +107,7 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
  bool send_media_player_state(media_player::MediaPlayer *media_player);
 | 
			
		||||
  void media_player_command(const MediaPlayerCommandRequest &msg) override;
 | 
			
		||||
#endif
 | 
			
		||||
  bool try_send_log_message(int level, const char *tag, const char *line);
 | 
			
		||||
  bool try_send_log_message(int level, const char *tag, const char *line, size_t message_len);
 | 
			
		||||
  void send_homeassistant_service_call(const HomeassistantServiceResponse &call) {
 | 
			
		||||
    if (!this->flags_.service_call_subscription)
 | 
			
		||||
      return;
 | 
			
		||||
@@ -195,7 +195,9 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
    // TODO
 | 
			
		||||
    return {};
 | 
			
		||||
  }
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
  void execute_service(const ExecuteServiceRequest &msg) override;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_API_NOISE
 | 
			
		||||
  NoiseEncryptionSetKeyResponse noise_encryption_set_key(const NoiseEncryptionSetKeyRequest &msg) override;
 | 
			
		||||
#endif
 | 
			
		||||
@@ -256,7 +258,7 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  bool try_to_clear_buffer(bool log_out_of_space);
 | 
			
		||||
  bool send_buffer(ProtoWriteBuffer buffer, uint16_t message_type) override;
 | 
			
		||||
  bool send_buffer(ProtoWriteBuffer buffer, uint8_t message_type) override;
 | 
			
		||||
 | 
			
		||||
  std::string get_client_combined_info() const {
 | 
			
		||||
    if (this->client_info_ == this->client_peername_) {
 | 
			
		||||
@@ -298,9 +300,14 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Non-template helper to encode any ProtoMessage
 | 
			
		||||
  static uint16_t encode_message_to_buffer(ProtoMessage &msg, uint16_t message_type, APIConnection *conn,
 | 
			
		||||
  static uint16_t encode_message_to_buffer(ProtoMessage &msg, uint8_t message_type, APIConnection *conn,
 | 
			
		||||
                                           uint32_t remaining_size, bool is_single);
 | 
			
		||||
 | 
			
		||||
#ifdef USE_VOICE_ASSISTANT
 | 
			
		||||
  // Helper to check voice assistant validity and connection ownership
 | 
			
		||||
  inline bool check_voice_assistant_api_connection_() const;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
  // Helper method to process multiple entities from an iterator in a batch
 | 
			
		||||
  template<typename Iterator> void process_iterator_batch_(Iterator &iterator) {
 | 
			
		||||
    size_t initial_size = this->deferred_batch_.size();
 | 
			
		||||
@@ -425,7 +432,7 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
  static uint16_t try_send_update_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                       bool is_single);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
  static uint16_t try_send_camera_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                       bool is_single);
 | 
			
		||||
#endif
 | 
			
		||||
@@ -438,9 +445,6 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
  static uint16_t try_send_disconnect_request(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                              bool is_single);
 | 
			
		||||
 | 
			
		||||
  // Helper function to get estimated message size for buffer pre-allocation
 | 
			
		||||
  static uint16_t get_estimated_message_size(uint16_t message_type);
 | 
			
		||||
 | 
			
		||||
  // Batch message method for ping requests
 | 
			
		||||
  static uint16_t try_send_ping_request(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
 | 
			
		||||
                                        bool is_single);
 | 
			
		||||
@@ -455,8 +459,8 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
  // These contain vectors/pointers internally, so putting them early ensures good alignment
 | 
			
		||||
  InitialStateIterator initial_state_iterator_;
 | 
			
		||||
  ListEntitiesIterator list_entities_iterator_;
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
  esp32_camera::CameraImageReader image_reader_;
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
  std::unique_ptr<camera::CameraImageReader> image_reader_;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
  // Group 3: Strings (12 bytes each on 32-bit, 4-byte aligned)
 | 
			
		||||
@@ -500,10 +504,10 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
 | 
			
		||||
    // Call operator - uses message_type to determine union type
 | 
			
		||||
    uint16_t operator()(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single,
 | 
			
		||||
                        uint16_t message_type) const;
 | 
			
		||||
                        uint8_t message_type) const;
 | 
			
		||||
 | 
			
		||||
    // Manual cleanup method - must be called before destruction for string types
 | 
			
		||||
    void cleanup(uint16_t message_type) {
 | 
			
		||||
    void cleanup(uint8_t message_type) {
 | 
			
		||||
#ifdef USE_EVENT
 | 
			
		||||
      if (message_type == EventResponse::MESSAGE_TYPE && data_.string_ptr != nullptr) {
 | 
			
		||||
        delete data_.string_ptr;
 | 
			
		||||
@@ -524,11 +528,12 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
    struct BatchItem {
 | 
			
		||||
      EntityBase *entity;      // Entity pointer
 | 
			
		||||
      MessageCreator creator;  // Function that creates the message when needed
 | 
			
		||||
      uint16_t message_type;   // Message type for overhead calculation
 | 
			
		||||
      uint8_t message_type;    // Message type for overhead calculation (max 255)
 | 
			
		||||
      uint8_t estimated_size;  // Estimated message size (max 255 bytes)
 | 
			
		||||
 | 
			
		||||
      // Constructor for creating BatchItem
 | 
			
		||||
      BatchItem(EntityBase *entity, MessageCreator creator, uint16_t message_type)
 | 
			
		||||
          : entity(entity), creator(std::move(creator)), message_type(message_type) {}
 | 
			
		||||
      BatchItem(EntityBase *entity, MessageCreator creator, uint8_t message_type, uint8_t estimated_size)
 | 
			
		||||
          : entity(entity), creator(std::move(creator)), message_type(message_type), estimated_size(estimated_size) {}
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    std::vector<BatchItem> items;
 | 
			
		||||
@@ -554,9 +559,9 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Add item to the batch
 | 
			
		||||
    void add_item(EntityBase *entity, MessageCreator creator, uint16_t message_type);
 | 
			
		||||
    void add_item(EntityBase *entity, MessageCreator creator, uint8_t message_type, uint8_t estimated_size);
 | 
			
		||||
    // Add item to the front of the batch (for high priority messages like ping)
 | 
			
		||||
    void add_item_front(EntityBase *entity, MessageCreator creator, uint16_t message_type);
 | 
			
		||||
    void add_item_front(EntityBase *entity, MessageCreator creator, uint8_t message_type, uint8_t estimated_size);
 | 
			
		||||
 | 
			
		||||
    // Clear all items with proper cleanup
 | 
			
		||||
    void clear() {
 | 
			
		||||
@@ -625,7 +630,7 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
  // to send in one go. This is the maximum size of a single packet
 | 
			
		||||
  // that can be sent over the network.
 | 
			
		||||
  // This is to avoid fragmentation of the packet.
 | 
			
		||||
  static constexpr size_t MAX_PACKET_SIZE = 1390;  // MTU
 | 
			
		||||
  static constexpr size_t MAX_BATCH_PACKET_SIZE = 1390;  // MTU
 | 
			
		||||
 | 
			
		||||
  bool schedule_batch_();
 | 
			
		||||
  void process_batch_();
 | 
			
		||||
@@ -636,9 +641,9 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
 | 
			
		||||
#ifdef HAS_PROTO_MESSAGE_DUMP
 | 
			
		||||
  // Helper to log a proto message from a MessageCreator object
 | 
			
		||||
  void log_proto_message_(EntityBase *entity, const MessageCreator &creator, uint16_t message_type) {
 | 
			
		||||
  void log_proto_message_(EntityBase *entity, const MessageCreator &creator, uint8_t message_type) {
 | 
			
		||||
    this->flags_.log_only_mode = true;
 | 
			
		||||
    creator(entity, this, MAX_PACKET_SIZE, true, message_type);
 | 
			
		||||
    creator(entity, this, MAX_BATCH_PACKET_SIZE, true, message_type);
 | 
			
		||||
    this->flags_.log_only_mode = false;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@@ -649,7 +654,8 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
  // Helper method to send a message either immediately or via batching
 | 
			
		||||
  bool send_message_smart_(EntityBase *entity, MessageCreatorPtr creator, uint16_t message_type) {
 | 
			
		||||
  bool send_message_smart_(EntityBase *entity, MessageCreatorPtr creator, uint8_t message_type,
 | 
			
		||||
                           uint8_t estimated_size) {
 | 
			
		||||
    // Try to send immediately if:
 | 
			
		||||
    // 1. We should try to send immediately (should_try_send_immediately = true)
 | 
			
		||||
    // 2. Batch delay is 0 (user has opted in to immediate sending)
 | 
			
		||||
@@ -657,7 +663,7 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
    if (this->flags_.should_try_send_immediately && this->get_batch_delay_ms_() == 0 &&
 | 
			
		||||
        this->helper_->can_write_without_blocking()) {
 | 
			
		||||
      // Now actually encode and send
 | 
			
		||||
      if (creator(entity, this, MAX_PACKET_SIZE, true) &&
 | 
			
		||||
      if (creator(entity, this, MAX_BATCH_PACKET_SIZE, true) &&
 | 
			
		||||
          this->send_buffer(ProtoWriteBuffer{&this->parent_->get_shared_buffer_ref()}, message_type)) {
 | 
			
		||||
#ifdef HAS_PROTO_MESSAGE_DUMP
 | 
			
		||||
        // Log the message in verbose mode
 | 
			
		||||
@@ -670,23 +676,25 @@ class APIConnection : public APIServerConnection {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Fall back to scheduled batching
 | 
			
		||||
    return this->schedule_message_(entity, creator, message_type);
 | 
			
		||||
    return this->schedule_message_(entity, creator, message_type, estimated_size);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Helper function to schedule a deferred message with known message type
 | 
			
		||||
  bool schedule_message_(EntityBase *entity, MessageCreator creator, uint16_t message_type) {
 | 
			
		||||
    this->deferred_batch_.add_item(entity, std::move(creator), message_type);
 | 
			
		||||
  bool schedule_message_(EntityBase *entity, MessageCreator creator, uint8_t message_type, uint8_t estimated_size) {
 | 
			
		||||
    this->deferred_batch_.add_item(entity, std::move(creator), message_type, estimated_size);
 | 
			
		||||
    return this->schedule_batch_();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Overload for function pointers (for info messages and current state reads)
 | 
			
		||||
  bool schedule_message_(EntityBase *entity, MessageCreatorPtr function_ptr, uint16_t message_type) {
 | 
			
		||||
    return schedule_message_(entity, MessageCreator(function_ptr), message_type);
 | 
			
		||||
  bool schedule_message_(EntityBase *entity, MessageCreatorPtr function_ptr, uint8_t message_type,
 | 
			
		||||
                         uint8_t estimated_size) {
 | 
			
		||||
    return schedule_message_(entity, MessageCreator(function_ptr), message_type, estimated_size);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Helper function to schedule a high priority message at the front of the batch
 | 
			
		||||
  bool schedule_message_front_(EntityBase *entity, MessageCreatorPtr function_ptr, uint16_t message_type) {
 | 
			
		||||
    this->deferred_batch_.add_item_front(entity, MessageCreator(function_ptr), message_type);
 | 
			
		||||
  bool schedule_message_front_(EntityBase *entity, MessageCreatorPtr function_ptr, uint8_t message_type,
 | 
			
		||||
                               uint8_t estimated_size) {
 | 
			
		||||
    this->deferred_batch_.add_item_front(entity, MessageCreator(function_ptr), message_type, estimated_size);
 | 
			
		||||
    return this->schedule_batch_();
 | 
			
		||||
  }
 | 
			
		||||
};
 | 
			
		||||
 
 | 
			
		||||
@@ -5,7 +5,6 @@
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
#include "proto.h"
 | 
			
		||||
#include "api_pb2_size.h"
 | 
			
		||||
#include <cstring>
 | 
			
		||||
#include <cinttypes>
 | 
			
		||||
 | 
			
		||||
@@ -225,6 +224,22 @@ APIError APIFrameHelper::init_common_() {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#define HELPER_LOG(msg, ...) ESP_LOGVV(TAG, "%s: " msg, this->info_.c_str(), ##__VA_ARGS__)
 | 
			
		||||
 | 
			
		||||
APIError APIFrameHelper::handle_socket_read_result_(ssize_t received) {
 | 
			
		||||
  if (received == -1) {
 | 
			
		||||
    if (errno == EWOULDBLOCK || errno == EAGAIN) {
 | 
			
		||||
      return APIError::WOULD_BLOCK;
 | 
			
		||||
    }
 | 
			
		||||
    state_ = State::FAILED;
 | 
			
		||||
    HELPER_LOG("Socket read failed with errno %d", errno);
 | 
			
		||||
    return APIError::SOCKET_READ_FAILED;
 | 
			
		||||
  } else if (received == 0) {
 | 
			
		||||
    state_ = State::FAILED;
 | 
			
		||||
    HELPER_LOG("Connection closed");
 | 
			
		||||
    return APIError::CONNECTION_CLOSED;
 | 
			
		||||
  }
 | 
			
		||||
  return APIError::OK;
 | 
			
		||||
}
 | 
			
		||||
// uncomment to log raw packets
 | 
			
		||||
//#define HELPER_LOG_PACKETS
 | 
			
		||||
 | 
			
		||||
@@ -327,17 +342,9 @@ APIError APINoiseFrameHelper::try_read_frame_(ParsedFrame *frame) {
 | 
			
		||||
    // no header information yet
 | 
			
		||||
    uint8_t to_read = 3 - rx_header_buf_len_;
 | 
			
		||||
    ssize_t received = this->socket_->read(&rx_header_buf_[rx_header_buf_len_], to_read);
 | 
			
		||||
    if (received == -1) {
 | 
			
		||||
      if (errno == EWOULDBLOCK || errno == EAGAIN) {
 | 
			
		||||
        return APIError::WOULD_BLOCK;
 | 
			
		||||
      }
 | 
			
		||||
      state_ = State::FAILED;
 | 
			
		||||
      HELPER_LOG("Socket read failed with errno %d", errno);
 | 
			
		||||
      return APIError::SOCKET_READ_FAILED;
 | 
			
		||||
    } else if (received == 0) {
 | 
			
		||||
      state_ = State::FAILED;
 | 
			
		||||
      HELPER_LOG("Connection closed");
 | 
			
		||||
      return APIError::CONNECTION_CLOSED;
 | 
			
		||||
    APIError err = handle_socket_read_result_(received);
 | 
			
		||||
    if (err != APIError::OK) {
 | 
			
		||||
      return err;
 | 
			
		||||
    }
 | 
			
		||||
    rx_header_buf_len_ += static_cast<uint8_t>(received);
 | 
			
		||||
    if (static_cast<uint8_t>(received) != to_read) {
 | 
			
		||||
@@ -372,17 +379,9 @@ APIError APINoiseFrameHelper::try_read_frame_(ParsedFrame *frame) {
 | 
			
		||||
    // more data to read
 | 
			
		||||
    uint16_t to_read = msg_size - rx_buf_len_;
 | 
			
		||||
    ssize_t received = this->socket_->read(&rx_buf_[rx_buf_len_], to_read);
 | 
			
		||||
    if (received == -1) {
 | 
			
		||||
      if (errno == EWOULDBLOCK || errno == EAGAIN) {
 | 
			
		||||
        return APIError::WOULD_BLOCK;
 | 
			
		||||
      }
 | 
			
		||||
      state_ = State::FAILED;
 | 
			
		||||
      HELPER_LOG("Socket read failed with errno %d", errno);
 | 
			
		||||
      return APIError::SOCKET_READ_FAILED;
 | 
			
		||||
    } else if (received == 0) {
 | 
			
		||||
      state_ = State::FAILED;
 | 
			
		||||
      HELPER_LOG("Connection closed");
 | 
			
		||||
      return APIError::CONNECTION_CLOSED;
 | 
			
		||||
    APIError err = handle_socket_read_result_(received);
 | 
			
		||||
    if (err != APIError::OK) {
 | 
			
		||||
      return err;
 | 
			
		||||
    }
 | 
			
		||||
    rx_buf_len_ += static_cast<uint16_t>(received);
 | 
			
		||||
    if (static_cast<uint16_t>(received) != to_read) {
 | 
			
		||||
@@ -613,21 +612,15 @@ APIError APINoiseFrameHelper::read_packet(ReadPacketBuffer *buffer) {
 | 
			
		||||
  buffer->type = type;
 | 
			
		||||
  return APIError::OK;
 | 
			
		||||
}
 | 
			
		||||
APIError APINoiseFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) {
 | 
			
		||||
  std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
 | 
			
		||||
  uint16_t payload_len = static_cast<uint16_t>(raw_buffer->size() - frame_header_padding_);
 | 
			
		||||
 | 
			
		||||
APIError APINoiseFrameHelper::write_protobuf_packet(uint8_t type, ProtoWriteBuffer buffer) {
 | 
			
		||||
  // Resize to include MAC space (required for Noise encryption)
 | 
			
		||||
  raw_buffer->resize(raw_buffer->size() + frame_footer_size_);
 | 
			
		||||
 | 
			
		||||
  // Use write_protobuf_packets with a single packet
 | 
			
		||||
  std::vector<PacketInfo> packets;
 | 
			
		||||
  packets.emplace_back(type, 0, payload_len);
 | 
			
		||||
 | 
			
		||||
  return write_protobuf_packets(buffer, packets);
 | 
			
		||||
  buffer.get_buffer()->resize(buffer.get_buffer()->size() + frame_footer_size_);
 | 
			
		||||
  PacketInfo packet{type, 0,
 | 
			
		||||
                    static_cast<uint16_t>(buffer.get_buffer()->size() - frame_header_padding_ - frame_footer_size_)};
 | 
			
		||||
  return write_protobuf_packets(buffer, std::span<const PacketInfo>(&packet, 1));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
APIError APINoiseFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer, const std::vector<PacketInfo> &packets) {
 | 
			
		||||
APIError APINoiseFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer, std::span<const PacketInfo> packets) {
 | 
			
		||||
  APIError aerr = state_action_();
 | 
			
		||||
  if (aerr != APIError::OK) {
 | 
			
		||||
    return aerr;
 | 
			
		||||
@@ -642,18 +635,15 @@ APIError APINoiseFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer, co
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
 | 
			
		||||
  uint8_t *buffer_data = raw_buffer->data();  // Cache buffer pointer
 | 
			
		||||
 | 
			
		||||
  this->reusable_iovs_.clear();
 | 
			
		||||
  this->reusable_iovs_.reserve(packets.size());
 | 
			
		||||
 | 
			
		||||
  // We need to encrypt each packet in place
 | 
			
		||||
  for (const auto &packet : packets) {
 | 
			
		||||
    uint16_t type = packet.message_type;
 | 
			
		||||
    uint16_t offset = packet.offset;
 | 
			
		||||
    uint16_t payload_len = packet.payload_size;
 | 
			
		||||
    uint16_t msg_len = 4 + payload_len;  // type(2) + data_len(2) + payload
 | 
			
		||||
 | 
			
		||||
    // The buffer already has padding at offset
 | 
			
		||||
    uint8_t *buf_start = raw_buffer->data() + offset;
 | 
			
		||||
    uint8_t *buf_start = buffer_data + packet.offset;
 | 
			
		||||
 | 
			
		||||
    // Write noise header
 | 
			
		||||
    buf_start[0] = 0x01;  // indicator
 | 
			
		||||
@@ -661,10 +651,10 @@ APIError APINoiseFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer, co
 | 
			
		||||
 | 
			
		||||
    // Write message header (to be encrypted)
 | 
			
		||||
    const uint8_t msg_offset = 3;
 | 
			
		||||
    buf_start[msg_offset + 0] = (uint8_t) (type >> 8);         // type high byte
 | 
			
		||||
    buf_start[msg_offset + 1] = (uint8_t) type;                // type low byte
 | 
			
		||||
    buf_start[msg_offset + 2] = (uint8_t) (payload_len >> 8);  // data_len high byte
 | 
			
		||||
    buf_start[msg_offset + 3] = (uint8_t) payload_len;         // data_len low byte
 | 
			
		||||
    buf_start[msg_offset] = static_cast<uint8_t>(packet.message_type >> 8);      // type high byte
 | 
			
		||||
    buf_start[msg_offset + 1] = static_cast<uint8_t>(packet.message_type);       // type low byte
 | 
			
		||||
    buf_start[msg_offset + 2] = static_cast<uint8_t>(packet.payload_size >> 8);  // data_len high byte
 | 
			
		||||
    buf_start[msg_offset + 3] = static_cast<uint8_t>(packet.payload_size);       // data_len low byte
 | 
			
		||||
    // payload data is already in the buffer starting at offset + 7
 | 
			
		||||
 | 
			
		||||
    // Make sure we have space for MAC
 | 
			
		||||
@@ -673,7 +663,8 @@ APIError APINoiseFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer, co
 | 
			
		||||
    // Encrypt the message in place
 | 
			
		||||
    NoiseBuffer mbuf;
 | 
			
		||||
    noise_buffer_init(mbuf);
 | 
			
		||||
    noise_buffer_set_inout(mbuf, buf_start + msg_offset, msg_len, msg_len + frame_footer_size_);
 | 
			
		||||
    noise_buffer_set_inout(mbuf, buf_start + msg_offset, 4 + packet.payload_size,
 | 
			
		||||
                           4 + packet.payload_size + frame_footer_size_);
 | 
			
		||||
 | 
			
		||||
    int err = noise_cipherstate_encrypt(send_cipher_, &mbuf);
 | 
			
		||||
    if (err != 0) {
 | 
			
		||||
@@ -683,14 +674,12 @@ APIError APINoiseFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer, co
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Fill in the encrypted size
 | 
			
		||||
    buf_start[1] = (uint8_t) (mbuf.size >> 8);
 | 
			
		||||
    buf_start[2] = (uint8_t) mbuf.size;
 | 
			
		||||
    buf_start[1] = static_cast<uint8_t>(mbuf.size >> 8);
 | 
			
		||||
    buf_start[2] = static_cast<uint8_t>(mbuf.size);
 | 
			
		||||
 | 
			
		||||
    // Add iovec for this encrypted packet
 | 
			
		||||
    struct iovec iov;
 | 
			
		||||
    iov.iov_base = buf_start;
 | 
			
		||||
    iov.iov_len = 3 + mbuf.size;  // indicator + size + encrypted data
 | 
			
		||||
    this->reusable_iovs_.push_back(iov);
 | 
			
		||||
    this->reusable_iovs_.push_back(
 | 
			
		||||
        {buf_start, static_cast<size_t>(3 + mbuf.size)});  // indicator + size + encrypted data
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Send all encrypted packets in one writev call
 | 
			
		||||
@@ -865,17 +854,9 @@ APIError APIPlaintextFrameHelper::try_read_frame_(ParsedFrame *frame) {
 | 
			
		||||
    // Try to get to at least 3 bytes total (indicator + 2 varint bytes), then read one byte at a time
 | 
			
		||||
    ssize_t received =
 | 
			
		||||
        this->socket_->read(&rx_header_buf_[rx_header_buf_pos_], rx_header_buf_pos_ < 3 ? 3 - rx_header_buf_pos_ : 1);
 | 
			
		||||
    if (received == -1) {
 | 
			
		||||
      if (errno == EWOULDBLOCK || errno == EAGAIN) {
 | 
			
		||||
        return APIError::WOULD_BLOCK;
 | 
			
		||||
      }
 | 
			
		||||
      state_ = State::FAILED;
 | 
			
		||||
      HELPER_LOG("Socket read failed with errno %d", errno);
 | 
			
		||||
      return APIError::SOCKET_READ_FAILED;
 | 
			
		||||
    } else if (received == 0) {
 | 
			
		||||
      state_ = State::FAILED;
 | 
			
		||||
      HELPER_LOG("Connection closed");
 | 
			
		||||
      return APIError::CONNECTION_CLOSED;
 | 
			
		||||
    APIError err = handle_socket_read_result_(received);
 | 
			
		||||
    if (err != APIError::OK) {
 | 
			
		||||
      return err;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // If this was the first read, validate the indicator byte
 | 
			
		||||
@@ -959,17 +940,9 @@ APIError APIPlaintextFrameHelper::try_read_frame_(ParsedFrame *frame) {
 | 
			
		||||
    // more data to read
 | 
			
		||||
    uint16_t to_read = rx_header_parsed_len_ - rx_buf_len_;
 | 
			
		||||
    ssize_t received = this->socket_->read(&rx_buf_[rx_buf_len_], to_read);
 | 
			
		||||
    if (received == -1) {
 | 
			
		||||
      if (errno == EWOULDBLOCK || errno == EAGAIN) {
 | 
			
		||||
        return APIError::WOULD_BLOCK;
 | 
			
		||||
      }
 | 
			
		||||
      state_ = State::FAILED;
 | 
			
		||||
      HELPER_LOG("Socket read failed with errno %d", errno);
 | 
			
		||||
      return APIError::SOCKET_READ_FAILED;
 | 
			
		||||
    } else if (received == 0) {
 | 
			
		||||
      state_ = State::FAILED;
 | 
			
		||||
      HELPER_LOG("Connection closed");
 | 
			
		||||
      return APIError::CONNECTION_CLOSED;
 | 
			
		||||
    APIError err = handle_socket_read_result_(received);
 | 
			
		||||
    if (err != APIError::OK) {
 | 
			
		||||
      return err;
 | 
			
		||||
    }
 | 
			
		||||
    rx_buf_len_ += static_cast<uint16_t>(received);
 | 
			
		||||
    if (static_cast<uint16_t>(received) != to_read) {
 | 
			
		||||
@@ -1028,19 +1001,12 @@ APIError APIPlaintextFrameHelper::read_packet(ReadPacketBuffer *buffer) {
 | 
			
		||||
  buffer->type = rx_header_parsed_type_;
 | 
			
		||||
  return APIError::OK;
 | 
			
		||||
}
 | 
			
		||||
APIError APIPlaintextFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) {
 | 
			
		||||
  std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
 | 
			
		||||
  uint16_t payload_len = static_cast<uint16_t>(raw_buffer->size() - frame_header_padding_);
 | 
			
		||||
 | 
			
		||||
  // Use write_protobuf_packets with a single packet
 | 
			
		||||
  std::vector<PacketInfo> packets;
 | 
			
		||||
  packets.emplace_back(type, 0, payload_len);
 | 
			
		||||
 | 
			
		||||
  return write_protobuf_packets(buffer, packets);
 | 
			
		||||
APIError APIPlaintextFrameHelper::write_protobuf_packet(uint8_t type, ProtoWriteBuffer buffer) {
 | 
			
		||||
  PacketInfo packet{type, 0, static_cast<uint16_t>(buffer.get_buffer()->size() - frame_header_padding_)};
 | 
			
		||||
  return write_protobuf_packets(buffer, std::span<const PacketInfo>(&packet, 1));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
APIError APIPlaintextFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer,
 | 
			
		||||
                                                         const std::vector<PacketInfo> &packets) {
 | 
			
		||||
APIError APIPlaintextFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer, std::span<const PacketInfo> packets) {
 | 
			
		||||
  if (state_ != State::DATA) {
 | 
			
		||||
    return APIError::BAD_STATE;
 | 
			
		||||
  }
 | 
			
		||||
@@ -1050,17 +1016,15 @@ APIError APIPlaintextFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
 | 
			
		||||
  uint8_t *buffer_data = raw_buffer->data();  // Cache buffer pointer
 | 
			
		||||
 | 
			
		||||
  this->reusable_iovs_.clear();
 | 
			
		||||
  this->reusable_iovs_.reserve(packets.size());
 | 
			
		||||
 | 
			
		||||
  for (const auto &packet : packets) {
 | 
			
		||||
    uint16_t type = packet.message_type;
 | 
			
		||||
    uint16_t offset = packet.offset;
 | 
			
		||||
    uint16_t payload_len = packet.payload_size;
 | 
			
		||||
 | 
			
		||||
    // Calculate varint sizes for header layout
 | 
			
		||||
    uint8_t size_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(payload_len));
 | 
			
		||||
    uint8_t type_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(type));
 | 
			
		||||
    uint8_t size_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(packet.payload_size));
 | 
			
		||||
    uint8_t type_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(packet.message_type));
 | 
			
		||||
    uint8_t total_header_len = 1 + size_varint_len + type_varint_len;
 | 
			
		||||
 | 
			
		||||
    // Calculate where to start writing the header
 | 
			
		||||
@@ -1088,23 +1052,20 @@ APIError APIPlaintextFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer
 | 
			
		||||
    //
 | 
			
		||||
    // The message starts at offset + frame_header_padding_
 | 
			
		||||
    // So we write the header starting at offset + frame_header_padding_ - total_header_len
 | 
			
		||||
    uint8_t *buf_start = raw_buffer->data() + offset;
 | 
			
		||||
    uint8_t *buf_start = buffer_data + packet.offset;
 | 
			
		||||
    uint32_t header_offset = frame_header_padding_ - total_header_len;
 | 
			
		||||
 | 
			
		||||
    // Write the plaintext header
 | 
			
		||||
    buf_start[header_offset] = 0x00;  // indicator
 | 
			
		||||
 | 
			
		||||
    // Encode size varint directly into buffer
 | 
			
		||||
    ProtoVarInt(payload_len).encode_to_buffer_unchecked(buf_start + header_offset + 1, size_varint_len);
 | 
			
		||||
 | 
			
		||||
    // Encode type varint directly into buffer
 | 
			
		||||
    ProtoVarInt(type).encode_to_buffer_unchecked(buf_start + header_offset + 1 + size_varint_len, type_varint_len);
 | 
			
		||||
    // Encode varints directly into buffer
 | 
			
		||||
    ProtoVarInt(packet.payload_size).encode_to_buffer_unchecked(buf_start + header_offset + 1, size_varint_len);
 | 
			
		||||
    ProtoVarInt(packet.message_type)
 | 
			
		||||
        .encode_to_buffer_unchecked(buf_start + header_offset + 1 + size_varint_len, type_varint_len);
 | 
			
		||||
 | 
			
		||||
    // Add iovec for this packet (header + payload)
 | 
			
		||||
    struct iovec iov;
 | 
			
		||||
    iov.iov_base = buf_start + header_offset;
 | 
			
		||||
    iov.iov_len = total_header_len + payload_len;
 | 
			
		||||
    this->reusable_iovs_.push_back(iov);
 | 
			
		||||
    this->reusable_iovs_.push_back(
 | 
			
		||||
        {buf_start + header_offset, static_cast<size_t>(total_header_len + packet.payload_size)});
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Send all packets in one writev call
 | 
			
		||||
 
 | 
			
		||||
@@ -2,6 +2,7 @@
 | 
			
		||||
#include <cstdint>
 | 
			
		||||
#include <deque>
 | 
			
		||||
#include <limits>
 | 
			
		||||
#include <span>
 | 
			
		||||
#include <utility>
 | 
			
		||||
#include <vector>
 | 
			
		||||
 | 
			
		||||
@@ -29,13 +30,11 @@ struct ReadPacketBuffer {
 | 
			
		||||
 | 
			
		||||
// Packed packet info structure to minimize memory usage
 | 
			
		||||
struct PacketInfo {
 | 
			
		||||
  uint16_t message_type;  // 2 bytes
 | 
			
		||||
  uint16_t offset;        // 2 bytes (sufficient for packet size ~1460 bytes)
 | 
			
		||||
  uint16_t payload_size;  // 2 bytes (up to 65535 bytes)
 | 
			
		||||
  uint16_t padding;       // 2 byte (for alignment)
 | 
			
		||||
  uint16_t offset;        // Offset in buffer where message starts
 | 
			
		||||
  uint16_t payload_size;  // Size of the message payload
 | 
			
		||||
  uint8_t message_type;   // Message type (0-255)
 | 
			
		||||
 | 
			
		||||
  PacketInfo(uint16_t type, uint16_t off, uint16_t size)
 | 
			
		||||
      : message_type(type), offset(off), payload_size(size), padding(0) {}
 | 
			
		||||
  PacketInfo(uint8_t type, uint16_t off, uint16_t size) : offset(off), payload_size(size), message_type(type) {}
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum class APIError : uint16_t {
 | 
			
		||||
@@ -97,11 +96,11 @@ class APIFrameHelper {
 | 
			
		||||
  }
 | 
			
		||||
  // Give this helper a name for logging
 | 
			
		||||
  void set_log_info(std::string info) { info_ = std::move(info); }
 | 
			
		||||
  virtual APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) = 0;
 | 
			
		||||
  virtual APIError write_protobuf_packet(uint8_t type, ProtoWriteBuffer buffer) = 0;
 | 
			
		||||
  // Write multiple protobuf packets in a single operation
 | 
			
		||||
  // packets contains (message_type, offset, length) for each message in the buffer
 | 
			
		||||
  // The buffer contains all messages with appropriate padding before each
 | 
			
		||||
  virtual APIError write_protobuf_packets(ProtoWriteBuffer buffer, const std::vector<PacketInfo> &packets) = 0;
 | 
			
		||||
  virtual APIError write_protobuf_packets(ProtoWriteBuffer buffer, std::span<const PacketInfo> packets) = 0;
 | 
			
		||||
  // Get the frame header padding required by this protocol
 | 
			
		||||
  virtual uint8_t frame_header_padding() = 0;
 | 
			
		||||
  // Get the frame footer size required by this protocol
 | 
			
		||||
@@ -175,6 +174,9 @@ class APIFrameHelper {
 | 
			
		||||
 | 
			
		||||
  // Common initialization for both plaintext and noise protocols
 | 
			
		||||
  APIError init_common_();
 | 
			
		||||
 | 
			
		||||
  // Helper method to handle socket read results
 | 
			
		||||
  APIError handle_socket_read_result_(ssize_t received);
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
#ifdef USE_API_NOISE
 | 
			
		||||
@@ -193,8 +195,8 @@ class APINoiseFrameHelper : public APIFrameHelper {
 | 
			
		||||
  APIError init() override;
 | 
			
		||||
  APIError loop() override;
 | 
			
		||||
  APIError read_packet(ReadPacketBuffer *buffer) override;
 | 
			
		||||
  APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) override;
 | 
			
		||||
  APIError write_protobuf_packets(ProtoWriteBuffer buffer, const std::vector<PacketInfo> &packets) override;
 | 
			
		||||
  APIError write_protobuf_packet(uint8_t type, ProtoWriteBuffer buffer) override;
 | 
			
		||||
  APIError write_protobuf_packets(ProtoWriteBuffer buffer, std::span<const PacketInfo> packets) override;
 | 
			
		||||
  // Get the frame header padding required by this protocol
 | 
			
		||||
  uint8_t frame_header_padding() override { return frame_header_padding_; }
 | 
			
		||||
  // Get the frame footer size required by this protocol
 | 
			
		||||
@@ -247,8 +249,8 @@ class APIPlaintextFrameHelper : public APIFrameHelper {
 | 
			
		||||
  APIError init() override;
 | 
			
		||||
  APIError loop() override;
 | 
			
		||||
  APIError read_packet(ReadPacketBuffer *buffer) override;
 | 
			
		||||
  APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) override;
 | 
			
		||||
  APIError write_protobuf_packets(ProtoWriteBuffer buffer, const std::vector<PacketInfo> &packets) override;
 | 
			
		||||
  APIError write_protobuf_packet(uint8_t type, ProtoWriteBuffer buffer) override;
 | 
			
		||||
  APIError write_protobuf_packets(ProtoWriteBuffer buffer, std::span<const PacketInfo> packets) override;
 | 
			
		||||
  uint8_t frame_header_padding() override { return frame_header_padding_; }
 | 
			
		||||
  // Get the frame footer size required by this protocol
 | 
			
		||||
  uint8_t frame_footer_size() override { return frame_footer_size_; }
 | 
			
		||||
 
 | 
			
		||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							@@ -195,6 +195,7 @@ void APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type,
 | 
			
		||||
      this->on_home_assistant_state_response(msg);
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
    case 42: {
 | 
			
		||||
      ExecuteServiceRequest msg;
 | 
			
		||||
      msg.decode(msg_data, msg_size);
 | 
			
		||||
@@ -204,7 +205,8 @@ void APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type,
 | 
			
		||||
      this->on_execute_service_request(msg);
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
    case 45: {
 | 
			
		||||
      CameraImageRequest msg;
 | 
			
		||||
      msg.decode(msg_data, msg_size);
 | 
			
		||||
@@ -660,11 +662,13 @@ void APIServerConnection::on_get_time_request(const GetTimeRequest &msg) {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
void APIServerConnection::on_execute_service_request(const ExecuteServiceRequest &msg) {
 | 
			
		||||
  if (this->check_authenticated_()) {
 | 
			
		||||
    this->execute_service(msg);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_API_NOISE
 | 
			
		||||
void APIServerConnection::on_noise_encryption_set_key_request(const NoiseEncryptionSetKeyRequest &msg) {
 | 
			
		||||
  if (this->check_authenticated_()) {
 | 
			
		||||
@@ -682,7 +686,7 @@ void APIServerConnection::on_button_command_request(const ButtonCommandRequest &
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
void APIServerConnection::on_camera_image_request(const CameraImageRequest &msg) {
 | 
			
		||||
  if (this->check_authenticated_()) {
 | 
			
		||||
    this->camera_image(msg);
 | 
			
		||||
 
 | 
			
		||||
@@ -69,9 +69,11 @@ class APIServerConnectionBase : public ProtoService {
 | 
			
		||||
  virtual void on_get_time_request(const GetTimeRequest &value){};
 | 
			
		||||
  virtual void on_get_time_response(const GetTimeResponse &value){};
 | 
			
		||||
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
  virtual void on_execute_service_request(const ExecuteServiceRequest &value){};
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
  virtual void on_camera_image_request(const CameraImageRequest &value){};
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
@@ -216,14 +218,16 @@ class APIServerConnection : public APIServerConnectionBase {
 | 
			
		||||
  virtual void subscribe_homeassistant_services(const SubscribeHomeassistantServicesRequest &msg) = 0;
 | 
			
		||||
  virtual void subscribe_home_assistant_states(const SubscribeHomeAssistantStatesRequest &msg) = 0;
 | 
			
		||||
  virtual GetTimeResponse get_time(const GetTimeRequest &msg) = 0;
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
  virtual void execute_service(const ExecuteServiceRequest &msg) = 0;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_API_NOISE
 | 
			
		||||
  virtual NoiseEncryptionSetKeyResponse noise_encryption_set_key(const NoiseEncryptionSetKeyRequest &msg) = 0;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_BUTTON
 | 
			
		||||
  virtual void button_command(const ButtonCommandRequest &msg) = 0;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
  virtual void camera_image(const CameraImageRequest &msg) = 0;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_CLIMATE
 | 
			
		||||
@@ -333,14 +337,16 @@ class APIServerConnection : public APIServerConnectionBase {
 | 
			
		||||
  void on_subscribe_homeassistant_services_request(const SubscribeHomeassistantServicesRequest &msg) override;
 | 
			
		||||
  void on_subscribe_home_assistant_states_request(const SubscribeHomeAssistantStatesRequest &msg) override;
 | 
			
		||||
  void on_get_time_request(const GetTimeRequest &msg) override;
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
  void on_execute_service_request(const ExecuteServiceRequest &msg) override;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_API_NOISE
 | 
			
		||||
  void on_noise_encryption_set_key_request(const NoiseEncryptionSetKeyRequest &msg) override;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_BUTTON
 | 
			
		||||
  void on_button_command_request(const ButtonCommandRequest &msg) override;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
  void on_camera_image_request(const CameraImageRequest &msg) override;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_CLIMATE
 | 
			
		||||
 
 | 
			
		||||
@@ -1,359 +0,0 @@
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "proto.h"
 | 
			
		||||
#include <cstdint>
 | 
			
		||||
#include <string>
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace api {
 | 
			
		||||
 | 
			
		||||
class ProtoSize {
 | 
			
		||||
 public:
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief ProtoSize class for Protocol Buffer serialization size calculation
 | 
			
		||||
   *
 | 
			
		||||
   * This class provides static methods to calculate the exact byte counts needed
 | 
			
		||||
   * for encoding various Protocol Buffer field types. All methods are designed to be
 | 
			
		||||
   * efficient for the common case where many fields have default values.
 | 
			
		||||
   *
 | 
			
		||||
   * Implements Protocol Buffer encoding size calculation according to:
 | 
			
		||||
   * https://protobuf.dev/programming-guides/encoding/
 | 
			
		||||
   *
 | 
			
		||||
   * Key features:
 | 
			
		||||
   * - Early-return optimization for zero/default values
 | 
			
		||||
   * - Direct total_size updates to avoid unnecessary additions
 | 
			
		||||
   * - Specialized handling for different field types according to protobuf spec
 | 
			
		||||
   * - Templated helpers for repeated fields and messages
 | 
			
		||||
   */
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates the size in bytes needed to encode a uint32_t value as a varint
 | 
			
		||||
   *
 | 
			
		||||
   * @param value The uint32_t value to calculate size for
 | 
			
		||||
   * @return The number of bytes needed to encode the value
 | 
			
		||||
   */
 | 
			
		||||
  static inline uint32_t varint(uint32_t value) {
 | 
			
		||||
    // Optimized varint size calculation using leading zeros
 | 
			
		||||
    // Each 7 bits requires one byte in the varint encoding
 | 
			
		||||
    if (value < 128)
 | 
			
		||||
      return 1;  // 7 bits, common case for small values
 | 
			
		||||
 | 
			
		||||
    // For larger values, count bytes needed based on the position of the highest bit set
 | 
			
		||||
    if (value < 16384) {
 | 
			
		||||
      return 2;  // 14 bits
 | 
			
		||||
    } else if (value < 2097152) {
 | 
			
		||||
      return 3;  // 21 bits
 | 
			
		||||
    } else if (value < 268435456) {
 | 
			
		||||
      return 4;  // 28 bits
 | 
			
		||||
    } else {
 | 
			
		||||
      return 5;  // 32 bits (maximum for uint32_t)
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates the size in bytes needed to encode a uint64_t value as a varint
 | 
			
		||||
   *
 | 
			
		||||
   * @param value The uint64_t value to calculate size for
 | 
			
		||||
   * @return The number of bytes needed to encode the value
 | 
			
		||||
   */
 | 
			
		||||
  static inline uint32_t varint(uint64_t value) {
 | 
			
		||||
    // Handle common case of values fitting in uint32_t (vast majority of use cases)
 | 
			
		||||
    if (value <= UINT32_MAX) {
 | 
			
		||||
      return varint(static_cast<uint32_t>(value));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // For larger values, determine size based on highest bit position
 | 
			
		||||
    if (value < (1ULL << 35)) {
 | 
			
		||||
      return 5;  // 35 bits
 | 
			
		||||
    } else if (value < (1ULL << 42)) {
 | 
			
		||||
      return 6;  // 42 bits
 | 
			
		||||
    } else if (value < (1ULL << 49)) {
 | 
			
		||||
      return 7;  // 49 bits
 | 
			
		||||
    } else if (value < (1ULL << 56)) {
 | 
			
		||||
      return 8;  // 56 bits
 | 
			
		||||
    } else if (value < (1ULL << 63)) {
 | 
			
		||||
      return 9;  // 63 bits
 | 
			
		||||
    } else {
 | 
			
		||||
      return 10;  // 64 bits (maximum for uint64_t)
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates the size in bytes needed to encode an int32_t value as a varint
 | 
			
		||||
   *
 | 
			
		||||
   * Special handling is needed for negative values, which are sign-extended to 64 bits
 | 
			
		||||
   * in Protocol Buffers, resulting in a 10-byte varint.
 | 
			
		||||
   *
 | 
			
		||||
   * @param value The int32_t value to calculate size for
 | 
			
		||||
   * @return The number of bytes needed to encode the value
 | 
			
		||||
   */
 | 
			
		||||
  static inline uint32_t varint(int32_t value) {
 | 
			
		||||
    // Negative values are sign-extended to 64 bits in protocol buffers,
 | 
			
		||||
    // which always results in a 10-byte varint for negative int32
 | 
			
		||||
    if (value < 0) {
 | 
			
		||||
      return 10;  // Negative int32 is always 10 bytes long
 | 
			
		||||
    }
 | 
			
		||||
    // For non-negative values, use the uint32_t implementation
 | 
			
		||||
    return varint(static_cast<uint32_t>(value));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates the size in bytes needed to encode an int64_t value as a varint
 | 
			
		||||
   *
 | 
			
		||||
   * @param value The int64_t value to calculate size for
 | 
			
		||||
   * @return The number of bytes needed to encode the value
 | 
			
		||||
   */
 | 
			
		||||
  static inline uint32_t varint(int64_t value) {
 | 
			
		||||
    // For int64_t, we convert to uint64_t and calculate the size
 | 
			
		||||
    // This works because the bit pattern determines the encoding size,
 | 
			
		||||
    // and we've handled negative int32 values as a special case above
 | 
			
		||||
    return varint(static_cast<uint64_t>(value));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates the size in bytes needed to encode a field ID and wire type
 | 
			
		||||
   *
 | 
			
		||||
   * @param field_id The field identifier
 | 
			
		||||
   * @param type The wire type value (from the WireType enum in the protobuf spec)
 | 
			
		||||
   * @return The number of bytes needed to encode the field ID and wire type
 | 
			
		||||
   */
 | 
			
		||||
  static inline uint32_t field(uint32_t field_id, uint32_t type) {
 | 
			
		||||
    uint32_t tag = (field_id << 3) | (type & 0b111);
 | 
			
		||||
    return varint(tag);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Common parameters for all add_*_field methods
 | 
			
		||||
   *
 | 
			
		||||
   * All add_*_field methods follow these common patterns:
 | 
			
		||||
   *
 | 
			
		||||
   * @param total_size Reference to the total message size to update
 | 
			
		||||
   * @param field_id_size Pre-calculated size of the field ID in bytes
 | 
			
		||||
   * @param value The value to calculate size for (type varies)
 | 
			
		||||
   * @param force Whether to calculate size even if the value is default/zero/empty
 | 
			
		||||
   *
 | 
			
		||||
   * Each method follows this implementation pattern:
 | 
			
		||||
   * 1. Skip calculation if value is default (0, false, empty) and not forced
 | 
			
		||||
   * 2. Calculate the size based on the field's encoding rules
 | 
			
		||||
   * 3. Add the field_id_size + calculated value size to total_size
 | 
			
		||||
   */
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of an int32 field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_int32_field(uint32_t &total_size, uint32_t field_id_size, int32_t value, bool force = false) {
 | 
			
		||||
    // Skip calculation if value is zero and not forced
 | 
			
		||||
    if (value == 0 && !force) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    if (value < 0) {
 | 
			
		||||
      // Negative values are encoded as 10-byte varints in protobuf
 | 
			
		||||
      total_size += field_id_size + 10;
 | 
			
		||||
    } else {
 | 
			
		||||
      // For non-negative values, use the standard varint size
 | 
			
		||||
      total_size += field_id_size + varint(static_cast<uint32_t>(value));
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a uint32 field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_uint32_field(uint32_t &total_size, uint32_t field_id_size, uint32_t value,
 | 
			
		||||
                                      bool force = false) {
 | 
			
		||||
    // Skip calculation if value is zero and not forced
 | 
			
		||||
    if (value == 0 && !force) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a boolean field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_bool_field(uint32_t &total_size, uint32_t field_id_size, bool value, bool force = false) {
 | 
			
		||||
    // Skip calculation if value is false and not forced
 | 
			
		||||
    if (!value && !force) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Boolean fields always use 1 byte when true
 | 
			
		||||
    total_size += field_id_size + 1;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a fixed field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * Fixed fields always take exactly N bytes (4 for fixed32/float, 8 for fixed64/double).
 | 
			
		||||
   *
 | 
			
		||||
   * @tparam NumBytes The number of bytes for this fixed field (4 or 8)
 | 
			
		||||
   * @param is_nonzero Whether the value is non-zero
 | 
			
		||||
   */
 | 
			
		||||
  template<uint32_t NumBytes>
 | 
			
		||||
  static inline void add_fixed_field(uint32_t &total_size, uint32_t field_id_size, bool is_nonzero,
 | 
			
		||||
                                     bool force = false) {
 | 
			
		||||
    // Skip calculation if value is zero and not forced
 | 
			
		||||
    if (!is_nonzero && !force) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Fixed fields always take exactly NumBytes
 | 
			
		||||
    total_size += field_id_size + NumBytes;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of an enum field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * Enum fields are encoded as uint32 varints.
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_enum_field(uint32_t &total_size, uint32_t field_id_size, uint32_t value, bool force = false) {
 | 
			
		||||
    // Skip calculation if value is zero and not forced
 | 
			
		||||
    if (value == 0 && !force) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Enums are encoded as uint32
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a sint32 field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * Sint32 fields use ZigZag encoding, which is more efficient for negative values.
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_sint32_field(uint32_t &total_size, uint32_t field_id_size, int32_t value, bool force = false) {
 | 
			
		||||
    // Skip calculation if value is zero and not forced
 | 
			
		||||
    if (value == 0 && !force) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // ZigZag encoding for sint32: (n << 1) ^ (n >> 31)
 | 
			
		||||
    uint32_t zigzag = (static_cast<uint32_t>(value) << 1) ^ (static_cast<uint32_t>(value >> 31));
 | 
			
		||||
    total_size += field_id_size + varint(zigzag);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of an int64 field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_int64_field(uint32_t &total_size, uint32_t field_id_size, int64_t value, bool force = false) {
 | 
			
		||||
    // Skip calculation if value is zero and not forced
 | 
			
		||||
    if (value == 0 && !force) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a uint64 field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_uint64_field(uint32_t &total_size, uint32_t field_id_size, uint64_t value,
 | 
			
		||||
                                      bool force = false) {
 | 
			
		||||
    // Skip calculation if value is zero and not forced
 | 
			
		||||
    if (value == 0 && !force) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a sint64 field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * Sint64 fields use ZigZag encoding, which is more efficient for negative values.
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_sint64_field(uint32_t &total_size, uint32_t field_id_size, int64_t value, bool force = false) {
 | 
			
		||||
    // Skip calculation if value is zero and not forced
 | 
			
		||||
    if (value == 0 && !force) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // ZigZag encoding for sint64: (n << 1) ^ (n >> 63)
 | 
			
		||||
    uint64_t zigzag = (static_cast<uint64_t>(value) << 1) ^ (static_cast<uint64_t>(value >> 63));
 | 
			
		||||
    total_size += field_id_size + varint(zigzag);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a string/bytes field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_string_field(uint32_t &total_size, uint32_t field_id_size, const std::string &str,
 | 
			
		||||
                                      bool force = false) {
 | 
			
		||||
    // Skip calculation if string is empty and not forced
 | 
			
		||||
    if (str.empty() && !force) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    const uint32_t str_size = static_cast<uint32_t>(str.size());
 | 
			
		||||
    total_size += field_id_size + varint(str_size) + str_size;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a nested message field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * This helper function directly updates the total_size reference if the nested size
 | 
			
		||||
   * is greater than zero or force is true.
 | 
			
		||||
   *
 | 
			
		||||
   * @param nested_size The pre-calculated size of the nested message
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_message_field(uint32_t &total_size, uint32_t field_id_size, uint32_t nested_size,
 | 
			
		||||
                                       bool force = false) {
 | 
			
		||||
    // Skip calculation if nested message is empty and not forced
 | 
			
		||||
    if (nested_size == 0 && !force) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    // Field ID + length varint + nested message content
 | 
			
		||||
    total_size += field_id_size + varint(nested_size) + nested_size;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a nested message field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * This version takes a ProtoMessage object, calculates its size internally,
 | 
			
		||||
   * and updates the total_size reference. This eliminates the need for a temporary variable
 | 
			
		||||
   * at the call site.
 | 
			
		||||
   *
 | 
			
		||||
   * @param message The nested message object
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_message_object(uint32_t &total_size, uint32_t field_id_size, const ProtoMessage &message,
 | 
			
		||||
                                        bool force = false) {
 | 
			
		||||
    uint32_t nested_size = 0;
 | 
			
		||||
    message.calculate_size(nested_size);
 | 
			
		||||
 | 
			
		||||
    // Use the base implementation with the calculated nested_size
 | 
			
		||||
    add_message_field(total_size, field_id_size, nested_size, force);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the sizes of all messages in a repeated field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * This helper processes a vector of message objects, calculating the size for each message
 | 
			
		||||
   * and adding it to the total size.
 | 
			
		||||
   *
 | 
			
		||||
   * @tparam MessageType The type of the nested messages in the vector
 | 
			
		||||
   * @param messages Vector of message objects
 | 
			
		||||
   */
 | 
			
		||||
  template<typename MessageType>
 | 
			
		||||
  static inline void add_repeated_message(uint32_t &total_size, uint32_t field_id_size,
 | 
			
		||||
                                          const std::vector<MessageType> &messages) {
 | 
			
		||||
    // Skip if the vector is empty
 | 
			
		||||
    if (messages.empty()) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // For repeated fields, always use force=true
 | 
			
		||||
    for (const auto &message : messages) {
 | 
			
		||||
      add_message_object(total_size, field_id_size, message, true);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace api
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -96,30 +96,30 @@ void APIServer::setup() {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_LOGGER
 | 
			
		||||
  if (logger::global_logger != nullptr) {
 | 
			
		||||
    logger::global_logger->add_on_log_callback([this](int level, const char *tag, const char *message) {
 | 
			
		||||
      if (this->shutting_down_) {
 | 
			
		||||
        // Don't try to send logs during shutdown
 | 
			
		||||
        // as it could result in a recursion and
 | 
			
		||||
        // we would be filling a buffer we are trying to clear
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
      for (auto &c : this->clients_) {
 | 
			
		||||
        if (!c->flags_.remove)
 | 
			
		||||
          c->try_send_log_message(level, tag, message);
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
    logger::global_logger->add_on_log_callback(
 | 
			
		||||
        [this](int level, const char *tag, const char *message, size_t message_len) {
 | 
			
		||||
          if (this->shutting_down_) {
 | 
			
		||||
            // Don't try to send logs during shutdown
 | 
			
		||||
            // as it could result in a recursion and
 | 
			
		||||
            // we would be filling a buffer we are trying to clear
 | 
			
		||||
            return;
 | 
			
		||||
          }
 | 
			
		||||
          for (auto &c : this->clients_) {
 | 
			
		||||
            if (!c->flags_.remove)
 | 
			
		||||
              c->try_send_log_message(level, tag, message, message_len);
 | 
			
		||||
          }
 | 
			
		||||
        });
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
  if (esp32_camera::global_esp32_camera != nullptr && !esp32_camera::global_esp32_camera->is_internal()) {
 | 
			
		||||
    esp32_camera::global_esp32_camera->add_image_callback(
 | 
			
		||||
        [this](const std::shared_ptr<esp32_camera::CameraImage> &image) {
 | 
			
		||||
          for (auto &c : this->clients_) {
 | 
			
		||||
            if (!c->flags_.remove)
 | 
			
		||||
              c->set_camera_state(image);
 | 
			
		||||
          }
 | 
			
		||||
        });
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
  if (camera::Camera::instance() != nullptr && !camera::Camera::instance()->is_internal()) {
 | 
			
		||||
    camera::Camera::instance()->add_image_callback([this](const std::shared_ptr<camera::CameraImage> &image) {
 | 
			
		||||
      for (auto &c : this->clients_) {
 | 
			
		||||
        if (!c->flags_.remove)
 | 
			
		||||
          c->set_camera_state(image);
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
@@ -253,180 +253,114 @@ bool APIServer::check_password(const std::string &password) const {
 | 
			
		||||
 | 
			
		||||
void APIServer::handle_disconnect(APIConnection *conn) {}
 | 
			
		||||
 | 
			
		||||
// Macro for entities without extra parameters
 | 
			
		||||
#define API_DISPATCH_UPDATE(entity_type, entity_name) \
 | 
			
		||||
  void APIServer::on_##entity_name##_update(entity_type *obj) { /* NOLINT(bugprone-macro-parentheses) */ \
 | 
			
		||||
    if (obj->is_internal()) \
 | 
			
		||||
      return; \
 | 
			
		||||
    for (auto &c : this->clients_) \
 | 
			
		||||
      c->send_##entity_name##_state(obj); \
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
// Macro for entities with extra parameters (but parameters not used in send)
 | 
			
		||||
#define API_DISPATCH_UPDATE_IGNORE_PARAMS(entity_type, entity_name, ...) \
 | 
			
		||||
  void APIServer::on_##entity_name##_update(entity_type *obj, __VA_ARGS__) { /* NOLINT(bugprone-macro-parentheses) */ \
 | 
			
		||||
    if (obj->is_internal()) \
 | 
			
		||||
      return; \
 | 
			
		||||
    for (auto &c : this->clients_) \
 | 
			
		||||
      c->send_##entity_name##_state(obj); \
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
#ifdef USE_BINARY_SENSOR
 | 
			
		||||
void APIServer::on_binary_sensor_update(binary_sensor::BinarySensor *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_binary_sensor_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(binary_sensor::BinarySensor, binary_sensor)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_COVER
 | 
			
		||||
void APIServer::on_cover_update(cover::Cover *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_cover_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(cover::Cover, cover)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_FAN
 | 
			
		||||
void APIServer::on_fan_update(fan::Fan *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_fan_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(fan::Fan, fan)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_LIGHT
 | 
			
		||||
void APIServer::on_light_update(light::LightState *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_light_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(light::LightState, light)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
void APIServer::on_sensor_update(sensor::Sensor *obj, float state) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_sensor_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE_IGNORE_PARAMS(sensor::Sensor, sensor, float state)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
void APIServer::on_switch_update(switch_::Switch *obj, bool state) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_switch_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE_IGNORE_PARAMS(switch_::Switch, switch, bool state)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
void APIServer::on_text_sensor_update(text_sensor::TextSensor *obj, const std::string &state) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_text_sensor_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE_IGNORE_PARAMS(text_sensor::TextSensor, text_sensor, const std::string &state)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_CLIMATE
 | 
			
		||||
void APIServer::on_climate_update(climate::Climate *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_climate_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(climate::Climate, climate)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
void APIServer::on_number_update(number::Number *obj, float state) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_number_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE_IGNORE_PARAMS(number::Number, number, float state)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_DATETIME_DATE
 | 
			
		||||
void APIServer::on_date_update(datetime::DateEntity *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_date_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(datetime::DateEntity, date)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_DATETIME_TIME
 | 
			
		||||
void APIServer::on_time_update(datetime::TimeEntity *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_time_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(datetime::TimeEntity, time)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_DATETIME_DATETIME
 | 
			
		||||
void APIServer::on_datetime_update(datetime::DateTimeEntity *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_datetime_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(datetime::DateTimeEntity, datetime)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_TEXT
 | 
			
		||||
void APIServer::on_text_update(text::Text *obj, const std::string &state) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_text_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE_IGNORE_PARAMS(text::Text, text, const std::string &state)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
void APIServer::on_select_update(select::Select *obj, const std::string &state, size_t index) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_select_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE_IGNORE_PARAMS(select::Select, select, const std::string &state, size_t index)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_LOCK
 | 
			
		||||
void APIServer::on_lock_update(lock::Lock *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_lock_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(lock::Lock, lock)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_VALVE
 | 
			
		||||
void APIServer::on_valve_update(valve::Valve *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_valve_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(valve::Valve, valve)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_MEDIA_PLAYER
 | 
			
		||||
void APIServer::on_media_player_update(media_player::MediaPlayer *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_media_player_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(media_player::MediaPlayer, media_player)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_EVENT
 | 
			
		||||
// Event is a special case - it's the only entity that passes extra parameters to the send method
 | 
			
		||||
void APIServer::on_event(event::Event *obj, const std::string &event_type) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_event(obj, event_type);
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_UPDATE
 | 
			
		||||
// Update is a special case - the method is called on_update, not on_update_update
 | 
			
		||||
void APIServer::on_update(update::UpdateEntity *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_update_state(obj);
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ALARM_CONTROL_PANEL
 | 
			
		||||
void APIServer::on_alarm_control_panel_update(alarm_control_panel::AlarmControlPanel *obj) {
 | 
			
		||||
  if (obj->is_internal())
 | 
			
		||||
    return;
 | 
			
		||||
  for (auto &c : this->clients_)
 | 
			
		||||
    c->send_alarm_control_panel_state(obj);
 | 
			
		||||
}
 | 
			
		||||
API_DISPATCH_UPDATE(alarm_control_panel::AlarmControlPanel, alarm_control_panel)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
float APIServer::get_setup_priority() const { return setup_priority::AFTER_WIFI; }
 | 
			
		||||
@@ -533,7 +467,8 @@ void APIServer::on_shutdown() {
 | 
			
		||||
    if (!c->send_message(DisconnectRequest())) {
 | 
			
		||||
      // If we can't send the disconnect request directly (tx_buffer full),
 | 
			
		||||
      // schedule it at the front of the batch so it will be sent with priority
 | 
			
		||||
      c->schedule_message_front_(nullptr, &APIConnection::try_send_disconnect_request, DisconnectRequest::MESSAGE_TYPE);
 | 
			
		||||
      c->schedule_message_front_(nullptr, &APIConnection::try_send_disconnect_request, DisconnectRequest::MESSAGE_TYPE,
 | 
			
		||||
                                 DisconnectRequest::ESTIMATED_SIZE);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -12,7 +12,9 @@
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
#include "list_entities.h"
 | 
			
		||||
#include "subscribe_state.h"
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
#include "user_services.h"
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#include <vector>
 | 
			
		||||
 | 
			
		||||
@@ -107,18 +109,9 @@ class APIServer : public Component, public Controller {
 | 
			
		||||
  void on_media_player_update(media_player::MediaPlayer *obj) override;
 | 
			
		||||
#endif
 | 
			
		||||
  void send_homeassistant_service_call(const HomeassistantServiceResponse &call);
 | 
			
		||||
  void register_user_service(UserServiceDescriptor *descriptor) {
 | 
			
		||||
#ifdef USE_API_YAML_SERVICES
 | 
			
		||||
    // Vector is pre-allocated when services are defined in YAML
 | 
			
		||||
    this->user_services_.push_back(descriptor);
 | 
			
		||||
#else
 | 
			
		||||
    // Lazy allocate vector on first use for CustomAPIDevice
 | 
			
		||||
    if (!this->user_services_) {
 | 
			
		||||
      this->user_services_ = std::make_unique<std::vector<UserServiceDescriptor *>>();
 | 
			
		||||
    }
 | 
			
		||||
    this->user_services_->push_back(descriptor);
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
  void register_user_service(UserServiceDescriptor *descriptor) { this->user_services_.push_back(descriptor); }
 | 
			
		||||
#endif
 | 
			
		||||
  }
 | 
			
		||||
#ifdef USE_HOMEASSISTANT_TIME
 | 
			
		||||
  void request_time();
 | 
			
		||||
#endif
 | 
			
		||||
@@ -147,14 +140,9 @@ class APIServer : public Component, public Controller {
 | 
			
		||||
  void get_home_assistant_state(std::string entity_id, optional<std::string> attribute,
 | 
			
		||||
                                std::function<void(std::string)> f);
 | 
			
		||||
  const std::vector<HomeAssistantStateSubscription> &get_state_subs() const;
 | 
			
		||||
  const std::vector<UserServiceDescriptor *> &get_user_services() const {
 | 
			
		||||
#ifdef USE_API_YAML_SERVICES
 | 
			
		||||
    return this->user_services_;
 | 
			
		||||
#else
 | 
			
		||||
    static const std::vector<UserServiceDescriptor *> EMPTY;
 | 
			
		||||
    return this->user_services_ ? *this->user_services_ : EMPTY;
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
  const std::vector<UserServiceDescriptor *> &get_user_services() const { return this->user_services_; }
 | 
			
		||||
#endif
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
#ifdef USE_API_CLIENT_CONNECTED_TRIGGER
 | 
			
		||||
  Trigger<std::string, std::string> *get_client_connected_trigger() const { return this->client_connected_trigger_; }
 | 
			
		||||
@@ -186,14 +174,8 @@ class APIServer : public Component, public Controller {
 | 
			
		||||
#endif
 | 
			
		||||
  std::vector<uint8_t> shared_write_buffer_;  // Shared proto write buffer for all connections
 | 
			
		||||
  std::vector<HomeAssistantStateSubscription> state_subs_;
 | 
			
		||||
#ifdef USE_API_YAML_SERVICES
 | 
			
		||||
  // When services are defined in YAML, we know at compile time that services will be registered
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
  std::vector<UserServiceDescriptor *> user_services_;
 | 
			
		||||
#else
 | 
			
		||||
  // Services can still be registered at runtime by CustomAPIDevice components even when not
 | 
			
		||||
  // defined in YAML. Using unique_ptr allows lazy allocation, saving 12 bytes in the common
 | 
			
		||||
  // case where no services (YAML or custom) are used.
 | 
			
		||||
  std::unique_ptr<std::vector<UserServiceDescriptor *>> user_services_;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
  // Group smaller types together
 | 
			
		||||
 
 | 
			
		||||
@@ -3,10 +3,13 @@
 | 
			
		||||
#include <map>
 | 
			
		||||
#include "api_server.h"
 | 
			
		||||
#ifdef USE_API
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
#include "user_services.h"
 | 
			
		||||
#endif
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace api {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
template<typename T, typename... Ts> class CustomAPIDeviceService : public UserServiceBase<Ts...> {
 | 
			
		||||
 public:
 | 
			
		||||
  CustomAPIDeviceService(const std::string &name, const std::array<std::string, sizeof...(Ts)> &arg_names, T *obj,
 | 
			
		||||
@@ -19,6 +22,7 @@ template<typename T, typename... Ts> class CustomAPIDeviceService : public UserS
 | 
			
		||||
  T *obj_;
 | 
			
		||||
  void (T::*callback_)(Ts...);
 | 
			
		||||
};
 | 
			
		||||
#endif  // USE_API_SERVICES
 | 
			
		||||
 | 
			
		||||
class CustomAPIDevice {
 | 
			
		||||
 public:
 | 
			
		||||
@@ -46,12 +50,14 @@ class CustomAPIDevice {
 | 
			
		||||
   * @param name The name of the service to register.
 | 
			
		||||
   * @param arg_names The name of the arguments for the service, must match the arguments of the function.
 | 
			
		||||
   */
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
  template<typename T, typename... Ts>
 | 
			
		||||
  void register_service(void (T::*callback)(Ts...), const std::string &name,
 | 
			
		||||
                        const std::array<std::string, sizeof...(Ts)> &arg_names) {
 | 
			
		||||
    auto *service = new CustomAPIDeviceService<T, Ts...>(name, arg_names, (T *) this, callback);  // NOLINT
 | 
			
		||||
    global_api_server->register_user_service(service);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
  /** Register a custom native API service that will show up in Home Assistant.
 | 
			
		||||
   *
 | 
			
		||||
@@ -71,10 +77,12 @@ class CustomAPIDevice {
 | 
			
		||||
   * @param callback The member function to call when the service is triggered.
 | 
			
		||||
   * @param name The name of the arguments for the service, must match the arguments of the function.
 | 
			
		||||
   */
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
  template<typename T> void register_service(void (T::*callback)(), const std::string &name) {
 | 
			
		||||
    auto *service = new CustomAPIDeviceService<T>(name, {}, (T *) this, callback);  // NOLINT
 | 
			
		||||
    global_api_server->register_user_service(service);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
  /** Subscribe to the state (or attribute state) of an entity from Home Assistant.
 | 
			
		||||
   *
 | 
			
		||||
 
 | 
			
		||||
@@ -40,8 +40,8 @@ LIST_ENTITIES_HANDLER(lock, lock::Lock, ListEntitiesLockResponse)
 | 
			
		||||
#ifdef USE_VALVE
 | 
			
		||||
LIST_ENTITIES_HANDLER(valve, valve::Valve, ListEntitiesValveResponse)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
LIST_ENTITIES_HANDLER(camera, esp32_camera::ESP32Camera, ListEntitiesCameraResponse)
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
LIST_ENTITIES_HANDLER(camera, camera::Camera, ListEntitiesCameraResponse)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_CLIMATE
 | 
			
		||||
LIST_ENTITIES_HANDLER(climate, climate::Climate, ListEntitiesClimateResponse)
 | 
			
		||||
@@ -83,10 +83,12 @@ bool ListEntitiesIterator::on_end() { return this->client_->send_list_info_done(
 | 
			
		||||
 | 
			
		||||
ListEntitiesIterator::ListEntitiesIterator(APIConnection *client) : client_(client) {}
 | 
			
		||||
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
bool ListEntitiesIterator::on_service(UserServiceDescriptor *service) {
 | 
			
		||||
  auto resp = service->encode_list_service_response();
 | 
			
		||||
  return this->client_->send_message(resp);
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
}  // namespace api
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
 
 | 
			
		||||
@@ -14,7 +14,7 @@ class APIConnection;
 | 
			
		||||
#define LIST_ENTITIES_HANDLER(entity_type, EntityClass, ResponseType) \
 | 
			
		||||
  bool ListEntitiesIterator::on_##entity_type(EntityClass *entity) { /* NOLINT(bugprone-macro-parentheses) */ \
 | 
			
		||||
    return this->client_->schedule_message_(entity, &APIConnection::try_send_##entity_type##_info, \
 | 
			
		||||
                                            ResponseType::MESSAGE_TYPE); \
 | 
			
		||||
                                            ResponseType::MESSAGE_TYPE, ResponseType::ESTIMATED_SIZE); \
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
class ListEntitiesIterator : public ComponentIterator {
 | 
			
		||||
@@ -44,9 +44,11 @@ class ListEntitiesIterator : public ComponentIterator {
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
  bool on_text_sensor(text_sensor::TextSensor *entity) override;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
  bool on_service(UserServiceDescriptor *service) override;
 | 
			
		||||
#ifdef USE_ESP32_CAMERA
 | 
			
		||||
  bool on_camera(esp32_camera::ESP32Camera *entity) override;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_CAMERA
 | 
			
		||||
  bool on_camera(camera::Camera *entity) override;
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_CLIMATE
 | 
			
		||||
  bool on_climate(climate::Climate *entity) override;
 | 
			
		||||
 
 | 
			
		||||
@@ -4,6 +4,7 @@
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
#include <cassert>
 | 
			
		||||
#include <vector>
 | 
			
		||||
 | 
			
		||||
#ifdef ESPHOME_LOG_HAS_VERY_VERBOSE
 | 
			
		||||
@@ -59,7 +60,6 @@ class ProtoVarInt {
 | 
			
		||||
  uint32_t as_uint32() const { return this->value_; }
 | 
			
		||||
  uint64_t as_uint64() const { return this->value_; }
 | 
			
		||||
  bool as_bool() const { return this->value_; }
 | 
			
		||||
  template<typename T> T as_enum() const { return static_cast<T>(this->as_uint32()); }
 | 
			
		||||
  int32_t as_int32() const {
 | 
			
		||||
    // Not ZigZag encoded
 | 
			
		||||
    return static_cast<int32_t>(this->as_int64());
 | 
			
		||||
@@ -133,15 +133,24 @@ class ProtoVarInt {
 | 
			
		||||
  uint64_t value_;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// Forward declaration for decode_to_message and encode_to_writer
 | 
			
		||||
class ProtoMessage;
 | 
			
		||||
 | 
			
		||||
class ProtoLengthDelimited {
 | 
			
		||||
 public:
 | 
			
		||||
  explicit ProtoLengthDelimited(const uint8_t *value, size_t length) : value_(value), length_(length) {}
 | 
			
		||||
  std::string as_string() const { return std::string(reinterpret_cast<const char *>(this->value_), this->length_); }
 | 
			
		||||
  template<class C> C as_message() const {
 | 
			
		||||
    auto msg = C();
 | 
			
		||||
    msg.decode(this->value_, this->length_);
 | 
			
		||||
    return msg;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * Decode the length-delimited data into an existing ProtoMessage instance.
 | 
			
		||||
   *
 | 
			
		||||
   * This method allows decoding without templates, enabling use in contexts
 | 
			
		||||
   * where the message type is not known at compile time. The ProtoMessage's
 | 
			
		||||
   * decode() method will be called with the raw data and length.
 | 
			
		||||
   *
 | 
			
		||||
   * @param msg The ProtoMessage instance to decode into
 | 
			
		||||
   */
 | 
			
		||||
  void decode_to_message(ProtoMessage &msg) const;
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  const uint8_t *const value_;
 | 
			
		||||
@@ -263,9 +272,6 @@ class ProtoWriteBuffer {
 | 
			
		||||
    this->write((value >> 48) & 0xFF);
 | 
			
		||||
    this->write((value >> 56) & 0xFF);
 | 
			
		||||
  }
 | 
			
		||||
  template<typename T> void encode_enum(uint32_t field_id, T value, bool force = false) {
 | 
			
		||||
    this->encode_uint32(field_id, static_cast<uint32_t>(value), force);
 | 
			
		||||
  }
 | 
			
		||||
  void encode_float(uint32_t field_id, float value, bool force = false) {
 | 
			
		||||
    if (value == 0.0f && !force)
 | 
			
		||||
      return;
 | 
			
		||||
@@ -306,18 +312,7 @@ class ProtoWriteBuffer {
 | 
			
		||||
    }
 | 
			
		||||
    this->encode_uint64(field_id, uvalue, force);
 | 
			
		||||
  }
 | 
			
		||||
  template<class C> void encode_message(uint32_t field_id, const C &value, bool force = false) {
 | 
			
		||||
    this->encode_field_raw(field_id, 2);  // type 2: Length-delimited message
 | 
			
		||||
    size_t begin = this->buffer_->size();
 | 
			
		||||
 | 
			
		||||
    value.encode(*this);
 | 
			
		||||
 | 
			
		||||
    const uint32_t nested_length = this->buffer_->size() - begin;
 | 
			
		||||
    // add size varint
 | 
			
		||||
    std::vector<uint8_t> var;
 | 
			
		||||
    ProtoVarInt(nested_length).encode(var);
 | 
			
		||||
    this->buffer_->insert(this->buffer_->begin() + begin, var.begin(), var.end());
 | 
			
		||||
  }
 | 
			
		||||
  void encode_message(uint32_t field_id, const ProtoMessage &value, bool force = false);
 | 
			
		||||
  std::vector<uint8_t> *get_buffer() const { return buffer_; }
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
@@ -345,6 +340,494 @@ class ProtoMessage {
 | 
			
		||||
  virtual bool decode_64bit(uint32_t field_id, Proto64Bit value) { return false; }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
class ProtoSize {
 | 
			
		||||
 public:
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief ProtoSize class for Protocol Buffer serialization size calculation
 | 
			
		||||
   *
 | 
			
		||||
   * This class provides static methods to calculate the exact byte counts needed
 | 
			
		||||
   * for encoding various Protocol Buffer field types. All methods are designed to be
 | 
			
		||||
   * efficient for the common case where many fields have default values.
 | 
			
		||||
   *
 | 
			
		||||
   * Implements Protocol Buffer encoding size calculation according to:
 | 
			
		||||
   * https://protobuf.dev/programming-guides/encoding/
 | 
			
		||||
   *
 | 
			
		||||
   * Key features:
 | 
			
		||||
   * - Early-return optimization for zero/default values
 | 
			
		||||
   * - Direct total_size updates to avoid unnecessary additions
 | 
			
		||||
   * - Specialized handling for different field types according to protobuf spec
 | 
			
		||||
   * - Templated helpers for repeated fields and messages
 | 
			
		||||
   */
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates the size in bytes needed to encode a uint32_t value as a varint
 | 
			
		||||
   *
 | 
			
		||||
   * @param value The uint32_t value to calculate size for
 | 
			
		||||
   * @return The number of bytes needed to encode the value
 | 
			
		||||
   */
 | 
			
		||||
  static inline uint32_t varint(uint32_t value) {
 | 
			
		||||
    // Optimized varint size calculation using leading zeros
 | 
			
		||||
    // Each 7 bits requires one byte in the varint encoding
 | 
			
		||||
    if (value < 128)
 | 
			
		||||
      return 1;  // 7 bits, common case for small values
 | 
			
		||||
 | 
			
		||||
    // For larger values, count bytes needed based on the position of the highest bit set
 | 
			
		||||
    if (value < 16384) {
 | 
			
		||||
      return 2;  // 14 bits
 | 
			
		||||
    } else if (value < 2097152) {
 | 
			
		||||
      return 3;  // 21 bits
 | 
			
		||||
    } else if (value < 268435456) {
 | 
			
		||||
      return 4;  // 28 bits
 | 
			
		||||
    } else {
 | 
			
		||||
      return 5;  // 32 bits (maximum for uint32_t)
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates the size in bytes needed to encode a uint64_t value as a varint
 | 
			
		||||
   *
 | 
			
		||||
   * @param value The uint64_t value to calculate size for
 | 
			
		||||
   * @return The number of bytes needed to encode the value
 | 
			
		||||
   */
 | 
			
		||||
  static inline uint32_t varint(uint64_t value) {
 | 
			
		||||
    // Handle common case of values fitting in uint32_t (vast majority of use cases)
 | 
			
		||||
    if (value <= UINT32_MAX) {
 | 
			
		||||
      return varint(static_cast<uint32_t>(value));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // For larger values, determine size based on highest bit position
 | 
			
		||||
    if (value < (1ULL << 35)) {
 | 
			
		||||
      return 5;  // 35 bits
 | 
			
		||||
    } else if (value < (1ULL << 42)) {
 | 
			
		||||
      return 6;  // 42 bits
 | 
			
		||||
    } else if (value < (1ULL << 49)) {
 | 
			
		||||
      return 7;  // 49 bits
 | 
			
		||||
    } else if (value < (1ULL << 56)) {
 | 
			
		||||
      return 8;  // 56 bits
 | 
			
		||||
    } else if (value < (1ULL << 63)) {
 | 
			
		||||
      return 9;  // 63 bits
 | 
			
		||||
    } else {
 | 
			
		||||
      return 10;  // 64 bits (maximum for uint64_t)
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates the size in bytes needed to encode an int32_t value as a varint
 | 
			
		||||
   *
 | 
			
		||||
   * Special handling is needed for negative values, which are sign-extended to 64 bits
 | 
			
		||||
   * in Protocol Buffers, resulting in a 10-byte varint.
 | 
			
		||||
   *
 | 
			
		||||
   * @param value The int32_t value to calculate size for
 | 
			
		||||
   * @return The number of bytes needed to encode the value
 | 
			
		||||
   */
 | 
			
		||||
  static inline uint32_t varint(int32_t value) {
 | 
			
		||||
    // Negative values are sign-extended to 64 bits in protocol buffers,
 | 
			
		||||
    // which always results in a 10-byte varint for negative int32
 | 
			
		||||
    if (value < 0) {
 | 
			
		||||
      return 10;  // Negative int32 is always 10 bytes long
 | 
			
		||||
    }
 | 
			
		||||
    // For non-negative values, use the uint32_t implementation
 | 
			
		||||
    return varint(static_cast<uint32_t>(value));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates the size in bytes needed to encode an int64_t value as a varint
 | 
			
		||||
   *
 | 
			
		||||
   * @param value The int64_t value to calculate size for
 | 
			
		||||
   * @return The number of bytes needed to encode the value
 | 
			
		||||
   */
 | 
			
		||||
  static inline uint32_t varint(int64_t value) {
 | 
			
		||||
    // For int64_t, we convert to uint64_t and calculate the size
 | 
			
		||||
    // This works because the bit pattern determines the encoding size,
 | 
			
		||||
    // and we've handled negative int32 values as a special case above
 | 
			
		||||
    return varint(static_cast<uint64_t>(value));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates the size in bytes needed to encode a field ID and wire type
 | 
			
		||||
   *
 | 
			
		||||
   * @param field_id The field identifier
 | 
			
		||||
   * @param type The wire type value (from the WireType enum in the protobuf spec)
 | 
			
		||||
   * @return The number of bytes needed to encode the field ID and wire type
 | 
			
		||||
   */
 | 
			
		||||
  static inline uint32_t field(uint32_t field_id, uint32_t type) {
 | 
			
		||||
    uint32_t tag = (field_id << 3) | (type & 0b111);
 | 
			
		||||
    return varint(tag);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Common parameters for all add_*_field methods
 | 
			
		||||
   *
 | 
			
		||||
   * All add_*_field methods follow these common patterns:
 | 
			
		||||
   *
 | 
			
		||||
   * @param total_size Reference to the total message size to update
 | 
			
		||||
   * @param field_id_size Pre-calculated size of the field ID in bytes
 | 
			
		||||
   * @param value The value to calculate size for (type varies)
 | 
			
		||||
   * @param force Whether to calculate size even if the value is default/zero/empty
 | 
			
		||||
   *
 | 
			
		||||
   * Each method follows this implementation pattern:
 | 
			
		||||
   * 1. Skip calculation if value is default (0, false, empty) and not forced
 | 
			
		||||
   * 2. Calculate the size based on the field's encoding rules
 | 
			
		||||
   * 3. Add the field_id_size + calculated value size to total_size
 | 
			
		||||
   */
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of an int32 field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_int32_field(uint32_t &total_size, uint32_t field_id_size, int32_t value) {
 | 
			
		||||
    // Skip calculation if value is zero
 | 
			
		||||
    if (value == 0) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    if (value < 0) {
 | 
			
		||||
      // Negative values are encoded as 10-byte varints in protobuf
 | 
			
		||||
      total_size += field_id_size + 10;
 | 
			
		||||
    } else {
 | 
			
		||||
      // For non-negative values, use the standard varint size
 | 
			
		||||
      total_size += field_id_size + varint(static_cast<uint32_t>(value));
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of an int32 field to the total message size (repeated field version)
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_int32_field_repeated(uint32_t &total_size, uint32_t field_id_size, int32_t value) {
 | 
			
		||||
    // Always calculate size for repeated fields
 | 
			
		||||
    if (value < 0) {
 | 
			
		||||
      // Negative values are encoded as 10-byte varints in protobuf
 | 
			
		||||
      total_size += field_id_size + 10;
 | 
			
		||||
    } else {
 | 
			
		||||
      // For non-negative values, use the standard varint size
 | 
			
		||||
      total_size += field_id_size + varint(static_cast<uint32_t>(value));
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a uint32 field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_uint32_field(uint32_t &total_size, uint32_t field_id_size, uint32_t value) {
 | 
			
		||||
    // Skip calculation if value is zero
 | 
			
		||||
    if (value == 0) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a uint32 field to the total message size (repeated field version)
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_uint32_field_repeated(uint32_t &total_size, uint32_t field_id_size, uint32_t value) {
 | 
			
		||||
    // Always calculate size for repeated fields
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a boolean field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_bool_field(uint32_t &total_size, uint32_t field_id_size, bool value) {
 | 
			
		||||
    // Skip calculation if value is false
 | 
			
		||||
    if (!value) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Boolean fields always use 1 byte when true
 | 
			
		||||
    total_size += field_id_size + 1;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a boolean field to the total message size (repeated field version)
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_bool_field_repeated(uint32_t &total_size, uint32_t field_id_size, bool value) {
 | 
			
		||||
    // Always calculate size for repeated fields
 | 
			
		||||
    // Boolean fields always use 1 byte
 | 
			
		||||
    total_size += field_id_size + 1;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a fixed field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * Fixed fields always take exactly N bytes (4 for fixed32/float, 8 for fixed64/double).
 | 
			
		||||
   *
 | 
			
		||||
   * @tparam NumBytes The number of bytes for this fixed field (4 or 8)
 | 
			
		||||
   * @param is_nonzero Whether the value is non-zero
 | 
			
		||||
   */
 | 
			
		||||
  template<uint32_t NumBytes>
 | 
			
		||||
  static inline void add_fixed_field(uint32_t &total_size, uint32_t field_id_size, bool is_nonzero) {
 | 
			
		||||
    // Skip calculation if value is zero
 | 
			
		||||
    if (!is_nonzero) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Fixed fields always take exactly NumBytes
 | 
			
		||||
    total_size += field_id_size + NumBytes;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of an enum field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * Enum fields are encoded as uint32 varints.
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_enum_field(uint32_t &total_size, uint32_t field_id_size, uint32_t value) {
 | 
			
		||||
    // Skip calculation if value is zero
 | 
			
		||||
    if (value == 0) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Enums are encoded as uint32
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of an enum field to the total message size (repeated field version)
 | 
			
		||||
   *
 | 
			
		||||
   * Enum fields are encoded as uint32 varints.
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_enum_field_repeated(uint32_t &total_size, uint32_t field_id_size, uint32_t value) {
 | 
			
		||||
    // Always calculate size for repeated fields
 | 
			
		||||
    // Enums are encoded as uint32
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a sint32 field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * Sint32 fields use ZigZag encoding, which is more efficient for negative values.
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_sint32_field(uint32_t &total_size, uint32_t field_id_size, int32_t value) {
 | 
			
		||||
    // Skip calculation if value is zero
 | 
			
		||||
    if (value == 0) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // ZigZag encoding for sint32: (n << 1) ^ (n >> 31)
 | 
			
		||||
    uint32_t zigzag = (static_cast<uint32_t>(value) << 1) ^ (static_cast<uint32_t>(value >> 31));
 | 
			
		||||
    total_size += field_id_size + varint(zigzag);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a sint32 field to the total message size (repeated field version)
 | 
			
		||||
   *
 | 
			
		||||
   * Sint32 fields use ZigZag encoding, which is more efficient for negative values.
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_sint32_field_repeated(uint32_t &total_size, uint32_t field_id_size, int32_t value) {
 | 
			
		||||
    // Always calculate size for repeated fields
 | 
			
		||||
    // ZigZag encoding for sint32: (n << 1) ^ (n >> 31)
 | 
			
		||||
    uint32_t zigzag = (static_cast<uint32_t>(value) << 1) ^ (static_cast<uint32_t>(value >> 31));
 | 
			
		||||
    total_size += field_id_size + varint(zigzag);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of an int64 field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_int64_field(uint32_t &total_size, uint32_t field_id_size, int64_t value) {
 | 
			
		||||
    // Skip calculation if value is zero
 | 
			
		||||
    if (value == 0) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of an int64 field to the total message size (repeated field version)
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_int64_field_repeated(uint32_t &total_size, uint32_t field_id_size, int64_t value) {
 | 
			
		||||
    // Always calculate size for repeated fields
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a uint64 field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_uint64_field(uint32_t &total_size, uint32_t field_id_size, uint64_t value) {
 | 
			
		||||
    // Skip calculation if value is zero
 | 
			
		||||
    if (value == 0) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a uint64 field to the total message size (repeated field version)
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_uint64_field_repeated(uint32_t &total_size, uint32_t field_id_size, uint64_t value) {
 | 
			
		||||
    // Always calculate size for repeated fields
 | 
			
		||||
    total_size += field_id_size + varint(value);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a sint64 field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * Sint64 fields use ZigZag encoding, which is more efficient for negative values.
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_sint64_field(uint32_t &total_size, uint32_t field_id_size, int64_t value) {
 | 
			
		||||
    // Skip calculation if value is zero
 | 
			
		||||
    if (value == 0) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // ZigZag encoding for sint64: (n << 1) ^ (n >> 63)
 | 
			
		||||
    uint64_t zigzag = (static_cast<uint64_t>(value) << 1) ^ (static_cast<uint64_t>(value >> 63));
 | 
			
		||||
    total_size += field_id_size + varint(zigzag);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a sint64 field to the total message size (repeated field version)
 | 
			
		||||
   *
 | 
			
		||||
   * Sint64 fields use ZigZag encoding, which is more efficient for negative values.
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_sint64_field_repeated(uint32_t &total_size, uint32_t field_id_size, int64_t value) {
 | 
			
		||||
    // Always calculate size for repeated fields
 | 
			
		||||
    // ZigZag encoding for sint64: (n << 1) ^ (n >> 63)
 | 
			
		||||
    uint64_t zigzag = (static_cast<uint64_t>(value) << 1) ^ (static_cast<uint64_t>(value >> 63));
 | 
			
		||||
    total_size += field_id_size + varint(zigzag);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a string/bytes field to the total message size
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_string_field(uint32_t &total_size, uint32_t field_id_size, const std::string &str) {
 | 
			
		||||
    // Skip calculation if string is empty
 | 
			
		||||
    if (str.empty()) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    const uint32_t str_size = static_cast<uint32_t>(str.size());
 | 
			
		||||
    total_size += field_id_size + varint(str_size) + str_size;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a string/bytes field to the total message size (repeated field version)
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_string_field_repeated(uint32_t &total_size, uint32_t field_id_size, const std::string &str) {
 | 
			
		||||
    // Always calculate size for repeated fields
 | 
			
		||||
    const uint32_t str_size = static_cast<uint32_t>(str.size());
 | 
			
		||||
    total_size += field_id_size + varint(str_size) + str_size;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a nested message field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * This helper function directly updates the total_size reference if the nested size
 | 
			
		||||
   * is greater than zero.
 | 
			
		||||
   *
 | 
			
		||||
   * @param nested_size The pre-calculated size of the nested message
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_message_field(uint32_t &total_size, uint32_t field_id_size, uint32_t nested_size) {
 | 
			
		||||
    // Skip calculation if nested message is empty
 | 
			
		||||
    if (nested_size == 0) {
 | 
			
		||||
      return;  // No need to update total_size
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Calculate and directly add to total_size
 | 
			
		||||
    // Field ID + length varint + nested message content
 | 
			
		||||
    total_size += field_id_size + varint(nested_size) + nested_size;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a nested message field to the total message size (repeated field version)
 | 
			
		||||
   *
 | 
			
		||||
   * @param nested_size The pre-calculated size of the nested message
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_message_field_repeated(uint32_t &total_size, uint32_t field_id_size, uint32_t nested_size) {
 | 
			
		||||
    // Always calculate size for repeated fields
 | 
			
		||||
    // Field ID + length varint + nested message content
 | 
			
		||||
    total_size += field_id_size + varint(nested_size) + nested_size;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a nested message field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * This version takes a ProtoMessage object, calculates its size internally,
 | 
			
		||||
   * and updates the total_size reference. This eliminates the need for a temporary variable
 | 
			
		||||
   * at the call site.
 | 
			
		||||
   *
 | 
			
		||||
   * @param message The nested message object
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_message_object(uint32_t &total_size, uint32_t field_id_size, const ProtoMessage &message) {
 | 
			
		||||
    uint32_t nested_size = 0;
 | 
			
		||||
    message.calculate_size(nested_size);
 | 
			
		||||
 | 
			
		||||
    // Use the base implementation with the calculated nested_size
 | 
			
		||||
    add_message_field(total_size, field_id_size, nested_size);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the size of a nested message field to the total message size (repeated field version)
 | 
			
		||||
   *
 | 
			
		||||
   * @param message The nested message object
 | 
			
		||||
   */
 | 
			
		||||
  static inline void add_message_object_repeated(uint32_t &total_size, uint32_t field_id_size,
 | 
			
		||||
                                                 const ProtoMessage &message) {
 | 
			
		||||
    uint32_t nested_size = 0;
 | 
			
		||||
    message.calculate_size(nested_size);
 | 
			
		||||
 | 
			
		||||
    // Use the base implementation with the calculated nested_size
 | 
			
		||||
    add_message_field_repeated(total_size, field_id_size, nested_size);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * @brief Calculates and adds the sizes of all messages in a repeated field to the total message size
 | 
			
		||||
   *
 | 
			
		||||
   * This helper processes a vector of message objects, calculating the size for each message
 | 
			
		||||
   * and adding it to the total size.
 | 
			
		||||
   *
 | 
			
		||||
   * @tparam MessageType The type of the nested messages in the vector
 | 
			
		||||
   * @param messages Vector of message objects
 | 
			
		||||
   */
 | 
			
		||||
  template<typename MessageType>
 | 
			
		||||
  static inline void add_repeated_message(uint32_t &total_size, uint32_t field_id_size,
 | 
			
		||||
                                          const std::vector<MessageType> &messages) {
 | 
			
		||||
    // Skip if the vector is empty
 | 
			
		||||
    if (messages.empty()) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Use the repeated field version for all messages
 | 
			
		||||
    for (const auto &message : messages) {
 | 
			
		||||
      add_message_object_repeated(total_size, field_id_size, message);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// Implementation of encode_message - must be after ProtoMessage is defined
 | 
			
		||||
inline void ProtoWriteBuffer::encode_message(uint32_t field_id, const ProtoMessage &value, bool force) {
 | 
			
		||||
  this->encode_field_raw(field_id, 2);  // type 2: Length-delimited message
 | 
			
		||||
 | 
			
		||||
  // Calculate the message size first
 | 
			
		||||
  uint32_t msg_length_bytes = 0;
 | 
			
		||||
  value.calculate_size(msg_length_bytes);
 | 
			
		||||
 | 
			
		||||
  // Calculate how many bytes the length varint needs
 | 
			
		||||
  uint32_t varint_length_bytes = ProtoSize::varint(msg_length_bytes);
 | 
			
		||||
 | 
			
		||||
  // Reserve exact space for the length varint
 | 
			
		||||
  size_t begin = this->buffer_->size();
 | 
			
		||||
  this->buffer_->resize(this->buffer_->size() + varint_length_bytes);
 | 
			
		||||
 | 
			
		||||
  // Write the length varint directly
 | 
			
		||||
  ProtoVarInt(msg_length_bytes).encode_to_buffer_unchecked(this->buffer_->data() + begin, varint_length_bytes);
 | 
			
		||||
 | 
			
		||||
  // Now encode the message content - it will append to the buffer
 | 
			
		||||
  value.encode(*this);
 | 
			
		||||
 | 
			
		||||
  // Verify that the encoded size matches what we calculated
 | 
			
		||||
  assert(this->buffer_->size() == begin + varint_length_bytes + msg_length_bytes);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Implementation of decode_to_message - must be after ProtoMessage is defined
 | 
			
		||||
inline void ProtoLengthDelimited::decode_to_message(ProtoMessage &msg) const {
 | 
			
		||||
  msg.decode(this->value_, this->length_);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
template<typename T> const char *proto_enum_to_string(T value);
 | 
			
		||||
 | 
			
		||||
class ProtoService {
 | 
			
		||||
@@ -363,11 +846,11 @@ class ProtoService {
 | 
			
		||||
   * @return A ProtoWriteBuffer object with the reserved size.
 | 
			
		||||
   */
 | 
			
		||||
  virtual ProtoWriteBuffer create_buffer(uint32_t reserve_size) = 0;
 | 
			
		||||
  virtual bool send_buffer(ProtoWriteBuffer buffer, uint16_t message_type) = 0;
 | 
			
		||||
  virtual bool send_buffer(ProtoWriteBuffer buffer, uint8_t message_type) = 0;
 | 
			
		||||
  virtual void read_message(uint32_t msg_size, uint32_t msg_type, uint8_t *msg_data) = 0;
 | 
			
		||||
 | 
			
		||||
  // Optimized method that pre-allocates buffer based on message size
 | 
			
		||||
  bool send_message_(const ProtoMessage &msg, uint16_t message_type) {
 | 
			
		||||
  bool send_message_(const ProtoMessage &msg, uint8_t message_type) {
 | 
			
		||||
    uint32_t msg_size = 0;
 | 
			
		||||
    msg.calculate_size(msg_size);
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -7,6 +7,7 @@
 | 
			
		||||
#include "esphome/core/automation.h"
 | 
			
		||||
#include "api_pb2.h"
 | 
			
		||||
 | 
			
		||||
#ifdef USE_API_SERVICES
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace api {
 | 
			
		||||
 | 
			
		||||
@@ -73,3 +74,4 @@ template<typename... Ts> class UserServiceTrigger : public UserServiceBase<Ts...
 | 
			
		||||
 | 
			
		||||
}  // namespace api
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
#endif  // USE_API_SERVICES
 | 
			
		||||
 
 | 
			
		||||
@@ -52,11 +52,21 @@ bool BluetoothProxy::parse_device(const esp32_ble_tracker::ESPBTDevice &device)
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static constexpr size_t FLUSH_BATCH_SIZE = 8;
 | 
			
		||||
static std::vector<api::BluetoothLERawAdvertisement> &get_batch_buffer() {
 | 
			
		||||
  static std::vector<api::BluetoothLERawAdvertisement> batch_buffer;
 | 
			
		||||
  return batch_buffer;
 | 
			
		||||
}
 | 
			
		||||
// Batch size for BLE advertisements to maximize WiFi efficiency
 | 
			
		||||
// Each advertisement is up to 80 bytes when packaged (including protocol overhead)
 | 
			
		||||
// Most advertisements are 20-30 bytes, allowing even more to fit per packet
 | 
			
		||||
// 16 advertisements × 80 bytes (worst case) = 1280 bytes out of ~1320 bytes usable payload
 | 
			
		||||
// This achieves ~97% WiFi MTU utilization while staying under the limit
 | 
			
		||||
static constexpr size_t FLUSH_BATCH_SIZE = 16;
 | 
			
		||||
 | 
			
		||||
namespace {
 | 
			
		||||
// Batch buffer in anonymous namespace to avoid guard variable (saves 8 bytes)
 | 
			
		||||
// This is initialized at program startup before any threads
 | 
			
		||||
// NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables)
 | 
			
		||||
std::vector<api::BluetoothLERawAdvertisement> batch_buffer;
 | 
			
		||||
}  // namespace
 | 
			
		||||
 | 
			
		||||
static std::vector<api::BluetoothLERawAdvertisement> &get_batch_buffer() { return batch_buffer; }
 | 
			
		||||
 | 
			
		||||
bool BluetoothProxy::parse_devices(const esp32_ble::BLEScanResult *scan_results, size_t count) {
 | 
			
		||||
  if (!api::global_api_server->is_connected() || this->api_connection_ == nullptr || !this->raw_advertisements_)
 | 
			
		||||
@@ -170,7 +180,7 @@ int BluetoothProxy::get_bluetooth_connections_free() {
 | 
			
		||||
void BluetoothProxy::loop() {
 | 
			
		||||
  if (!api::global_api_server->is_connected() || this->api_connection_ == nullptr) {
 | 
			
		||||
    for (auto *connection : this->connections_) {
 | 
			
		||||
      if (connection->get_address() != 0) {
 | 
			
		||||
      if (connection->get_address() != 0 && !connection->disconnect_pending()) {
 | 
			
		||||
        connection->disconnect();
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										1
									
								
								esphome/components/camera/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								esphome/components/camera/__init__.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1 @@
 | 
			
		||||
CODEOWNERS = ["@DT-art1", "@bdraco"]
 | 
			
		||||
							
								
								
									
										22
									
								
								esphome/components/camera/camera.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								esphome/components/camera/camera.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,22 @@
 | 
			
		||||
#include "camera.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace camera {
 | 
			
		||||
 | 
			
		||||
// NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables)
 | 
			
		||||
Camera *Camera::global_camera = nullptr;
 | 
			
		||||
 | 
			
		||||
Camera::Camera() {
 | 
			
		||||
  if (global_camera != nullptr) {
 | 
			
		||||
    this->status_set_error("Multiple cameras are configured, but only one is supported.");
 | 
			
		||||
    this->mark_failed();
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  global_camera = this;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Camera *Camera::instance() { return global_camera; }
 | 
			
		||||
 | 
			
		||||
}  // namespace camera
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
							
								
								
									
										80
									
								
								esphome/components/camera/camera.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								esphome/components/camera/camera.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,80 @@
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "esphome/core/automation.h"
 | 
			
		||||
#include "esphome/core/component.h"
 | 
			
		||||
#include "esphome/core/entity_base.h"
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace camera {
 | 
			
		||||
 | 
			
		||||
/** Different sources for filtering.
 | 
			
		||||
 *  IDLE: Camera requests to send an image to the API.
 | 
			
		||||
 *  API_REQUESTER: API requests a new image.
 | 
			
		||||
 *  WEB_REQUESTER: ESP32 web server request an image. Ignored by API.
 | 
			
		||||
 */
 | 
			
		||||
enum CameraRequester : uint8_t { IDLE, API_REQUESTER, WEB_REQUESTER };
 | 
			
		||||
 | 
			
		||||
/** Abstract camera image base class.
 | 
			
		||||
 *  Encapsulates the JPEG encoded data and it is shared among
 | 
			
		||||
 *  all connected clients.
 | 
			
		||||
 */
 | 
			
		||||
class CameraImage {
 | 
			
		||||
 public:
 | 
			
		||||
  virtual uint8_t *get_data_buffer() = 0;
 | 
			
		||||
  virtual size_t get_data_length() = 0;
 | 
			
		||||
  virtual bool was_requested_by(CameraRequester requester) const = 0;
 | 
			
		||||
  virtual ~CameraImage() {}
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/** Abstract image reader base class.
 | 
			
		||||
 *  Keeps track of the data offset of the camera image and
 | 
			
		||||
 *  how many bytes are remaining to read. When the image
 | 
			
		||||
 *  is returned, the shared_ptr is reset and the camera can
 | 
			
		||||
 *  reuse the memory of the camera image.
 | 
			
		||||
 */
 | 
			
		||||
class CameraImageReader {
 | 
			
		||||
 public:
 | 
			
		||||
  virtual void set_image(std::shared_ptr<CameraImage> image) = 0;
 | 
			
		||||
  virtual size_t available() const = 0;
 | 
			
		||||
  virtual uint8_t *peek_data_buffer() = 0;
 | 
			
		||||
  virtual void consume_data(size_t consumed) = 0;
 | 
			
		||||
  virtual void return_image() = 0;
 | 
			
		||||
  virtual ~CameraImageReader() {}
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/** Abstract camera base class. Collaborates with API.
 | 
			
		||||
 *  1) API server starts and installs callback (add_image_callback)
 | 
			
		||||
 *     which is called by the camera when a new image is available.
 | 
			
		||||
 *  2) New API client connects and creates a new image reader (create_image_reader).
 | 
			
		||||
 *  3) API connection receives protobuf CameraImageRequest and calls request_image.
 | 
			
		||||
 *  3.a) API connection receives protobuf CameraImageRequest and calls start_stream.
 | 
			
		||||
 *  4) Camera implementation provides JPEG data in the CameraImage and calls callback.
 | 
			
		||||
 *  5) API connection sets the image in the image reader.
 | 
			
		||||
 *  6) API connection consumes data from the image reader and returns the image when finished.
 | 
			
		||||
 *  7.a) Camera captures a new image and continues with 4) until start_stream is called.
 | 
			
		||||
 */
 | 
			
		||||
class Camera : public EntityBase, public Component {
 | 
			
		||||
 public:
 | 
			
		||||
  Camera();
 | 
			
		||||
  // Camera implementation invokes callback to publish a new image.
 | 
			
		||||
  virtual void add_image_callback(std::function<void(std::shared_ptr<CameraImage>)> &&callback) = 0;
 | 
			
		||||
  /// Returns a new camera image reader that keeps track of the JPEG data in the camera image.
 | 
			
		||||
  virtual CameraImageReader *create_image_reader() = 0;
 | 
			
		||||
  // Connection, camera or web server requests one new JPEG image.
 | 
			
		||||
  virtual void request_image(CameraRequester requester) = 0;
 | 
			
		||||
  // Connection, camera or web server requests a stream of images.
 | 
			
		||||
  virtual void start_stream(CameraRequester requester) = 0;
 | 
			
		||||
  // Connection or web server stops the previously started stream.
 | 
			
		||||
  virtual void stop_stream(CameraRequester requester) = 0;
 | 
			
		||||
  virtual ~Camera() {}
 | 
			
		||||
  /// The singleton instance of the camera implementation.
 | 
			
		||||
  static Camera *instance();
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  // NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables)
 | 
			
		||||
  static Camera *global_camera;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace camera
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -2,6 +2,7 @@
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@esphome/core"]
 | 
			
		||||
 | 
			
		||||
CONF_BYTE_ORDER = "byte_order"
 | 
			
		||||
CONF_DRAW_ROUNDING = "draw_rounding"
 | 
			
		||||
CONF_ON_STATE_CHANGE = "on_state_change"
 | 
			
		||||
CONF_REQUEST_HEADERS = "request_headers"
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,5 @@
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.config_helpers import filter_source_files_from_platform
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_BLOCK,
 | 
			
		||||
@@ -7,6 +8,7 @@ from esphome.const import (
 | 
			
		||||
    CONF_FREE,
 | 
			
		||||
    CONF_ID,
 | 
			
		||||
    CONF_LOOP_TIME,
 | 
			
		||||
    PlatformFramework,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@OttoWinter"]
 | 
			
		||||
@@ -44,3 +46,21 @@ CONFIG_SCHEMA = cv.All(
 | 
			
		||||
async def to_code(config):
 | 
			
		||||
    var = cg.new_Pvariable(config[CONF_ID])
 | 
			
		||||
    await cg.register_component(var, config)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FILTER_SOURCE_FILES = filter_source_files_from_platform(
 | 
			
		||||
    {
 | 
			
		||||
        "debug_esp32.cpp": {
 | 
			
		||||
            PlatformFramework.ESP32_ARDUINO,
 | 
			
		||||
            PlatformFramework.ESP32_IDF,
 | 
			
		||||
        },
 | 
			
		||||
        "debug_esp8266.cpp": {PlatformFramework.ESP8266_ARDUINO},
 | 
			
		||||
        "debug_host.cpp": {PlatformFramework.HOST_NATIVE},
 | 
			
		||||
        "debug_rp2040.cpp": {PlatformFramework.RP2040_ARDUINO},
 | 
			
		||||
        "debug_libretiny.cpp": {
 | 
			
		||||
            PlatformFramework.BK72XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.RTL87XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.LN882X_ARDUINO,
 | 
			
		||||
        },
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
 
 | 
			
		||||
@@ -53,6 +53,7 @@ void DebugComponent::on_shutdown() {
 | 
			
		||||
  auto pref = global_preferences->make_preference(REBOOT_MAX_LEN, fnv1_hash(REBOOT_KEY + App.get_name()));
 | 
			
		||||
  if (component != nullptr) {
 | 
			
		||||
    strncpy(buffer, component->get_component_source(), REBOOT_MAX_LEN - 1);
 | 
			
		||||
    buffer[REBOOT_MAX_LEN - 1] = '\0';
 | 
			
		||||
  }
 | 
			
		||||
  ESP_LOGD(TAG, "Storing reboot source: %s", buffer);
 | 
			
		||||
  pref.save(&buffer);
 | 
			
		||||
@@ -68,6 +69,7 @@ std::string DebugComponent::get_reset_reason_() {
 | 
			
		||||
      auto pref = global_preferences->make_preference(REBOOT_MAX_LEN, fnv1_hash(REBOOT_KEY + App.get_name()));
 | 
			
		||||
      char buffer[REBOOT_MAX_LEN]{};
 | 
			
		||||
      if (pref.load(&buffer)) {
 | 
			
		||||
        buffer[REBOOT_MAX_LEN - 1] = '\0';
 | 
			
		||||
        reset_reason = "Reboot request from " + std::string(buffer);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
from esphome import automation, pins
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components import time
 | 
			
		||||
from esphome.components import esp32, time
 | 
			
		||||
from esphome.components.esp32 import get_esp32_variant
 | 
			
		||||
from esphome.components.esp32.const import (
 | 
			
		||||
    VARIANT_ESP32,
 | 
			
		||||
@@ -11,6 +11,7 @@ from esphome.components.esp32.const import (
 | 
			
		||||
    VARIANT_ESP32S2,
 | 
			
		||||
    VARIANT_ESP32S3,
 | 
			
		||||
)
 | 
			
		||||
from esphome.config_helpers import filter_source_files_from_platform
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_DEFAULT,
 | 
			
		||||
@@ -27,6 +28,7 @@ from esphome.const import (
 | 
			
		||||
    CONF_WAKEUP_PIN,
 | 
			
		||||
    PLATFORM_ESP32,
 | 
			
		||||
    PLATFORM_ESP8266,
 | 
			
		||||
    PlatformFramework,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
WAKEUP_PINS = {
 | 
			
		||||
@@ -114,12 +116,20 @@ def validate_pin_number(value):
 | 
			
		||||
    return value
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def validate_config(config):
 | 
			
		||||
    if get_esp32_variant() == VARIANT_ESP32C3 and CONF_ESP32_EXT1_WAKEUP in config:
 | 
			
		||||
        raise cv.Invalid("ESP32-C3 does not support wakeup from touch.")
 | 
			
		||||
    if get_esp32_variant() == VARIANT_ESP32C3 and CONF_TOUCH_WAKEUP in config:
 | 
			
		||||
        raise cv.Invalid("ESP32-C3 does not support wakeup from ext1")
 | 
			
		||||
    return config
 | 
			
		||||
def _validate_ex1_wakeup_mode(value):
 | 
			
		||||
    if value == "ALL_LOW":
 | 
			
		||||
        esp32.only_on_variant(supported=[VARIANT_ESP32], msg_prefix="ALL_LOW")(value)
 | 
			
		||||
    if value == "ANY_LOW":
 | 
			
		||||
        esp32.only_on_variant(
 | 
			
		||||
            supported=[
 | 
			
		||||
                VARIANT_ESP32S2,
 | 
			
		||||
                VARIANT_ESP32S3,
 | 
			
		||||
                VARIANT_ESP32C6,
 | 
			
		||||
                VARIANT_ESP32H2,
 | 
			
		||||
            ],
 | 
			
		||||
            msg_prefix="ANY_LOW",
 | 
			
		||||
        )(value)
 | 
			
		||||
    return value
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
deep_sleep_ns = cg.esphome_ns.namespace("deep_sleep")
 | 
			
		||||
@@ -146,6 +156,7 @@ WAKEUP_PIN_MODES = {
 | 
			
		||||
esp_sleep_ext1_wakeup_mode_t = cg.global_ns.enum("esp_sleep_ext1_wakeup_mode_t")
 | 
			
		||||
Ext1Wakeup = deep_sleep_ns.struct("Ext1Wakeup")
 | 
			
		||||
EXT1_WAKEUP_MODES = {
 | 
			
		||||
    "ANY_LOW": esp_sleep_ext1_wakeup_mode_t.ESP_EXT1_WAKEUP_ANY_LOW,
 | 
			
		||||
    "ALL_LOW": esp_sleep_ext1_wakeup_mode_t.ESP_EXT1_WAKEUP_ALL_LOW,
 | 
			
		||||
    "ANY_HIGH": esp_sleep_ext1_wakeup_mode_t.ESP_EXT1_WAKEUP_ANY_HIGH,
 | 
			
		||||
}
 | 
			
		||||
@@ -185,16 +196,28 @@ CONFIG_SCHEMA = cv.All(
 | 
			
		||||
            ),
 | 
			
		||||
            cv.Optional(CONF_ESP32_EXT1_WAKEUP): cv.All(
 | 
			
		||||
                cv.only_on_esp32,
 | 
			
		||||
                esp32.only_on_variant(
 | 
			
		||||
                    unsupported=[VARIANT_ESP32C3], msg_prefix="Wakeup from ext1"
 | 
			
		||||
                ),
 | 
			
		||||
                cv.Schema(
 | 
			
		||||
                    {
 | 
			
		||||
                        cv.Required(CONF_PINS): cv.ensure_list(
 | 
			
		||||
                            pins.internal_gpio_input_pin_schema, validate_pin_number
 | 
			
		||||
                        ),
 | 
			
		||||
                        cv.Required(CONF_MODE): cv.enum(EXT1_WAKEUP_MODES, upper=True),
 | 
			
		||||
                        cv.Required(CONF_MODE): cv.All(
 | 
			
		||||
                            cv.enum(EXT1_WAKEUP_MODES, upper=True),
 | 
			
		||||
                            _validate_ex1_wakeup_mode,
 | 
			
		||||
                        ),
 | 
			
		||||
                    }
 | 
			
		||||
                ),
 | 
			
		||||
            ),
 | 
			
		||||
            cv.Optional(CONF_TOUCH_WAKEUP): cv.All(cv.only_on_esp32, cv.boolean),
 | 
			
		||||
            cv.Optional(CONF_TOUCH_WAKEUP): cv.All(
 | 
			
		||||
                cv.only_on_esp32,
 | 
			
		||||
                esp32.only_on_variant(
 | 
			
		||||
                    unsupported=[VARIANT_ESP32C3], msg_prefix="Wakeup from touch"
 | 
			
		||||
                ),
 | 
			
		||||
                cv.boolean,
 | 
			
		||||
            ),
 | 
			
		||||
        }
 | 
			
		||||
    ).extend(cv.COMPONENT_SCHEMA),
 | 
			
		||||
    cv.only_on([PLATFORM_ESP32, PLATFORM_ESP8266]),
 | 
			
		||||
@@ -313,3 +336,14 @@ async def deep_sleep_action_to_code(config, action_id, template_arg, args):
 | 
			
		||||
    var = cg.new_Pvariable(action_id, template_arg)
 | 
			
		||||
    await cg.register_parented(var, config[CONF_ID])
 | 
			
		||||
    return var
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FILTER_SOURCE_FILES = filter_source_files_from_platform(
 | 
			
		||||
    {
 | 
			
		||||
        "deep_sleep_esp32.cpp": {
 | 
			
		||||
            PlatformFramework.ESP32_ARDUINO,
 | 
			
		||||
            PlatformFramework.ESP32_IDF,
 | 
			
		||||
        },
 | 
			
		||||
        "deep_sleep_esp8266.cpp": {PlatformFramework.ESP8266_ARDUINO},
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
 
 | 
			
		||||
@@ -189,7 +189,7 @@ def get_download_types(storage_json):
 | 
			
		||||
    ]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def only_on_variant(*, supported=None, unsupported=None):
 | 
			
		||||
def only_on_variant(*, supported=None, unsupported=None, msg_prefix="This feature"):
 | 
			
		||||
    """Config validator for features only available on some ESP32 variants."""
 | 
			
		||||
    if supported is not None and not isinstance(supported, list):
 | 
			
		||||
        supported = [supported]
 | 
			
		||||
@@ -200,11 +200,11 @@ def only_on_variant(*, supported=None, unsupported=None):
 | 
			
		||||
        variant = get_esp32_variant()
 | 
			
		||||
        if supported is not None and variant not in supported:
 | 
			
		||||
            raise cv.Invalid(
 | 
			
		||||
                f"This feature is only available on {', '.join(supported)}"
 | 
			
		||||
                f"{msg_prefix} is only available on {', '.join(supported)}"
 | 
			
		||||
            )
 | 
			
		||||
        if unsupported is not None and variant in unsupported:
 | 
			
		||||
            raise cv.Invalid(
 | 
			
		||||
                f"This feature is not available on {', '.join(unsupported)}"
 | 
			
		||||
                f"{msg_prefix} is not available on {', '.join(unsupported)}"
 | 
			
		||||
            )
 | 
			
		||||
        return obj
 | 
			
		||||
 | 
			
		||||
@@ -707,6 +707,7 @@ async def to_code(config):
 | 
			
		||||
    cg.add_define("ESPHOME_VARIANT", VARIANT_FRIENDLY[config[CONF_VARIANT]])
 | 
			
		||||
 | 
			
		||||
    cg.add_platformio_option("lib_ldf_mode", "off")
 | 
			
		||||
    cg.add_platformio_option("lib_compat_mode", "strict")
 | 
			
		||||
 | 
			
		||||
    framework_ver: cv.Version = CORE.data[KEY_CORE][KEY_FRAMEWORK_VERSION]
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										69
									
								
								esphome/components/esp32/helpers.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								esphome/components/esp32/helpers.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,69 @@
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP32
 | 
			
		||||
 | 
			
		||||
#include "esp_efuse.h"
 | 
			
		||||
#include "esp_efuse_table.h"
 | 
			
		||||
#include "esp_mac.h"
 | 
			
		||||
 | 
			
		||||
#include <freertos/FreeRTOS.h>
 | 
			
		||||
#include <freertos/portmacro.h>
 | 
			
		||||
#include "esp_random.h"
 | 
			
		||||
#include "esp_system.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
 | 
			
		||||
uint32_t random_uint32() { return esp_random(); }
 | 
			
		||||
bool random_bytes(uint8_t *data, size_t len) {
 | 
			
		||||
  esp_fill_random(data, len);
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Mutex::Mutex() { handle_ = xSemaphoreCreateMutex(); }
 | 
			
		||||
Mutex::~Mutex() {}
 | 
			
		||||
void Mutex::lock() { xSemaphoreTake(this->handle_, portMAX_DELAY); }
 | 
			
		||||
bool Mutex::try_lock() { return xSemaphoreTake(this->handle_, 0) == pdTRUE; }
 | 
			
		||||
void Mutex::unlock() { xSemaphoreGive(this->handle_); }
 | 
			
		||||
 | 
			
		||||
// only affects the executing core
 | 
			
		||||
// so should not be used as a mutex lock, only to get accurate timing
 | 
			
		||||
IRAM_ATTR InterruptLock::InterruptLock() { portDISABLE_INTERRUPTS(); }
 | 
			
		||||
IRAM_ATTR InterruptLock::~InterruptLock() { portENABLE_INTERRUPTS(); }
 | 
			
		||||
 | 
			
		||||
void get_mac_address_raw(uint8_t *mac) {  // NOLINT(readability-non-const-parameter)
 | 
			
		||||
#if defined(CONFIG_SOC_IEEE802154_SUPPORTED)
 | 
			
		||||
  // When CONFIG_SOC_IEEE802154_SUPPORTED is defined, esp_efuse_mac_get_default
 | 
			
		||||
  // returns the 802.15.4 EUI-64 address, so we read directly from eFuse instead.
 | 
			
		||||
  if (has_custom_mac_address()) {
 | 
			
		||||
    esp_efuse_read_field_blob(ESP_EFUSE_MAC_CUSTOM, mac, 48);
 | 
			
		||||
  } else {
 | 
			
		||||
    esp_efuse_read_field_blob(ESP_EFUSE_MAC_FACTORY, mac, 48);
 | 
			
		||||
  }
 | 
			
		||||
#else
 | 
			
		||||
  if (has_custom_mac_address()) {
 | 
			
		||||
    esp_efuse_mac_get_custom(mac);
 | 
			
		||||
  } else {
 | 
			
		||||
    esp_efuse_mac_get_default(mac);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void set_mac_address(uint8_t *mac) { esp_base_mac_addr_set(mac); }
 | 
			
		||||
 | 
			
		||||
bool has_custom_mac_address() {
 | 
			
		||||
#if !defined(USE_ESP32_IGNORE_EFUSE_CUSTOM_MAC)
 | 
			
		||||
  uint8_t mac[6];
 | 
			
		||||
  // do not use 'esp_efuse_mac_get_custom(mac)' because it drops an error in the logs whenever it fails
 | 
			
		||||
#ifndef USE_ESP32_VARIANT_ESP32
 | 
			
		||||
  return (esp_efuse_read_field_blob(ESP_EFUSE_USER_DATA_MAC_CUSTOM, mac, 48) == ESP_OK) && mac_address_is_valid(mac);
 | 
			
		||||
#else
 | 
			
		||||
  return (esp_efuse_read_field_blob(ESP_EFUSE_MAC_CUSTOM, mac, 48) == ESP_OK) && mac_address_is_valid(mac);
 | 
			
		||||
#endif
 | 
			
		||||
#else
 | 
			
		||||
  return false;
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
 | 
			
		||||
#endif  // USE_ESP32
 | 
			
		||||
@@ -25,10 +25,15 @@ namespace esphome {
 | 
			
		||||
namespace esp32_ble {
 | 
			
		||||
 | 
			
		||||
// Maximum number of BLE scan results to buffer
 | 
			
		||||
// Sized to handle bursts of advertisements while allowing for processing delays
 | 
			
		||||
// With 16 advertisements per batch and some safety margin:
 | 
			
		||||
// - Without PSRAM: 24 entries (1.5× batch size)
 | 
			
		||||
// - With PSRAM: 36 entries (2.25× batch size)
 | 
			
		||||
// The reduced structure size (~80 bytes vs ~400 bytes) allows for larger buffers
 | 
			
		||||
#ifdef USE_PSRAM
 | 
			
		||||
static constexpr uint8_t SCAN_RESULT_BUFFER_SIZE = 32;
 | 
			
		||||
static constexpr uint8_t SCAN_RESULT_BUFFER_SIZE = 36;
 | 
			
		||||
#else
 | 
			
		||||
static constexpr uint8_t SCAN_RESULT_BUFFER_SIZE = 20;
 | 
			
		||||
static constexpr uint8_t SCAN_RESULT_BUFFER_SIZE = 24;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
// Maximum size of the BLE event queue - must be power of 2 for lock-free queue
 | 
			
		||||
@@ -51,7 +56,7 @@ enum IoCapability {
 | 
			
		||||
  IO_CAP_KBDISP = ESP_IO_CAP_KBDISP,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum BLEComponentState {
 | 
			
		||||
enum BLEComponentState : uint8_t {
 | 
			
		||||
  /** Nothing has been initialized yet. */
 | 
			
		||||
  BLE_COMPONENT_STATE_OFF = 0,
 | 
			
		||||
  /** BLE should be disabled on next loop. */
 | 
			
		||||
@@ -141,21 +146,31 @@ class ESP32BLE : public Component {
 | 
			
		||||
 private:
 | 
			
		||||
  template<typename... Args> friend void enqueue_ble_event(Args... args);
 | 
			
		||||
 | 
			
		||||
  // Vectors (12 bytes each on 32-bit, naturally aligned to 4 bytes)
 | 
			
		||||
  std::vector<GAPEventHandler *> gap_event_handlers_;
 | 
			
		||||
  std::vector<GAPScanEventHandler *> gap_scan_event_handlers_;
 | 
			
		||||
  std::vector<GATTcEventHandler *> gattc_event_handlers_;
 | 
			
		||||
  std::vector<GATTsEventHandler *> gatts_event_handlers_;
 | 
			
		||||
  std::vector<BLEStatusEventHandler *> ble_status_event_handlers_;
 | 
			
		||||
  BLEComponentState state_{BLE_COMPONENT_STATE_OFF};
 | 
			
		||||
 | 
			
		||||
  // Large objects (size depends on template parameters, but typically aligned to 4 bytes)
 | 
			
		||||
  esphome::LockFreeQueue<BLEEvent, MAX_BLE_QUEUE_SIZE> ble_events_;
 | 
			
		||||
  esphome::EventPool<BLEEvent, MAX_BLE_QUEUE_SIZE> ble_event_pool_;
 | 
			
		||||
  BLEAdvertising *advertising_{};
 | 
			
		||||
  esp_ble_io_cap_t io_cap_{ESP_IO_CAP_NONE};
 | 
			
		||||
  uint32_t advertising_cycle_time_{};
 | 
			
		||||
  bool enable_on_boot_{};
 | 
			
		||||
 | 
			
		||||
  // optional<string> (typically 16+ bytes on 32-bit, aligned to 4 bytes)
 | 
			
		||||
  optional<std::string> name_;
 | 
			
		||||
  uint16_t appearance_{0};
 | 
			
		||||
 | 
			
		||||
  // 4-byte aligned members
 | 
			
		||||
  BLEAdvertising *advertising_{};             // 4 bytes (pointer)
 | 
			
		||||
  esp_ble_io_cap_t io_cap_{ESP_IO_CAP_NONE};  // 4 bytes (enum)
 | 
			
		||||
  uint32_t advertising_cycle_time_{};         // 4 bytes
 | 
			
		||||
 | 
			
		||||
  // 2-byte aligned members
 | 
			
		||||
  uint16_t appearance_{0};  // 2 bytes
 | 
			
		||||
 | 
			
		||||
  // 1-byte aligned members (grouped together to minimize padding)
 | 
			
		||||
  BLEComponentState state_{BLE_COMPONENT_STATE_OFF};  // 1 byte (uint8_t enum)
 | 
			
		||||
  bool enable_on_boot_{};                             // 1 byte
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables)
 | 
			
		||||
 
 | 
			
		||||
@@ -23,7 +23,7 @@ from esphome.core.entity_helpers import setup_entity
 | 
			
		||||
 | 
			
		||||
DEPENDENCIES = ["esp32"]
 | 
			
		||||
 | 
			
		||||
AUTO_LOAD = ["psram"]
 | 
			
		||||
AUTO_LOAD = ["camera", "psram"]
 | 
			
		||||
 | 
			
		||||
esp32_camera_ns = cg.esphome_ns.namespace("esp32_camera")
 | 
			
		||||
ESP32Camera = esp32_camera_ns.class_("ESP32Camera", cg.PollingComponent, cg.EntityBase)
 | 
			
		||||
@@ -283,6 +283,7 @@ SETTERS = {
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
async def to_code(config):
 | 
			
		||||
    cg.add_define("USE_CAMERA")
 | 
			
		||||
    var = cg.new_Pvariable(config[CONF_ID])
 | 
			
		||||
    await setup_entity(var, config, "camera")
 | 
			
		||||
    await cg.register_component(var, config)
 | 
			
		||||
@@ -307,7 +308,7 @@ async def to_code(config):
 | 
			
		||||
    cg.add(var.set_frame_buffer_count(config[CONF_FRAME_BUFFER_COUNT]))
 | 
			
		||||
    cg.add(var.set_frame_size(config[CONF_RESOLUTION]))
 | 
			
		||||
 | 
			
		||||
    cg.add_define("USE_ESP32_CAMERA")
 | 
			
		||||
    cg.add_define("USE_CAMERA")
 | 
			
		||||
 | 
			
		||||
    if CORE.using_esp_idf:
 | 
			
		||||
        add_idf_component(name="espressif/esp32-camera", ref="2.0.15")
 | 
			
		||||
 
 | 
			
		||||
@@ -14,8 +14,6 @@ static const char *const TAG = "esp32_camera";
 | 
			
		||||
 | 
			
		||||
/* ---------------- public API (derivated) ---------------- */
 | 
			
		||||
void ESP32Camera::setup() {
 | 
			
		||||
  global_esp32_camera = this;
 | 
			
		||||
 | 
			
		||||
#ifdef USE_I2C
 | 
			
		||||
  if (this->i2c_bus_ != nullptr) {
 | 
			
		||||
    this->config_.sccb_i2c_port = this->i2c_bus_->get_port();
 | 
			
		||||
@@ -43,7 +41,7 @@ void ESP32Camera::setup() {
 | 
			
		||||
  xTaskCreatePinnedToCore(&ESP32Camera::framebuffer_task,
 | 
			
		||||
                          "framebuffer_task",  // name
 | 
			
		||||
                          1024,                // stack size
 | 
			
		||||
                          nullptr,             // task pv params
 | 
			
		||||
                          this,                // task pv params
 | 
			
		||||
                          1,                   // priority
 | 
			
		||||
                          nullptr,             // handle
 | 
			
		||||
                          1                    // core
 | 
			
		||||
@@ -176,7 +174,7 @@ void ESP32Camera::loop() {
 | 
			
		||||
  const uint32_t now = App.get_loop_component_start_time();
 | 
			
		||||
  if (this->idle_update_interval_ != 0 && now - this->last_idle_request_ > this->idle_update_interval_) {
 | 
			
		||||
    this->last_idle_request_ = now;
 | 
			
		||||
    this->request_image(IDLE);
 | 
			
		||||
    this->request_image(camera::IDLE);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Check if we should fetch a new image
 | 
			
		||||
@@ -202,7 +200,7 @@ void ESP32Camera::loop() {
 | 
			
		||||
    xQueueSend(this->framebuffer_return_queue_, &fb, portMAX_DELAY);
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  this->current_image_ = std::make_shared<CameraImage>(fb, this->single_requesters_ | this->stream_requesters_);
 | 
			
		||||
  this->current_image_ = std::make_shared<ESP32CameraImage>(fb, this->single_requesters_ | this->stream_requesters_);
 | 
			
		||||
 | 
			
		||||
  ESP_LOGD(TAG, "Got Image: len=%u", fb->len);
 | 
			
		||||
  this->new_image_callback_.call(this->current_image_);
 | 
			
		||||
@@ -225,8 +223,6 @@ ESP32Camera::ESP32Camera() {
 | 
			
		||||
  this->config_.fb_count = 1;
 | 
			
		||||
  this->config_.grab_mode = CAMERA_GRAB_WHEN_EMPTY;
 | 
			
		||||
  this->config_.fb_location = CAMERA_FB_IN_PSRAM;
 | 
			
		||||
 | 
			
		||||
  global_esp32_camera = this;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* ---------------- setters ---------------- */
 | 
			
		||||
@@ -356,7 +352,7 @@ void ESP32Camera::set_frame_buffer_count(uint8_t fb_count) {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* ---------------- public API (specific) ---------------- */
 | 
			
		||||
void ESP32Camera::add_image_callback(std::function<void(std::shared_ptr<CameraImage>)> &&callback) {
 | 
			
		||||
void ESP32Camera::add_image_callback(std::function<void(std::shared_ptr<camera::CameraImage>)> &&callback) {
 | 
			
		||||
  this->new_image_callback_.add(std::move(callback));
 | 
			
		||||
}
 | 
			
		||||
void ESP32Camera::add_stream_start_callback(std::function<void()> &&callback) {
 | 
			
		||||
@@ -365,15 +361,16 @@ void ESP32Camera::add_stream_start_callback(std::function<void()> &&callback) {
 | 
			
		||||
void ESP32Camera::add_stream_stop_callback(std::function<void()> &&callback) {
 | 
			
		||||
  this->stream_stop_callback_.add(std::move(callback));
 | 
			
		||||
}
 | 
			
		||||
void ESP32Camera::start_stream(CameraRequester requester) {
 | 
			
		||||
void ESP32Camera::start_stream(camera::CameraRequester requester) {
 | 
			
		||||
  this->stream_start_callback_.call();
 | 
			
		||||
  this->stream_requesters_ |= (1U << requester);
 | 
			
		||||
}
 | 
			
		||||
void ESP32Camera::stop_stream(CameraRequester requester) {
 | 
			
		||||
void ESP32Camera::stop_stream(camera::CameraRequester requester) {
 | 
			
		||||
  this->stream_stop_callback_.call();
 | 
			
		||||
  this->stream_requesters_ &= ~(1U << requester);
 | 
			
		||||
}
 | 
			
		||||
void ESP32Camera::request_image(CameraRequester requester) { this->single_requesters_ |= (1U << requester); }
 | 
			
		||||
void ESP32Camera::request_image(camera::CameraRequester requester) { this->single_requesters_ |= (1U << requester); }
 | 
			
		||||
camera::CameraImageReader *ESP32Camera::create_image_reader() { return new ESP32CameraImageReader; }
 | 
			
		||||
void ESP32Camera::update_camera_parameters() {
 | 
			
		||||
  sensor_t *s = esp_camera_sensor_get();
 | 
			
		||||
  /* update image */
 | 
			
		||||
@@ -402,39 +399,39 @@ void ESP32Camera::update_camera_parameters() {
 | 
			
		||||
bool ESP32Camera::has_requested_image_() const { return this->single_requesters_ || this->stream_requesters_; }
 | 
			
		||||
bool ESP32Camera::can_return_image_() const { return this->current_image_.use_count() == 1; }
 | 
			
		||||
void ESP32Camera::framebuffer_task(void *pv) {
 | 
			
		||||
  ESP32Camera *that = (ESP32Camera *) pv;
 | 
			
		||||
  while (true) {
 | 
			
		||||
    camera_fb_t *framebuffer = esp_camera_fb_get();
 | 
			
		||||
    xQueueSend(global_esp32_camera->framebuffer_get_queue_, &framebuffer, portMAX_DELAY);
 | 
			
		||||
    xQueueSend(that->framebuffer_get_queue_, &framebuffer, portMAX_DELAY);
 | 
			
		||||
    // return is no-op for config with 1 fb
 | 
			
		||||
    xQueueReceive(global_esp32_camera->framebuffer_return_queue_, &framebuffer, portMAX_DELAY);
 | 
			
		||||
    xQueueReceive(that->framebuffer_return_queue_, &framebuffer, portMAX_DELAY);
 | 
			
		||||
    esp_camera_fb_return(framebuffer);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
ESP32Camera *global_esp32_camera;  // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
 | 
			
		||||
 | 
			
		||||
/* ---------------- CameraImageReader class ---------------- */
 | 
			
		||||
void CameraImageReader::set_image(std::shared_ptr<CameraImage> image) {
 | 
			
		||||
  this->image_ = std::move(image);
 | 
			
		||||
/* ---------------- ESP32CameraImageReader class ----------- */
 | 
			
		||||
void ESP32CameraImageReader::set_image(std::shared_ptr<camera::CameraImage> image) {
 | 
			
		||||
  this->image_ = std::static_pointer_cast<ESP32CameraImage>(image);
 | 
			
		||||
  this->offset_ = 0;
 | 
			
		||||
}
 | 
			
		||||
size_t CameraImageReader::available() const {
 | 
			
		||||
size_t ESP32CameraImageReader::available() const {
 | 
			
		||||
  if (!this->image_)
 | 
			
		||||
    return 0;
 | 
			
		||||
 | 
			
		||||
  return this->image_->get_data_length() - this->offset_;
 | 
			
		||||
}
 | 
			
		||||
void CameraImageReader::return_image() { this->image_.reset(); }
 | 
			
		||||
void CameraImageReader::consume_data(size_t consumed) { this->offset_ += consumed; }
 | 
			
		||||
uint8_t *CameraImageReader::peek_data_buffer() { return this->image_->get_data_buffer() + this->offset_; }
 | 
			
		||||
void ESP32CameraImageReader::return_image() { this->image_.reset(); }
 | 
			
		||||
void ESP32CameraImageReader::consume_data(size_t consumed) { this->offset_ += consumed; }
 | 
			
		||||
uint8_t *ESP32CameraImageReader::peek_data_buffer() { return this->image_->get_data_buffer() + this->offset_; }
 | 
			
		||||
 | 
			
		||||
/* ---------------- CameraImage class ---------------- */
 | 
			
		||||
CameraImage::CameraImage(camera_fb_t *buffer, uint8_t requesters) : buffer_(buffer), requesters_(requesters) {}
 | 
			
		||||
/* ---------------- ESP32CameraImage class ----------- */
 | 
			
		||||
ESP32CameraImage::ESP32CameraImage(camera_fb_t *buffer, uint8_t requesters)
 | 
			
		||||
    : buffer_(buffer), requesters_(requesters) {}
 | 
			
		||||
 | 
			
		||||
camera_fb_t *CameraImage::get_raw_buffer() { return this->buffer_; }
 | 
			
		||||
uint8_t *CameraImage::get_data_buffer() { return this->buffer_->buf; }
 | 
			
		||||
size_t CameraImage::get_data_length() { return this->buffer_->len; }
 | 
			
		||||
bool CameraImage::was_requested_by(CameraRequester requester) const {
 | 
			
		||||
camera_fb_t *ESP32CameraImage::get_raw_buffer() { return this->buffer_; }
 | 
			
		||||
uint8_t *ESP32CameraImage::get_data_buffer() { return this->buffer_->buf; }
 | 
			
		||||
size_t ESP32CameraImage::get_data_length() { return this->buffer_->len; }
 | 
			
		||||
bool ESP32CameraImage::was_requested_by(camera::CameraRequester requester) const {
 | 
			
		||||
  return (this->requesters_ & (1 << requester)) != 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -7,7 +7,7 @@
 | 
			
		||||
#include <freertos/queue.h>
 | 
			
		||||
#include "esphome/core/automation.h"
 | 
			
		||||
#include "esphome/core/component.h"
 | 
			
		||||
#include "esphome/core/entity_base.h"
 | 
			
		||||
#include "esphome/components/camera/camera.h"
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
 | 
			
		||||
#ifdef USE_I2C
 | 
			
		||||
@@ -19,9 +19,6 @@ namespace esp32_camera {
 | 
			
		||||
 | 
			
		||||
class ESP32Camera;
 | 
			
		||||
 | 
			
		||||
/* ---------------- enum classes ---------------- */
 | 
			
		||||
enum CameraRequester { IDLE, API_REQUESTER, WEB_REQUESTER };
 | 
			
		||||
 | 
			
		||||
enum ESP32CameraFrameSize {
 | 
			
		||||
  ESP32_CAMERA_SIZE_160X120,    // QQVGA
 | 
			
		||||
  ESP32_CAMERA_SIZE_176X144,    // QCIF
 | 
			
		||||
@@ -77,13 +74,13 @@ enum ESP32SpecialEffect {
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* ---------------- CameraImage class ---------------- */
 | 
			
		||||
class CameraImage {
 | 
			
		||||
class ESP32CameraImage : public camera::CameraImage {
 | 
			
		||||
 public:
 | 
			
		||||
  CameraImage(camera_fb_t *buffer, uint8_t requester);
 | 
			
		||||
  ESP32CameraImage(camera_fb_t *buffer, uint8_t requester);
 | 
			
		||||
  camera_fb_t *get_raw_buffer();
 | 
			
		||||
  uint8_t *get_data_buffer();
 | 
			
		||||
  size_t get_data_length();
 | 
			
		||||
  bool was_requested_by(CameraRequester requester) const;
 | 
			
		||||
  uint8_t *get_data_buffer() override;
 | 
			
		||||
  size_t get_data_length() override;
 | 
			
		||||
  bool was_requested_by(camera::CameraRequester requester) const override;
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  camera_fb_t *buffer_;
 | 
			
		||||
@@ -96,21 +93,21 @@ struct CameraImageData {
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* ---------------- CameraImageReader class ---------------- */
 | 
			
		||||
class CameraImageReader {
 | 
			
		||||
class ESP32CameraImageReader : public camera::CameraImageReader {
 | 
			
		||||
 public:
 | 
			
		||||
  void set_image(std::shared_ptr<CameraImage> image);
 | 
			
		||||
  size_t available() const;
 | 
			
		||||
  uint8_t *peek_data_buffer();
 | 
			
		||||
  void consume_data(size_t consumed);
 | 
			
		||||
  void return_image();
 | 
			
		||||
  void set_image(std::shared_ptr<camera::CameraImage> image) override;
 | 
			
		||||
  size_t available() const override;
 | 
			
		||||
  uint8_t *peek_data_buffer() override;
 | 
			
		||||
  void consume_data(size_t consumed) override;
 | 
			
		||||
  void return_image() override;
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  std::shared_ptr<CameraImage> image_;
 | 
			
		||||
  std::shared_ptr<ESP32CameraImage> image_;
 | 
			
		||||
  size_t offset_{0};
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* ---------------- ESP32Camera class ---------------- */
 | 
			
		||||
class ESP32Camera : public EntityBase, public Component {
 | 
			
		||||
class ESP32Camera : public camera::Camera {
 | 
			
		||||
 public:
 | 
			
		||||
  ESP32Camera();
 | 
			
		||||
 | 
			
		||||
@@ -162,14 +159,15 @@ class ESP32Camera : public EntityBase, public Component {
 | 
			
		||||
  void dump_config() override;
 | 
			
		||||
  float get_setup_priority() const override;
 | 
			
		||||
  /* public API (specific) */
 | 
			
		||||
  void start_stream(CameraRequester requester);
 | 
			
		||||
  void stop_stream(CameraRequester requester);
 | 
			
		||||
  void request_image(CameraRequester requester);
 | 
			
		||||
  void start_stream(camera::CameraRequester requester) override;
 | 
			
		||||
  void stop_stream(camera::CameraRequester requester) override;
 | 
			
		||||
  void request_image(camera::CameraRequester requester) override;
 | 
			
		||||
  void update_camera_parameters();
 | 
			
		||||
 | 
			
		||||
  void add_image_callback(std::function<void(std::shared_ptr<CameraImage>)> &&callback);
 | 
			
		||||
  void add_image_callback(std::function<void(std::shared_ptr<camera::CameraImage>)> &&callback) override;
 | 
			
		||||
  void add_stream_start_callback(std::function<void()> &&callback);
 | 
			
		||||
  void add_stream_stop_callback(std::function<void()> &&callback);
 | 
			
		||||
  camera::CameraImageReader *create_image_reader() override;
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  /* internal methods */
 | 
			
		||||
@@ -206,12 +204,12 @@ class ESP32Camera : public EntityBase, public Component {
 | 
			
		||||
  uint32_t idle_update_interval_{15000};
 | 
			
		||||
 | 
			
		||||
  esp_err_t init_error_{ESP_OK};
 | 
			
		||||
  std::shared_ptr<CameraImage> current_image_;
 | 
			
		||||
  std::shared_ptr<ESP32CameraImage> current_image_;
 | 
			
		||||
  uint8_t single_requesters_{0};
 | 
			
		||||
  uint8_t stream_requesters_{0};
 | 
			
		||||
  QueueHandle_t framebuffer_get_queue_;
 | 
			
		||||
  QueueHandle_t framebuffer_return_queue_;
 | 
			
		||||
  CallbackManager<void(std::shared_ptr<CameraImage>)> new_image_callback_{};
 | 
			
		||||
  CallbackManager<void(std::shared_ptr<camera::CameraImage>)> new_image_callback_{};
 | 
			
		||||
  CallbackManager<void()> stream_start_callback_{};
 | 
			
		||||
  CallbackManager<void()> stream_stop_callback_{};
 | 
			
		||||
 | 
			
		||||
@@ -222,13 +220,10 @@ class ESP32Camera : public EntityBase, public Component {
 | 
			
		||||
#endif  // USE_I2C
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables)
 | 
			
		||||
extern ESP32Camera *global_esp32_camera;
 | 
			
		||||
 | 
			
		||||
class ESP32CameraImageTrigger : public Trigger<CameraImageData> {
 | 
			
		||||
 public:
 | 
			
		||||
  explicit ESP32CameraImageTrigger(ESP32Camera *parent) {
 | 
			
		||||
    parent->add_image_callback([this](const std::shared_ptr<esp32_camera::CameraImage> &image) {
 | 
			
		||||
    parent->add_image_callback([this](const std::shared_ptr<camera::CameraImage> &image) {
 | 
			
		||||
      CameraImageData camera_image_data{};
 | 
			
		||||
      camera_image_data.length = image->get_data_length();
 | 
			
		||||
      camera_image_data.data = image->get_data_buffer();
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,8 @@ import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import CONF_ID, CONF_MODE, CONF_PORT
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@ayufan"]
 | 
			
		||||
DEPENDENCIES = ["esp32_camera", "network"]
 | 
			
		||||
AUTO_LOAD = ["camera"]
 | 
			
		||||
DEPENDENCIES = ["network"]
 | 
			
		||||
MULTI_CONF = True
 | 
			
		||||
 | 
			
		||||
esp32_camera_web_server_ns = cg.esphome_ns.namespace("esp32_camera_web_server")
 | 
			
		||||
 
 | 
			
		||||
@@ -40,7 +40,7 @@ CameraWebServer::CameraWebServer() {}
 | 
			
		||||
CameraWebServer::~CameraWebServer() {}
 | 
			
		||||
 | 
			
		||||
void CameraWebServer::setup() {
 | 
			
		||||
  if (!esp32_camera::global_esp32_camera || esp32_camera::global_esp32_camera->is_failed()) {
 | 
			
		||||
  if (!camera::Camera::instance() || camera::Camera::instance()->is_failed()) {
 | 
			
		||||
    this->mark_failed();
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
@@ -67,8 +67,8 @@ void CameraWebServer::setup() {
 | 
			
		||||
 | 
			
		||||
  httpd_register_uri_handler(this->httpd_, &uri);
 | 
			
		||||
 | 
			
		||||
  esp32_camera::global_esp32_camera->add_image_callback([this](std::shared_ptr<esp32_camera::CameraImage> image) {
 | 
			
		||||
    if (this->running_ && image->was_requested_by(esp32_camera::WEB_REQUESTER)) {
 | 
			
		||||
  camera::Camera::instance()->add_image_callback([this](std::shared_ptr<camera::CameraImage> image) {
 | 
			
		||||
    if (this->running_ && image->was_requested_by(camera::WEB_REQUESTER)) {
 | 
			
		||||
      this->image_ = std::move(image);
 | 
			
		||||
      xSemaphoreGive(this->semaphore_);
 | 
			
		||||
    }
 | 
			
		||||
@@ -108,8 +108,8 @@ void CameraWebServer::loop() {
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
std::shared_ptr<esphome::esp32_camera::CameraImage> CameraWebServer::wait_for_image_() {
 | 
			
		||||
  std::shared_ptr<esphome::esp32_camera::CameraImage> image;
 | 
			
		||||
std::shared_ptr<esphome::camera::CameraImage> CameraWebServer::wait_for_image_() {
 | 
			
		||||
  std::shared_ptr<esphome::camera::CameraImage> image;
 | 
			
		||||
  image.swap(this->image_);
 | 
			
		||||
 | 
			
		||||
  if (!image) {
 | 
			
		||||
@@ -172,7 +172,7 @@ esp_err_t CameraWebServer::streaming_handler_(struct httpd_req *req) {
 | 
			
		||||
  uint32_t last_frame = millis();
 | 
			
		||||
  uint32_t frames = 0;
 | 
			
		||||
 | 
			
		||||
  esp32_camera::global_esp32_camera->start_stream(esphome::esp32_camera::WEB_REQUESTER);
 | 
			
		||||
  camera::Camera::instance()->start_stream(esphome::camera::WEB_REQUESTER);
 | 
			
		||||
 | 
			
		||||
  while (res == ESP_OK && this->running_) {
 | 
			
		||||
    auto image = this->wait_for_image_();
 | 
			
		||||
@@ -205,7 +205,7 @@ esp_err_t CameraWebServer::streaming_handler_(struct httpd_req *req) {
 | 
			
		||||
    res = httpd_send_all(req, STREAM_ERROR, strlen(STREAM_ERROR));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  esp32_camera::global_esp32_camera->stop_stream(esphome::esp32_camera::WEB_REQUESTER);
 | 
			
		||||
  camera::Camera::instance()->stop_stream(esphome::camera::WEB_REQUESTER);
 | 
			
		||||
 | 
			
		||||
  ESP_LOGI(TAG, "STREAM: closed. Frames: %" PRIu32, frames);
 | 
			
		||||
 | 
			
		||||
@@ -215,7 +215,7 @@ esp_err_t CameraWebServer::streaming_handler_(struct httpd_req *req) {
 | 
			
		||||
esp_err_t CameraWebServer::snapshot_handler_(struct httpd_req *req) {
 | 
			
		||||
  esp_err_t res = ESP_OK;
 | 
			
		||||
 | 
			
		||||
  esp32_camera::global_esp32_camera->request_image(esphome::esp32_camera::WEB_REQUESTER);
 | 
			
		||||
  camera::Camera::instance()->request_image(esphome::camera::WEB_REQUESTER);
 | 
			
		||||
 | 
			
		||||
  auto image = this->wait_for_image_();
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -6,7 +6,7 @@
 | 
			
		||||
#include <freertos/FreeRTOS.h>
 | 
			
		||||
#include <freertos/semphr.h>
 | 
			
		||||
 | 
			
		||||
#include "esphome/components/esp32_camera/esp32_camera.h"
 | 
			
		||||
#include "esphome/components/camera/camera.h"
 | 
			
		||||
#include "esphome/core/component.h"
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
#include "esphome/core/preferences.h"
 | 
			
		||||
@@ -32,7 +32,7 @@ class CameraWebServer : public Component {
 | 
			
		||||
  void loop() override;
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  std::shared_ptr<esphome::esp32_camera::CameraImage> wait_for_image_();
 | 
			
		||||
  std::shared_ptr<camera::CameraImage> wait_for_image_();
 | 
			
		||||
  esp_err_t handler_(struct httpd_req *req);
 | 
			
		||||
  esp_err_t streaming_handler_(struct httpd_req *req);
 | 
			
		||||
  esp_err_t snapshot_handler_(struct httpd_req *req);
 | 
			
		||||
@@ -40,7 +40,7 @@ class CameraWebServer : public Component {
 | 
			
		||||
  uint16_t port_{0};
 | 
			
		||||
  void *httpd_{nullptr};
 | 
			
		||||
  SemaphoreHandle_t semaphore_;
 | 
			
		||||
  std::shared_ptr<esphome::esp32_camera::CameraImage> image_;
 | 
			
		||||
  std::shared_ptr<camera::CameraImage> image_;
 | 
			
		||||
  bool running_{false};
 | 
			
		||||
  Mode mode_{STREAM};
 | 
			
		||||
};
 | 
			
		||||
 
 | 
			
		||||
@@ -109,6 +109,7 @@ void ESP32TouchComponent::loop() {
 | 
			
		||||
 | 
			
		||||
      // Only publish if state changed - this filters out repeated events
 | 
			
		||||
      if (new_state != child->last_state_) {
 | 
			
		||||
        child->initial_state_published_ = true;
 | 
			
		||||
        child->last_state_ = new_state;
 | 
			
		||||
        child->publish_state(new_state);
 | 
			
		||||
        // Original ESP32: ISR only fires when touched, release is detected by timeout
 | 
			
		||||
@@ -175,6 +176,9 @@ void ESP32TouchComponent::on_shutdown() {
 | 
			
		||||
void IRAM_ATTR ESP32TouchComponent::touch_isr_handler(void *arg) {
 | 
			
		||||
  ESP32TouchComponent *component = static_cast<ESP32TouchComponent *>(arg);
 | 
			
		||||
 | 
			
		||||
  uint32_t mask = 0;
 | 
			
		||||
  touch_ll_read_trigger_status_mask(&mask);
 | 
			
		||||
  touch_ll_clear_trigger_status_mask();
 | 
			
		||||
  touch_pad_clear_status();
 | 
			
		||||
 | 
			
		||||
  // INTERRUPT BEHAVIOR: On ESP32 v1 hardware, the interrupt fires when ANY configured
 | 
			
		||||
@@ -184,6 +188,11 @@ void IRAM_ATTR ESP32TouchComponent::touch_isr_handler(void *arg) {
 | 
			
		||||
  // as any pad remains touched. This allows us to detect both new touches and
 | 
			
		||||
  // continued touches, but releases must be detected by timeout in the main loop.
 | 
			
		||||
 | 
			
		||||
  // IMPORTANT: ESP32 v1 touch detection logic - INVERTED compared to v2!
 | 
			
		||||
  // ESP32 v1: Touch is detected when capacitance INCREASES, causing the measured value to DECREASE
 | 
			
		||||
  // Therefore: touched = (value < threshold)
 | 
			
		||||
  // This is opposite to ESP32-S2/S3 v2 where touched = (value > threshold)
 | 
			
		||||
 | 
			
		||||
  // Process all configured pads to check their current state
 | 
			
		||||
  // Note: ESP32 v1 doesn't tell us which specific pad triggered the interrupt,
 | 
			
		||||
  // so we must scan all configured pads to find which ones were touched
 | 
			
		||||
@@ -201,19 +210,12 @@ void IRAM_ATTR ESP32TouchComponent::touch_isr_handler(void *arg) {
 | 
			
		||||
      value = touch_ll_read_raw_data(pad);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Skip pads with 0 value - they haven't been measured in this cycle
 | 
			
		||||
    // This is important: not all pads are measured every interrupt cycle,
 | 
			
		||||
    // only those that the hardware has updated
 | 
			
		||||
    if (value == 0) {
 | 
			
		||||
    // Skip pads that aren’t in the trigger mask
 | 
			
		||||
    bool is_touched = (mask >> pad) & 1;
 | 
			
		||||
    if (!is_touched) {
 | 
			
		||||
      continue;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // IMPORTANT: ESP32 v1 touch detection logic - INVERTED compared to v2!
 | 
			
		||||
    // ESP32 v1: Touch is detected when capacitance INCREASES, causing the measured value to DECREASE
 | 
			
		||||
    // Therefore: touched = (value < threshold)
 | 
			
		||||
    // This is opposite to ESP32-S2/S3 v2 where touched = (value > threshold)
 | 
			
		||||
    bool is_touched = value < child->get_threshold();
 | 
			
		||||
 | 
			
		||||
    // Always send the current state - the main loop will filter for changes
 | 
			
		||||
    // We send both touched and untouched states because the ISR doesn't
 | 
			
		||||
    // track previous state (to keep ISR fast and simple)
 | 
			
		||||
 
 | 
			
		||||
@@ -180,6 +180,7 @@ async def to_code(config):
 | 
			
		||||
    cg.add(esp8266_ns.setup_preferences())
 | 
			
		||||
 | 
			
		||||
    cg.add_platformio_option("lib_ldf_mode", "off")
 | 
			
		||||
    cg.add_platformio_option("lib_compat_mode", "strict")
 | 
			
		||||
 | 
			
		||||
    cg.add_platformio_option("board", config[CONF_BOARD])
 | 
			
		||||
    cg.add_build_flag("-DUSE_ESP8266")
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										31
									
								
								esphome/components/esp8266/helpers.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								esphome/components/esp8266/helpers.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,31 @@
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
 | 
			
		||||
#ifdef USE_ESP8266
 | 
			
		||||
 | 
			
		||||
#include <osapi.h>
 | 
			
		||||
#include <user_interface.h>
 | 
			
		||||
// for xt_rsil()/xt_wsr_ps()
 | 
			
		||||
#include <Arduino.h>
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
 | 
			
		||||
uint32_t random_uint32() { return os_random(); }
 | 
			
		||||
bool random_bytes(uint8_t *data, size_t len) { return os_get_random(data, len) == 0; }
 | 
			
		||||
 | 
			
		||||
// ESP8266 doesn't have mutexes, but that shouldn't be an issue as it's single-core and non-preemptive OS.
 | 
			
		||||
Mutex::Mutex() {}
 | 
			
		||||
Mutex::~Mutex() {}
 | 
			
		||||
void Mutex::lock() {}
 | 
			
		||||
bool Mutex::try_lock() { return true; }
 | 
			
		||||
void Mutex::unlock() {}
 | 
			
		||||
 | 
			
		||||
IRAM_ATTR InterruptLock::InterruptLock() { state_ = xt_rsil(15); }
 | 
			
		||||
IRAM_ATTR InterruptLock::~InterruptLock() { xt_wsr_ps(state_); }
 | 
			
		||||
 | 
			
		||||
void get_mac_address_raw(uint8_t *mac) {  // NOLINT(readability-non-const-parameter)
 | 
			
		||||
  wifi_get_macaddr(STATION_IF, mac);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
 | 
			
		||||
#endif  // USE_ESP8266
 | 
			
		||||
@@ -20,14 +20,16 @@ adjusted_ids = set()
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = cv.All(
 | 
			
		||||
    cv.ensure_list(
 | 
			
		||||
        {
 | 
			
		||||
            cv.GenerateID(): cv.declare_id(EspLdo),
 | 
			
		||||
            cv.Required(CONF_VOLTAGE): cv.All(
 | 
			
		||||
                cv.voltage, cv.float_range(min=0.5, max=2.7)
 | 
			
		||||
            ),
 | 
			
		||||
            cv.Required(CONF_CHANNEL): cv.one_of(*CHANNELS, int=True),
 | 
			
		||||
            cv.Optional(CONF_ADJUSTABLE, default=False): cv.boolean,
 | 
			
		||||
        }
 | 
			
		||||
        cv.COMPONENT_SCHEMA.extend(
 | 
			
		||||
            {
 | 
			
		||||
                cv.GenerateID(): cv.declare_id(EspLdo),
 | 
			
		||||
                cv.Required(CONF_VOLTAGE): cv.All(
 | 
			
		||||
                    cv.voltage, cv.float_range(min=0.5, max=2.7)
 | 
			
		||||
                ),
 | 
			
		||||
                cv.Required(CONF_CHANNEL): cv.one_of(*CHANNELS, int=True),
 | 
			
		||||
                cv.Optional(CONF_ADJUSTABLE, default=False): cv.boolean,
 | 
			
		||||
            }
 | 
			
		||||
        )
 | 
			
		||||
    ),
 | 
			
		||||
    cv.only_with_esp_idf,
 | 
			
		||||
    only_on_variant(supported=[VARIANT_ESP32P4]),
 | 
			
		||||
 
 | 
			
		||||
@@ -17,6 +17,9 @@ class EspLdo : public Component {
 | 
			
		||||
  void set_adjustable(bool adjustable) { this->adjustable_ = adjustable; }
 | 
			
		||||
  void set_voltage(float voltage) { this->voltage_ = voltage; }
 | 
			
		||||
  void adjust_voltage(float voltage);
 | 
			
		||||
  float get_setup_priority() const override {
 | 
			
		||||
    return setup_priority::BUS;  // LDO setup should be done early
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  int channel_;
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										0
									
								
								esphome/components/gl_r01_i2c/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								esphome/components/gl_r01_i2c/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										68
									
								
								esphome/components/gl_r01_i2c/gl_r01_i2c.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								esphome/components/gl_r01_i2c/gl_r01_i2c.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,68 @@
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
#include "esphome/core/hal.h"
 | 
			
		||||
#include "gl_r01_i2c.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace gl_r01_i2c {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "gl_r01_i2c";
 | 
			
		||||
 | 
			
		||||
// Register definitions from datasheet
 | 
			
		||||
static const uint8_t REG_VERSION = 0x00;
 | 
			
		||||
static const uint8_t REG_DISTANCE = 0x02;
 | 
			
		||||
static const uint8_t REG_TRIGGER = 0x10;
 | 
			
		||||
static const uint8_t CMD_TRIGGER = 0xB0;
 | 
			
		||||
static const uint8_t RESTART_CMD1 = 0x5A;
 | 
			
		||||
static const uint8_t RESTART_CMD2 = 0xA5;
 | 
			
		||||
static const uint8_t READ_DELAY = 40;  // minimum milliseconds from datasheet to safely read measurement result
 | 
			
		||||
 | 
			
		||||
void GLR01I2CComponent::setup() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Setting up GL-R01 I2C...");
 | 
			
		||||
  // Verify sensor presence
 | 
			
		||||
  if (!this->read_byte_16(REG_VERSION, &this->version_)) {
 | 
			
		||||
    ESP_LOGE(TAG, "Failed to communicate with GL-R01 I2C sensor!");
 | 
			
		||||
    this->mark_failed();
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  ESP_LOGD(TAG, "Found GL-R01 I2C with version 0x%04X", this->version_);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void GLR01I2CComponent::dump_config() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "GL-R01 I2C:");
 | 
			
		||||
  ESP_LOGCONFIG(TAG, " Firmware Version: 0x%04X", this->version_);
 | 
			
		||||
  LOG_I2C_DEVICE(this);
 | 
			
		||||
  LOG_SENSOR(" ", "Distance", this);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void GLR01I2CComponent::update() {
 | 
			
		||||
  // Trigger a new measurement
 | 
			
		||||
  if (!this->write_byte(REG_TRIGGER, CMD_TRIGGER)) {
 | 
			
		||||
    ESP_LOGE(TAG, "Failed to trigger measurement!");
 | 
			
		||||
    this->status_set_warning();
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Schedule reading the result after the read delay
 | 
			
		||||
  this->set_timeout(READ_DELAY, [this]() { this->read_distance_(); });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void GLR01I2CComponent::read_distance_() {
 | 
			
		||||
  uint16_t distance = 0;
 | 
			
		||||
  if (!this->read_byte_16(REG_DISTANCE, &distance)) {
 | 
			
		||||
    ESP_LOGE(TAG, "Failed to read distance value!");
 | 
			
		||||
    this->status_set_warning();
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (distance == 0xFFFF) {
 | 
			
		||||
    ESP_LOGW(TAG, "Invalid measurement received!");
 | 
			
		||||
    this->status_set_warning();
 | 
			
		||||
  } else {
 | 
			
		||||
    ESP_LOGV(TAG, "Distance: %umm", distance);
 | 
			
		||||
    this->publish_state(distance);
 | 
			
		||||
    this->status_clear_warning();
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}  // namespace gl_r01_i2c
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
							
								
								
									
										22
									
								
								esphome/components/gl_r01_i2c/gl_r01_i2c.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								esphome/components/gl_r01_i2c/gl_r01_i2c.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,22 @@
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "esphome/core/component.h"
 | 
			
		||||
#include "esphome/components/sensor/sensor.h"
 | 
			
		||||
#include "esphome/components/i2c/i2c.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace gl_r01_i2c {
 | 
			
		||||
 | 
			
		||||
class GLR01I2CComponent : public sensor::Sensor, public i2c::I2CDevice, public PollingComponent {
 | 
			
		||||
 public:
 | 
			
		||||
  void setup() override;
 | 
			
		||||
  void dump_config() override;
 | 
			
		||||
  void update() override;
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  void read_distance_();
 | 
			
		||||
  uint16_t version_{0};
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace gl_r01_i2c
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
							
								
								
									
										36
									
								
								esphome/components/gl_r01_i2c/sensor.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								esphome/components/gl_r01_i2c/sensor.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,36 @@
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.components import i2c, sensor
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_ID,
 | 
			
		||||
    DEVICE_CLASS_DISTANCE,
 | 
			
		||||
    STATE_CLASS_MEASUREMENT,
 | 
			
		||||
    UNIT_MILLIMETER,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@pkejval"]
 | 
			
		||||
DEPENDENCIES = ["i2c"]
 | 
			
		||||
 | 
			
		||||
gl_r01_i2c_ns = cg.esphome_ns.namespace("gl_r01_i2c")
 | 
			
		||||
GLR01I2CComponent = gl_r01_i2c_ns.class_(
 | 
			
		||||
    "GLR01I2CComponent", i2c.I2CDevice, cg.PollingComponent
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = (
 | 
			
		||||
    sensor.sensor_schema(
 | 
			
		||||
        GLR01I2CComponent,
 | 
			
		||||
        unit_of_measurement=UNIT_MILLIMETER,
 | 
			
		||||
        accuracy_decimals=0,
 | 
			
		||||
        device_class=DEVICE_CLASS_DISTANCE,
 | 
			
		||||
        state_class=STATE_CLASS_MEASUREMENT,
 | 
			
		||||
    )
 | 
			
		||||
    .extend(cv.polling_component_schema("60s"))
 | 
			
		||||
    .extend(i2c.i2c_device_schema(0x74))
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
async def to_code(config):
 | 
			
		||||
    var = cg.new_Pvariable(config[CONF_ID])
 | 
			
		||||
    await cg.register_component(var, config)
 | 
			
		||||
    await sensor.register_sensor(var, config)
 | 
			
		||||
    await i2c.register_i2c_device(var, config)
 | 
			
		||||
@@ -1,11 +1,16 @@
 | 
			
		||||
import logging
 | 
			
		||||
 | 
			
		||||
from esphome import pins
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components import binary_sensor
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import CONF_PIN
 | 
			
		||||
from esphome.const import CONF_ID, CONF_NAME, CONF_NUMBER, CONF_PIN
 | 
			
		||||
from esphome.core import CORE
 | 
			
		||||
 | 
			
		||||
from .. import gpio_ns
 | 
			
		||||
 | 
			
		||||
_LOGGER = logging.getLogger(__name__)
 | 
			
		||||
 | 
			
		||||
GPIOBinarySensor = gpio_ns.class_(
 | 
			
		||||
    "GPIOBinarySensor", binary_sensor.BinarySensor, cg.Component
 | 
			
		||||
)
 | 
			
		||||
@@ -41,6 +46,22 @@ async def to_code(config):
 | 
			
		||||
    pin = await cg.gpio_pin_expression(config[CONF_PIN])
 | 
			
		||||
    cg.add(var.set_pin(pin))
 | 
			
		||||
 | 
			
		||||
    cg.add(var.set_use_interrupt(config[CONF_USE_INTERRUPT]))
 | 
			
		||||
    if config[CONF_USE_INTERRUPT]:
 | 
			
		||||
    # Check for ESP8266 GPIO16 interrupt limitation
 | 
			
		||||
    # GPIO16 on ESP8266 is a special pin that doesn't support interrupts through
 | 
			
		||||
    # the Arduino attachInterrupt() function. This is the only known GPIO pin
 | 
			
		||||
    # across all supported platforms that has this limitation, so we handle it
 | 
			
		||||
    # here instead of in the platform-specific code.
 | 
			
		||||
    use_interrupt = config[CONF_USE_INTERRUPT]
 | 
			
		||||
    if use_interrupt and CORE.is_esp8266 and config[CONF_PIN][CONF_NUMBER] == 16:
 | 
			
		||||
        _LOGGER.warning(
 | 
			
		||||
            "GPIO binary_sensor '%s': GPIO16 on ESP8266 doesn't support interrupts. "
 | 
			
		||||
            "Falling back to polling mode (same as in ESPHome <2025.7). "
 | 
			
		||||
            "The sensor will work exactly as before, but other pins have better "
 | 
			
		||||
            "performance with interrupts.",
 | 
			
		||||
            config.get(CONF_NAME, config[CONF_ID]),
 | 
			
		||||
        )
 | 
			
		||||
        use_interrupt = False
 | 
			
		||||
 | 
			
		||||
    cg.add(var.set_use_interrupt(use_interrupt))
 | 
			
		||||
    if use_interrupt:
 | 
			
		||||
        cg.add(var.set_interrupt_type(config[CONF_INTERRUPT_TYPE]))
 | 
			
		||||
 
 | 
			
		||||
@@ -45,3 +45,4 @@ async def to_code(config):
 | 
			
		||||
    cg.add_define("ESPHOME_BOARD", "host")
 | 
			
		||||
    cg.add_platformio_option("platform", "platformio/native")
 | 
			
		||||
    cg.add_platformio_option("lib_ldf_mode", "off")
 | 
			
		||||
    cg.add_platformio_option("lib_compat_mode", "strict")
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										57
									
								
								esphome/components/host/helpers.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								esphome/components/host/helpers.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,57 @@
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
 | 
			
		||||
#ifdef USE_HOST
 | 
			
		||||
 | 
			
		||||
#ifndef _WIN32
 | 
			
		||||
#include <net/if.h>
 | 
			
		||||
#include <netinet/in.h>
 | 
			
		||||
#include <sys/ioctl.h>
 | 
			
		||||
#endif
 | 
			
		||||
#include <unistd.h>
 | 
			
		||||
#include <limits>
 | 
			
		||||
#include <random>
 | 
			
		||||
 | 
			
		||||
#include "esphome/core/defines.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "helpers.host";
 | 
			
		||||
 | 
			
		||||
uint32_t random_uint32() {
 | 
			
		||||
  std::random_device dev;
 | 
			
		||||
  std::mt19937 rng(dev());
 | 
			
		||||
  std::uniform_int_distribution<uint32_t> dist(0, std::numeric_limits<uint32_t>::max());
 | 
			
		||||
  return dist(rng);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool random_bytes(uint8_t *data, size_t len) {
 | 
			
		||||
  FILE *fp = fopen("/dev/urandom", "r");
 | 
			
		||||
  if (fp == nullptr) {
 | 
			
		||||
    ESP_LOGW(TAG, "Could not open /dev/urandom, errno=%d", errno);
 | 
			
		||||
    exit(1);
 | 
			
		||||
  }
 | 
			
		||||
  size_t read = fread(data, 1, len, fp);
 | 
			
		||||
  if (read != len) {
 | 
			
		||||
    ESP_LOGW(TAG, "Not enough data from /dev/urandom");
 | 
			
		||||
    exit(1);
 | 
			
		||||
  }
 | 
			
		||||
  fclose(fp);
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Host platform uses std::mutex for proper thread synchronization
 | 
			
		||||
Mutex::Mutex() { handle_ = new std::mutex(); }
 | 
			
		||||
Mutex::~Mutex() { delete static_cast<std::mutex *>(handle_); }
 | 
			
		||||
void Mutex::lock() { static_cast<std::mutex *>(handle_)->lock(); }
 | 
			
		||||
bool Mutex::try_lock() { return static_cast<std::mutex *>(handle_)->try_lock(); }
 | 
			
		||||
void Mutex::unlock() { static_cast<std::mutex *>(handle_)->unlock(); }
 | 
			
		||||
 | 
			
		||||
void get_mac_address_raw(uint8_t *mac) {  // NOLINT(readability-non-const-parameter)
 | 
			
		||||
  static const uint8_t esphome_host_mac_address[6] = USE_ESPHOME_HOST_MAC_ADDRESS;
 | 
			
		||||
  memcpy(mac, esphome_host_mac_address, sizeof(esphome_host_mac_address));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
 | 
			
		||||
#endif  // USE_HOST
 | 
			
		||||
@@ -2,6 +2,7 @@ from esphome import automation
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components import esp32
 | 
			
		||||
from esphome.components.const import CONF_REQUEST_HEADERS
 | 
			
		||||
from esphome.config_helpers import filter_source_files_from_platform
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_ESP8266_DISABLE_SSL_SUPPORT,
 | 
			
		||||
@@ -13,6 +14,7 @@ from esphome.const import (
 | 
			
		||||
    CONF_URL,
 | 
			
		||||
    CONF_WATCHDOG_TIMEOUT,
 | 
			
		||||
    PLATFORM_HOST,
 | 
			
		||||
    PlatformFramework,
 | 
			
		||||
    __version__,
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import CORE, Lambda
 | 
			
		||||
@@ -319,3 +321,19 @@ async def http_request_action_to_code(config, action_id, template_arg, args):
 | 
			
		||||
        await automation.build_automation(trigger, [], conf)
 | 
			
		||||
 | 
			
		||||
    return var
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FILTER_SOURCE_FILES = filter_source_files_from_platform(
 | 
			
		||||
    {
 | 
			
		||||
        "http_request_host.cpp": {PlatformFramework.HOST_NATIVE},
 | 
			
		||||
        "http_request_arduino.cpp": {
 | 
			
		||||
            PlatformFramework.ESP32_ARDUINO,
 | 
			
		||||
            PlatformFramework.ESP8266_ARDUINO,
 | 
			
		||||
            PlatformFramework.RP2040_ARDUINO,
 | 
			
		||||
            PlatformFramework.BK72XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.RTL87XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.LN882X_ARDUINO,
 | 
			
		||||
        },
 | 
			
		||||
        "http_request_idf.cpp": {PlatformFramework.ESP32_IDF},
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
 
 | 
			
		||||
@@ -50,7 +50,8 @@ void HttpRequestUpdate::update_task(void *params) {
 | 
			
		||||
 | 
			
		||||
  if (container == nullptr || container->status_code != HTTP_STATUS_OK) {
 | 
			
		||||
    std::string msg = str_sprintf("Failed to fetch manifest from %s", this_update->source_url_.c_str());
 | 
			
		||||
    this_update->status_set_error(msg.c_str());
 | 
			
		||||
    // Defer to main loop to avoid race condition on component_state_ read-modify-write
 | 
			
		||||
    this_update->defer([this_update, msg]() { this_update->status_set_error(msg.c_str()); });
 | 
			
		||||
    UPDATE_RETURN;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@@ -58,7 +59,8 @@ void HttpRequestUpdate::update_task(void *params) {
 | 
			
		||||
  uint8_t *data = allocator.allocate(container->content_length);
 | 
			
		||||
  if (data == nullptr) {
 | 
			
		||||
    std::string msg = str_sprintf("Failed to allocate %zu bytes for manifest", container->content_length);
 | 
			
		||||
    this_update->status_set_error(msg.c_str());
 | 
			
		||||
    // Defer to main loop to avoid race condition on component_state_ read-modify-write
 | 
			
		||||
    this_update->defer([this_update, msg]() { this_update->status_set_error(msg.c_str()); });
 | 
			
		||||
    container->end();
 | 
			
		||||
    UPDATE_RETURN;
 | 
			
		||||
  }
 | 
			
		||||
@@ -120,7 +122,8 @@ void HttpRequestUpdate::update_task(void *params) {
 | 
			
		||||
 | 
			
		||||
  if (!valid) {
 | 
			
		||||
    std::string msg = str_sprintf("Failed to parse JSON from %s", this_update->source_url_.c_str());
 | 
			
		||||
    this_update->status_set_error(msg.c_str());
 | 
			
		||||
    // Defer to main loop to avoid race condition on component_state_ read-modify-write
 | 
			
		||||
    this_update->defer([this_update, msg]() { this_update->status_set_error(msg.c_str()); });
 | 
			
		||||
    UPDATE_RETURN;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@@ -147,18 +150,34 @@ void HttpRequestUpdate::update_task(void *params) {
 | 
			
		||||
    this_update->update_info_.current_version = current_version;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  bool trigger_update_available = false;
 | 
			
		||||
 | 
			
		||||
  if (this_update->update_info_.latest_version.empty() ||
 | 
			
		||||
      this_update->update_info_.latest_version == this_update->update_info_.current_version) {
 | 
			
		||||
    this_update->state_ = update::UPDATE_STATE_NO_UPDATE;
 | 
			
		||||
  } else {
 | 
			
		||||
    if (this_update->state_ != update::UPDATE_STATE_AVAILABLE) {
 | 
			
		||||
      trigger_update_available = true;
 | 
			
		||||
    }
 | 
			
		||||
    this_update->state_ = update::UPDATE_STATE_AVAILABLE;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  this_update->update_info_.has_progress = false;
 | 
			
		||||
  this_update->update_info_.progress = 0.0f;
 | 
			
		||||
  // Defer to main loop to ensure thread-safe execution of:
 | 
			
		||||
  // - status_clear_error() performs non-atomic read-modify-write on component_state_
 | 
			
		||||
  // - publish_state() triggers API callbacks that write to the shared protobuf buffer
 | 
			
		||||
  //   which can be corrupted if accessed concurrently from task and main loop threads
 | 
			
		||||
  // - update_available trigger to ensure consistent state when the trigger fires
 | 
			
		||||
  this_update->defer([this_update, trigger_update_available]() {
 | 
			
		||||
    this_update->update_info_.has_progress = false;
 | 
			
		||||
    this_update->update_info_.progress = 0.0f;
 | 
			
		||||
 | 
			
		||||
  this_update->status_clear_error();
 | 
			
		||||
  this_update->publish_state();
 | 
			
		||||
    this_update->status_clear_error();
 | 
			
		||||
    this_update->publish_state();
 | 
			
		||||
 | 
			
		||||
    if (trigger_update_available) {
 | 
			
		||||
      this_update->get_update_available_trigger()->trigger(this_update->update_info_);
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  UPDATE_RETURN;
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -111,8 +111,8 @@ CONFIG_SCHEMA = cv.All(
 | 
			
		||||
            cv.Optional(CONF_MOISTURE): sensor.sensor_schema(
 | 
			
		||||
                unit_of_measurement=UNIT_INTENSITY,
 | 
			
		||||
                accuracy_decimals=0,
 | 
			
		||||
                device_class=DEVICE_CLASS_PRECIPITATION_INTENSITY,
 | 
			
		||||
                state_class=STATE_CLASS_MEASUREMENT,
 | 
			
		||||
                icon="mdi:weather-rainy",
 | 
			
		||||
            ),
 | 
			
		||||
            cv.Optional(CONF_TEMPERATURE): sensor.sensor_schema(
 | 
			
		||||
                unit_of_measurement=UNIT_CELSIUS,
 | 
			
		||||
 
 | 
			
		||||
@@ -3,6 +3,7 @@ import logging
 | 
			
		||||
from esphome import pins
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components import esp32
 | 
			
		||||
from esphome.config_helpers import filter_source_files_from_platform
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_ADDRESS,
 | 
			
		||||
@@ -18,6 +19,7 @@ from esphome.const import (
 | 
			
		||||
    PLATFORM_ESP32,
 | 
			
		||||
    PLATFORM_ESP8266,
 | 
			
		||||
    PLATFORM_RP2040,
 | 
			
		||||
    PlatformFramework,
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import CORE, coroutine_with_priority
 | 
			
		||||
import esphome.final_validate as fv
 | 
			
		||||
@@ -205,3 +207,18 @@ def final_validate_device_schema(
 | 
			
		||||
        {cv.Required(CONF_I2C_ID): fv.id_declaration_match_schema(hub_schema)},
 | 
			
		||||
        extra=cv.ALLOW_EXTRA,
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FILTER_SOURCE_FILES = filter_source_files_from_platform(
 | 
			
		||||
    {
 | 
			
		||||
        "i2c_bus_arduino.cpp": {
 | 
			
		||||
            PlatformFramework.ESP32_ARDUINO,
 | 
			
		||||
            PlatformFramework.ESP8266_ARDUINO,
 | 
			
		||||
            PlatformFramework.RP2040_ARDUINO,
 | 
			
		||||
            PlatformFramework.BK72XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.RTL87XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.LN882X_ARDUINO,
 | 
			
		||||
        },
 | 
			
		||||
        "i2c_bus_esp_idf.cpp": {PlatformFramework.ESP32_IDF},
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
 
 | 
			
		||||
@@ -180,7 +180,7 @@ async def to_code(config):
 | 
			
		||||
    await speaker.register_speaker(var, config)
 | 
			
		||||
 | 
			
		||||
    if config[CONF_DAC_TYPE] == "internal":
 | 
			
		||||
        cg.add(var.set_internal_dac_mode(config[CONF_CHANNEL]))
 | 
			
		||||
        cg.add(var.set_internal_dac_mode(config[CONF_MODE]))
 | 
			
		||||
    else:
 | 
			
		||||
        cg.add(var.set_dout_pin(config[CONF_I2S_DOUT_PIN]))
 | 
			
		||||
        if use_legacy():
 | 
			
		||||
 
 | 
			
		||||
@@ -10,8 +10,10 @@ from PIL import Image, UnidentifiedImageError
 | 
			
		||||
 | 
			
		||||
from esphome import core, external_files
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components.const import CONF_BYTE_ORDER
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_DEFAULTS,
 | 
			
		||||
    CONF_DITHER,
 | 
			
		||||
    CONF_FILE,
 | 
			
		||||
    CONF_ICON,
 | 
			
		||||
@@ -38,6 +40,7 @@ CONF_OPAQUE = "opaque"
 | 
			
		||||
CONF_CHROMA_KEY = "chroma_key"
 | 
			
		||||
CONF_ALPHA_CHANNEL = "alpha_channel"
 | 
			
		||||
CONF_INVERT_ALPHA = "invert_alpha"
 | 
			
		||||
CONF_IMAGES = "images"
 | 
			
		||||
 | 
			
		||||
TRANSPARENCY_TYPES = (
 | 
			
		||||
    CONF_OPAQUE,
 | 
			
		||||
@@ -188,6 +191,10 @@ class ImageRGB565(ImageEncoder):
 | 
			
		||||
            dither,
 | 
			
		||||
            invert_alpha,
 | 
			
		||||
        )
 | 
			
		||||
        self.big_endian = True
 | 
			
		||||
 | 
			
		||||
    def set_big_endian(self, big_endian: bool) -> None:
 | 
			
		||||
        self.big_endian = big_endian
 | 
			
		||||
 | 
			
		||||
    def convert(self, image, path):
 | 
			
		||||
        return image.convert("RGBA")
 | 
			
		||||
@@ -205,10 +212,16 @@ class ImageRGB565(ImageEncoder):
 | 
			
		||||
                g = 1
 | 
			
		||||
                b = 0
 | 
			
		||||
        rgb = (r << 11) | (g << 5) | b
 | 
			
		||||
        self.data[self.index] = rgb >> 8
 | 
			
		||||
        self.index += 1
 | 
			
		||||
        self.data[self.index] = rgb & 0xFF
 | 
			
		||||
        self.index += 1
 | 
			
		||||
        if self.big_endian:
 | 
			
		||||
            self.data[self.index] = rgb >> 8
 | 
			
		||||
            self.index += 1
 | 
			
		||||
            self.data[self.index] = rgb & 0xFF
 | 
			
		||||
            self.index += 1
 | 
			
		||||
        else:
 | 
			
		||||
            self.data[self.index] = rgb & 0xFF
 | 
			
		||||
            self.index += 1
 | 
			
		||||
            self.data[self.index] = rgb >> 8
 | 
			
		||||
            self.index += 1
 | 
			
		||||
        if self.transparency == CONF_ALPHA_CHANNEL:
 | 
			
		||||
            if self.invert_alpha:
 | 
			
		||||
                a ^= 0xFF
 | 
			
		||||
@@ -364,7 +377,7 @@ def validate_file_shorthand(value):
 | 
			
		||||
    value = cv.string_strict(value)
 | 
			
		||||
    parts = value.strip().split(":")
 | 
			
		||||
    if len(parts) == 2 and parts[0] in MDI_SOURCES:
 | 
			
		||||
        match = re.match(r"[a-zA-Z0-9\-]+", parts[1])
 | 
			
		||||
        match = re.match(r"^[a-zA-Z0-9\-]+$", parts[1])
 | 
			
		||||
        if match is None:
 | 
			
		||||
            raise cv.Invalid(f"Could not parse mdi icon name from '{value}'.")
 | 
			
		||||
        return download_gh_svg(parts[1], parts[0])
 | 
			
		||||
@@ -434,20 +447,29 @@ def validate_type(image_types):
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def validate_settings(value):
 | 
			
		||||
    type = value[CONF_TYPE]
 | 
			
		||||
    """
 | 
			
		||||
    Validate the settings for a single image configuration.
 | 
			
		||||
    """
 | 
			
		||||
    conf_type = value[CONF_TYPE]
 | 
			
		||||
    type_class = IMAGE_TYPE[conf_type]
 | 
			
		||||
    transparency = value[CONF_TRANSPARENCY].lower()
 | 
			
		||||
    allow_config = IMAGE_TYPE[type].allow_config
 | 
			
		||||
    if transparency not in allow_config:
 | 
			
		||||
    if transparency not in type_class.allow_config:
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            f"Image format '{type}' cannot have transparency: {transparency}"
 | 
			
		||||
            f"Image format '{conf_type}' cannot have transparency: {transparency}"
 | 
			
		||||
        )
 | 
			
		||||
    invert_alpha = value.get(CONF_INVERT_ALPHA, False)
 | 
			
		||||
    if (
 | 
			
		||||
        invert_alpha
 | 
			
		||||
        and transparency != CONF_ALPHA_CHANNEL
 | 
			
		||||
        and CONF_INVERT_ALPHA not in allow_config
 | 
			
		||||
        and CONF_INVERT_ALPHA not in type_class.allow_config
 | 
			
		||||
    ):
 | 
			
		||||
        raise cv.Invalid("No alpha channel to invert")
 | 
			
		||||
    if value.get(CONF_BYTE_ORDER) is not None and not callable(
 | 
			
		||||
        getattr(type_class, "set_big_endian", None)
 | 
			
		||||
    ):
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            f"Image format '{conf_type}' does not support byte order configuration"
 | 
			
		||||
        )
 | 
			
		||||
    if file := value.get(CONF_FILE):
 | 
			
		||||
        file = Path(file)
 | 
			
		||||
        if is_svg_file(file):
 | 
			
		||||
@@ -456,31 +478,82 @@ def validate_settings(value):
 | 
			
		||||
            try:
 | 
			
		||||
                Image.open(file)
 | 
			
		||||
            except UnidentifiedImageError as exc:
 | 
			
		||||
                raise cv.Invalid(f"File can't be opened as image: {file}") from exc
 | 
			
		||||
                raise cv.Invalid(
 | 
			
		||||
                    f"File can't be opened as image: {file.absolute()}"
 | 
			
		||||
                ) from exc
 | 
			
		||||
    return value
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
IMAGE_ID_SCHEMA = {
 | 
			
		||||
    cv.Required(CONF_ID): cv.declare_id(Image_),
 | 
			
		||||
    cv.Required(CONF_FILE): cv.Any(validate_file_shorthand, TYPED_FILE_SCHEMA),
 | 
			
		||||
    cv.GenerateID(CONF_RAW_DATA_ID): cv.declare_id(cg.uint8),
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
OPTIONS_SCHEMA = {
 | 
			
		||||
    cv.Optional(CONF_RESIZE): cv.dimensions,
 | 
			
		||||
    cv.Optional(CONF_DITHER, default="NONE"): cv.one_of(
 | 
			
		||||
        "NONE", "FLOYDSTEINBERG", upper=True
 | 
			
		||||
    ),
 | 
			
		||||
    cv.Optional(CONF_INVERT_ALPHA, default=False): cv.boolean,
 | 
			
		||||
    cv.Optional(CONF_BYTE_ORDER): cv.one_of("BIG_ENDIAN", "LITTLE_ENDIAN", upper=True),
 | 
			
		||||
    cv.Optional(CONF_TRANSPARENCY, default=CONF_OPAQUE): validate_transparency(),
 | 
			
		||||
    cv.Optional(CONF_TYPE): validate_type(IMAGE_TYPE),
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
OPTIONS = [key.schema for key in OPTIONS_SCHEMA]
 | 
			
		||||
 | 
			
		||||
# image schema with no defaults, used with `CONF_IMAGES` in the config
 | 
			
		||||
IMAGE_SCHEMA_NO_DEFAULTS = {
 | 
			
		||||
    **IMAGE_ID_SCHEMA,
 | 
			
		||||
    **{cv.Optional(key): OPTIONS_SCHEMA[key] for key in OPTIONS},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
BASE_SCHEMA = cv.Schema(
 | 
			
		||||
    {
 | 
			
		||||
        cv.Required(CONF_ID): cv.declare_id(Image_),
 | 
			
		||||
        cv.Required(CONF_FILE): cv.Any(validate_file_shorthand, TYPED_FILE_SCHEMA),
 | 
			
		||||
        cv.Optional(CONF_RESIZE): cv.dimensions,
 | 
			
		||||
        cv.Optional(CONF_DITHER, default="NONE"): cv.one_of(
 | 
			
		||||
            "NONE", "FLOYDSTEINBERG", upper=True
 | 
			
		||||
        ),
 | 
			
		||||
        cv.Optional(CONF_INVERT_ALPHA, default=False): cv.boolean,
 | 
			
		||||
        cv.GenerateID(CONF_RAW_DATA_ID): cv.declare_id(cg.uint8),
 | 
			
		||||
        **IMAGE_ID_SCHEMA,
 | 
			
		||||
        **OPTIONS_SCHEMA,
 | 
			
		||||
    }
 | 
			
		||||
).add_extra(validate_settings)
 | 
			
		||||
 | 
			
		||||
IMAGE_SCHEMA = BASE_SCHEMA.extend(
 | 
			
		||||
    {
 | 
			
		||||
        cv.Required(CONF_TYPE): validate_type(IMAGE_TYPE),
 | 
			
		||||
        cv.Optional(CONF_TRANSPARENCY, default=CONF_OPAQUE): validate_transparency(),
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def validate_defaults(value):
 | 
			
		||||
    """
 | 
			
		||||
    Validate the options for images with defaults
 | 
			
		||||
    """
 | 
			
		||||
    defaults = value[CONF_DEFAULTS]
 | 
			
		||||
    result = []
 | 
			
		||||
    for index, image in enumerate(value[CONF_IMAGES]):
 | 
			
		||||
        type = image.get(CONF_TYPE, defaults.get(CONF_TYPE))
 | 
			
		||||
        if type is None:
 | 
			
		||||
            raise cv.Invalid(
 | 
			
		||||
                "Type is required either in the image config or in the defaults",
 | 
			
		||||
                path=[CONF_IMAGES, index],
 | 
			
		||||
            )
 | 
			
		||||
        type_class = IMAGE_TYPE[type]
 | 
			
		||||
        # A default byte order should be simply ignored if the type does not support it
 | 
			
		||||
        available_options = [*OPTIONS]
 | 
			
		||||
        if (
 | 
			
		||||
            not callable(getattr(type_class, "set_big_endian", None))
 | 
			
		||||
            and CONF_BYTE_ORDER not in image
 | 
			
		||||
        ):
 | 
			
		||||
            available_options.remove(CONF_BYTE_ORDER)
 | 
			
		||||
        config = {
 | 
			
		||||
            **{key: image.get(key, defaults.get(key)) for key in available_options},
 | 
			
		||||
            **{key.schema: image[key.schema] for key in IMAGE_ID_SCHEMA},
 | 
			
		||||
        }
 | 
			
		||||
        validate_settings(config)
 | 
			
		||||
        result.append(config)
 | 
			
		||||
    return result
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def typed_image_schema(image_type):
 | 
			
		||||
    """
 | 
			
		||||
    Construct a schema for a specific image type, allowing transparency options
 | 
			
		||||
@@ -523,10 +596,33 @@ def typed_image_schema(image_type):
 | 
			
		||||
 | 
			
		||||
# The config schema can be a (possibly empty) single list of images,
 | 
			
		||||
# or a dictionary of image types each with a list of images
 | 
			
		||||
CONFIG_SCHEMA = cv.Any(
 | 
			
		||||
    cv.Schema({cv.Optional(t.lower()): typed_image_schema(t) for t in IMAGE_TYPE}),
 | 
			
		||||
    cv.ensure_list(IMAGE_SCHEMA),
 | 
			
		||||
)
 | 
			
		||||
# or a dictionary with keys `defaults:` and `images:`
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _config_schema(config):
 | 
			
		||||
    if isinstance(config, list):
 | 
			
		||||
        return cv.Schema([IMAGE_SCHEMA])(config)
 | 
			
		||||
    if not isinstance(config, dict):
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            "Badly formed image configuration, expected a list or a dictionary"
 | 
			
		||||
        )
 | 
			
		||||
    if CONF_DEFAULTS in config or CONF_IMAGES in config:
 | 
			
		||||
        return validate_defaults(
 | 
			
		||||
            cv.Schema(
 | 
			
		||||
                {
 | 
			
		||||
                    cv.Required(CONF_DEFAULTS): OPTIONS_SCHEMA,
 | 
			
		||||
                    cv.Required(CONF_IMAGES): cv.ensure_list(IMAGE_SCHEMA_NO_DEFAULTS),
 | 
			
		||||
                }
 | 
			
		||||
            )(config)
 | 
			
		||||
        )
 | 
			
		||||
    if CONF_ID in config or CONF_FILE in config:
 | 
			
		||||
        return cv.ensure_list(IMAGE_SCHEMA)([config])
 | 
			
		||||
    return cv.Schema(
 | 
			
		||||
        {cv.Optional(t.lower()): typed_image_schema(t) for t in IMAGE_TYPE}
 | 
			
		||||
    )(config)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = _config_schema
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
async def write_image(config, all_frames=False):
 | 
			
		||||
@@ -585,6 +681,9 @@ async def write_image(config, all_frames=False):
 | 
			
		||||
 | 
			
		||||
    total_rows = height * frame_count
 | 
			
		||||
    encoder = IMAGE_TYPE[type](width, total_rows, transparency, dither, invert_alpha)
 | 
			
		||||
    if byte_order := config.get(CONF_BYTE_ORDER):
 | 
			
		||||
        # Check for valid type has already been done in validate_settings
 | 
			
		||||
        encoder.set_big_endian(byte_order == "BIG_ENDIAN")
 | 
			
		||||
    for frame_index in range(frame_count):
 | 
			
		||||
        image.seek(frame_index)
 | 
			
		||||
        pixels = encoder.convert(image.resize((width, height)), path).getdata()
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,7 @@
 | 
			
		||||
from esphome import pins
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components import display, i2c
 | 
			
		||||
from esphome.components.esp32 import CONF_CPU_FREQUENCY
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_FULL_UPDATE_EVERY,
 | 
			
		||||
@@ -13,7 +14,9 @@ from esphome.const import (
 | 
			
		||||
    CONF_PAGES,
 | 
			
		||||
    CONF_TRANSFORM,
 | 
			
		||||
    CONF_WAKEUP_PIN,
 | 
			
		||||
    PLATFORM_ESP32,
 | 
			
		||||
)
 | 
			
		||||
import esphome.final_validate as fv
 | 
			
		||||
 | 
			
		||||
DEPENDENCIES = ["i2c", "esp32"]
 | 
			
		||||
AUTO_LOAD = ["psram"]
 | 
			
		||||
@@ -120,6 +123,18 @@ CONFIG_SCHEMA = cv.All(
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def _validate_cpu_frequency(config):
 | 
			
		||||
    esp32_config = fv.full_config.get()[PLATFORM_ESP32]
 | 
			
		||||
    if esp32_config[CONF_CPU_FREQUENCY] != "240MHZ":
 | 
			
		||||
        raise cv.Invalid(
 | 
			
		||||
            "Inkplate requires 240MHz CPU frequency (set in esp32 component)"
 | 
			
		||||
        )
 | 
			
		||||
    return config
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FINAL_VALIDATE_SCHEMA = _validate_cpu_frequency
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
async def to_code(config):
 | 
			
		||||
    var = cg.new_Pvariable(config[CONF_ID])
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -14,8 +14,8 @@ from esphome.const import (
 | 
			
		||||
 | 
			
		||||
from .. import CONF_LD2410_ID, LD2410Component, ld2410_ns
 | 
			
		||||
 | 
			
		||||
FactoryResetButton = ld2410_ns.class_("FactoryResetButton", button.Button)
 | 
			
		||||
QueryButton = ld2410_ns.class_("QueryButton", button.Button)
 | 
			
		||||
ResetButton = ld2410_ns.class_("ResetButton", button.Button)
 | 
			
		||||
RestartButton = ld2410_ns.class_("RestartButton", button.Button)
 | 
			
		||||
 | 
			
		||||
CONF_QUERY_PARAMS = "query_params"
 | 
			
		||||
@@ -23,7 +23,7 @@ CONF_QUERY_PARAMS = "query_params"
 | 
			
		||||
CONFIG_SCHEMA = {
 | 
			
		||||
    cv.GenerateID(CONF_LD2410_ID): cv.use_id(LD2410Component),
 | 
			
		||||
    cv.Optional(CONF_FACTORY_RESET): button.button_schema(
 | 
			
		||||
        ResetButton,
 | 
			
		||||
        FactoryResetButton,
 | 
			
		||||
        device_class=DEVICE_CLASS_RESTART,
 | 
			
		||||
        entity_category=ENTITY_CATEGORY_CONFIG,
 | 
			
		||||
        icon=ICON_RESTART_ALERT,
 | 
			
		||||
@@ -47,7 +47,7 @@ async def to_code(config):
 | 
			
		||||
    if factory_reset_config := config.get(CONF_FACTORY_RESET):
 | 
			
		||||
        b = await button.new_button(factory_reset_config)
 | 
			
		||||
        await cg.register_parented(b, config[CONF_LD2410_ID])
 | 
			
		||||
        cg.add(ld2410_component.set_reset_button(b))
 | 
			
		||||
        cg.add(ld2410_component.set_factory_reset_button(b))
 | 
			
		||||
    if restart_config := config.get(CONF_RESTART):
 | 
			
		||||
        b = await button.new_button(restart_config)
 | 
			
		||||
        await cg.register_parented(b, config[CONF_LD2410_ID])
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,9 @@
 | 
			
		||||
#include "factory_reset_button.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2410 {
 | 
			
		||||
 | 
			
		||||
void FactoryResetButton::press_action() { this->parent_->factory_reset(); }
 | 
			
		||||
 | 
			
		||||
}  // namespace ld2410
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -6,9 +6,9 @@
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2410 {
 | 
			
		||||
 | 
			
		||||
class ResetButton : public button::Button, public Parented<LD2410Component> {
 | 
			
		||||
class FactoryResetButton : public button::Button, public Parented<LD2410Component> {
 | 
			
		||||
 public:
 | 
			
		||||
  ResetButton() = default;
 | 
			
		||||
  FactoryResetButton() = default;
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  void press_action() override;
 | 
			
		||||
@@ -1,9 +0,0 @@
 | 
			
		||||
#include "reset_button.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2410 {
 | 
			
		||||
 | 
			
		||||
void ResetButton::press_action() { this->parent_->factory_reset(); }
 | 
			
		||||
 | 
			
		||||
}  // namespace ld2410
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -18,11 +18,10 @@ namespace esphome {
 | 
			
		||||
namespace ld2410 {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "ld2410";
 | 
			
		||||
static const char *const NO_MAC = "08:05:04:03:02:01";
 | 
			
		||||
static const char *const UNKNOWN_MAC = "unknown";
 | 
			
		||||
static const char *const VERSION_FMT = "%u.%02X.%02X%02X%02X%02X";
 | 
			
		||||
 | 
			
		||||
enum BaudRateStructure : uint8_t {
 | 
			
		||||
enum BaudRate : uint8_t {
 | 
			
		||||
  BAUD_RATE_9600 = 1,
 | 
			
		||||
  BAUD_RATE_19200 = 2,
 | 
			
		||||
  BAUD_RATE_38400 = 3,
 | 
			
		||||
@@ -33,23 +32,23 @@ enum BaudRateStructure : uint8_t {
 | 
			
		||||
  BAUD_RATE_460800 = 8,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum DistanceResolutionStructure : uint8_t {
 | 
			
		||||
enum DistanceResolution : uint8_t {
 | 
			
		||||
  DISTANCE_RESOLUTION_0_2 = 0x01,
 | 
			
		||||
  DISTANCE_RESOLUTION_0_75 = 0x00,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum LightFunctionStructure : uint8_t {
 | 
			
		||||
enum LightFunction : uint8_t {
 | 
			
		||||
  LIGHT_FUNCTION_OFF = 0x00,
 | 
			
		||||
  LIGHT_FUNCTION_BELOW = 0x01,
 | 
			
		||||
  LIGHT_FUNCTION_ABOVE = 0x02,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum OutPinLevelStructure : uint8_t {
 | 
			
		||||
enum OutPinLevel : uint8_t {
 | 
			
		||||
  OUT_PIN_LEVEL_LOW = 0x00,
 | 
			
		||||
  OUT_PIN_LEVEL_HIGH = 0x01,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum PeriodicDataStructure : uint8_t {
 | 
			
		||||
enum PeriodicData : uint8_t {
 | 
			
		||||
  DATA_TYPES = 6,
 | 
			
		||||
  TARGET_STATES = 8,
 | 
			
		||||
  MOVING_TARGET_LOW = 9,
 | 
			
		||||
@@ -67,12 +66,12 @@ enum PeriodicDataStructure : uint8_t {
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum PeriodicDataValue : uint8_t {
 | 
			
		||||
  HEAD = 0xAA,
 | 
			
		||||
  END = 0x55,
 | 
			
		||||
  HEADER = 0xAA,
 | 
			
		||||
  FOOTER = 0x55,
 | 
			
		||||
  CHECK = 0x00,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum AckDataStructure : uint8_t {
 | 
			
		||||
enum AckData : uint8_t {
 | 
			
		||||
  COMMAND = 6,
 | 
			
		||||
  COMMAND_STATUS = 7,
 | 
			
		||||
};
 | 
			
		||||
@@ -80,11 +79,11 @@ enum AckDataStructure : uint8_t {
 | 
			
		||||
// Memory-efficient lookup tables
 | 
			
		||||
struct StringToUint8 {
 | 
			
		||||
  const char *str;
 | 
			
		||||
  uint8_t value;
 | 
			
		||||
  const uint8_t value;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct Uint8ToString {
 | 
			
		||||
  uint8_t value;
 | 
			
		||||
  const uint8_t value;
 | 
			
		||||
  const char *str;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
@@ -144,96 +143,114 @@ template<size_t N> const char *find_str(const Uint8ToString (&arr)[N], uint8_t v
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Commands
 | 
			
		||||
static const uint8_t CMD_ENABLE_CONF = 0xFF;
 | 
			
		||||
static const uint8_t CMD_DISABLE_CONF = 0xFE;
 | 
			
		||||
static const uint8_t CMD_ENABLE_ENG = 0x62;
 | 
			
		||||
static const uint8_t CMD_DISABLE_ENG = 0x63;
 | 
			
		||||
static const uint8_t CMD_MAXDIST_DURATION = 0x60;
 | 
			
		||||
static const uint8_t CMD_QUERY = 0x61;
 | 
			
		||||
static const uint8_t CMD_GATE_SENS = 0x64;
 | 
			
		||||
static const uint8_t CMD_VERSION = 0xA0;
 | 
			
		||||
static const uint8_t CMD_QUERY_DISTANCE_RESOLUTION = 0xAB;
 | 
			
		||||
static const uint8_t CMD_SET_DISTANCE_RESOLUTION = 0xAA;
 | 
			
		||||
static const uint8_t CMD_QUERY_LIGHT_CONTROL = 0xAE;
 | 
			
		||||
static const uint8_t CMD_SET_LIGHT_CONTROL = 0xAD;
 | 
			
		||||
static const uint8_t CMD_SET_BAUD_RATE = 0xA1;
 | 
			
		||||
static const uint8_t CMD_BT_PASSWORD = 0xA9;
 | 
			
		||||
static const uint8_t CMD_MAC = 0xA5;
 | 
			
		||||
static const uint8_t CMD_RESET = 0xA2;
 | 
			
		||||
static const uint8_t CMD_RESTART = 0xA3;
 | 
			
		||||
static const uint8_t CMD_BLUETOOTH = 0xA4;
 | 
			
		||||
static constexpr uint8_t CMD_ENABLE_CONF = 0xFF;
 | 
			
		||||
static constexpr uint8_t CMD_DISABLE_CONF = 0xFE;
 | 
			
		||||
static constexpr uint8_t CMD_ENABLE_ENG = 0x62;
 | 
			
		||||
static constexpr uint8_t CMD_DISABLE_ENG = 0x63;
 | 
			
		||||
static constexpr uint8_t CMD_MAXDIST_DURATION = 0x60;
 | 
			
		||||
static constexpr uint8_t CMD_QUERY = 0x61;
 | 
			
		||||
static constexpr uint8_t CMD_GATE_SENS = 0x64;
 | 
			
		||||
static constexpr uint8_t CMD_QUERY_VERSION = 0xA0;
 | 
			
		||||
static constexpr uint8_t CMD_QUERY_DISTANCE_RESOLUTION = 0xAB;
 | 
			
		||||
static constexpr uint8_t CMD_SET_DISTANCE_RESOLUTION = 0xAA;
 | 
			
		||||
static constexpr uint8_t CMD_QUERY_LIGHT_CONTROL = 0xAE;
 | 
			
		||||
static constexpr uint8_t CMD_SET_LIGHT_CONTROL = 0xAD;
 | 
			
		||||
static constexpr uint8_t CMD_SET_BAUD_RATE = 0xA1;
 | 
			
		||||
static constexpr uint8_t CMD_BT_PASSWORD = 0xA9;
 | 
			
		||||
static constexpr uint8_t CMD_QUERY_MAC_ADDRESS = 0xA5;
 | 
			
		||||
static constexpr uint8_t CMD_RESET = 0xA2;
 | 
			
		||||
static constexpr uint8_t CMD_RESTART = 0xA3;
 | 
			
		||||
static constexpr uint8_t CMD_BLUETOOTH = 0xA4;
 | 
			
		||||
// Commands values
 | 
			
		||||
static const uint8_t CMD_MAX_MOVE_VALUE = 0x00;
 | 
			
		||||
static const uint8_t CMD_MAX_STILL_VALUE = 0x01;
 | 
			
		||||
static const uint8_t CMD_DURATION_VALUE = 0x02;
 | 
			
		||||
static constexpr uint8_t CMD_MAX_MOVE_VALUE = 0x00;
 | 
			
		||||
static constexpr uint8_t CMD_MAX_STILL_VALUE = 0x01;
 | 
			
		||||
static constexpr uint8_t CMD_DURATION_VALUE = 0x02;
 | 
			
		||||
// Header & Footer size
 | 
			
		||||
static constexpr uint8_t HEADER_FOOTER_SIZE = 4;
 | 
			
		||||
// Command Header & Footer
 | 
			
		||||
static const uint8_t CMD_FRAME_HEADER[4] = {0xFD, 0xFC, 0xFB, 0xFA};
 | 
			
		||||
static const uint8_t CMD_FRAME_END[4] = {0x04, 0x03, 0x02, 0x01};
 | 
			
		||||
static constexpr uint8_t CMD_FRAME_HEADER[HEADER_FOOTER_SIZE] = {0xFD, 0xFC, 0xFB, 0xFA};
 | 
			
		||||
static constexpr uint8_t CMD_FRAME_FOOTER[HEADER_FOOTER_SIZE] = {0x04, 0x03, 0x02, 0x01};
 | 
			
		||||
// Data Header & Footer
 | 
			
		||||
static const uint8_t DATA_FRAME_HEADER[4] = {0xF4, 0xF3, 0xF2, 0xF1};
 | 
			
		||||
static const uint8_t DATA_FRAME_END[4] = {0xF8, 0xF7, 0xF6, 0xF5};
 | 
			
		||||
static constexpr uint8_t DATA_FRAME_HEADER[HEADER_FOOTER_SIZE] = {0xF4, 0xF3, 0xF2, 0xF1};
 | 
			
		||||
static constexpr uint8_t DATA_FRAME_FOOTER[HEADER_FOOTER_SIZE] = {0xF8, 0xF7, 0xF6, 0xF5};
 | 
			
		||||
// MAC address the module uses when Bluetooth is disabled
 | 
			
		||||
static constexpr uint8_t NO_MAC[] = {0x08, 0x05, 0x04, 0x03, 0x02, 0x01};
 | 
			
		||||
 | 
			
		||||
static inline int two_byte_to_int(char firstbyte, char secondbyte) { return (int16_t) (secondbyte << 8) + firstbyte; }
 | 
			
		||||
 | 
			
		||||
static inline bool validate_header_footer(const uint8_t *header_footer, const uint8_t *buffer) {
 | 
			
		||||
  return std::memcmp(header_footer, buffer, HEADER_FOOTER_SIZE) == 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::dump_config() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "LD2410:");
 | 
			
		||||
  std::string mac_str =
 | 
			
		||||
      mac_address_is_valid(this->mac_address_) ? format_mac_address_pretty(this->mac_address_) : UNKNOWN_MAC;
 | 
			
		||||
  std::string version = str_sprintf(VERSION_FMT, this->version_[1], this->version_[0], this->version_[5],
 | 
			
		||||
                                    this->version_[4], this->version_[3], this->version_[2]);
 | 
			
		||||
  ESP_LOGCONFIG(TAG,
 | 
			
		||||
                "LD2410:\n"
 | 
			
		||||
                "  Firmware version: %s\n"
 | 
			
		||||
                "  MAC address: %s\n"
 | 
			
		||||
                "  Throttle: %u ms",
 | 
			
		||||
                version.c_str(), mac_str.c_str(), this->throttle_);
 | 
			
		||||
#ifdef USE_BINARY_SENSOR
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "TargetBinarySensor", this->target_binary_sensor_);
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "MovingTargetBinarySensor", this->moving_target_binary_sensor_);
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "StillTargetBinarySensor", this->still_target_binary_sensor_);
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "OutPinPresenceStatusBinarySensor", this->out_pin_presence_status_binary_sensor_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
  LOG_SWITCH("  ", "EngineeringModeSwitch", this->engineering_mode_switch_);
 | 
			
		||||
  LOG_SWITCH("  ", "BluetoothSwitch", this->bluetooth_switch_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_BUTTON
 | 
			
		||||
  LOG_BUTTON("  ", "ResetButton", this->reset_button_);
 | 
			
		||||
  LOG_BUTTON("  ", "RestartButton", this->restart_button_);
 | 
			
		||||
  LOG_BUTTON("  ", "QueryButton", this->query_button_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Binary Sensors:");
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "Target", this->target_binary_sensor_);
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "MovingTarget", this->moving_target_binary_sensor_);
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "StillTarget", this->still_target_binary_sensor_);
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "OutPinPresenceStatus", this->out_pin_presence_status_binary_sensor_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
  LOG_SENSOR("  ", "LightSensor", this->light_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "MovingTargetDistanceSensor", this->moving_target_distance_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "StillTargetDistanceSensor", this->still_target_distance_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "MovingTargetEnergySensor", this->moving_target_energy_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "StillTargetEnergySensor", this->still_target_energy_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "DetectionDistanceSensor", this->detection_distance_sensor_);
 | 
			
		||||
  for (sensor::Sensor *s : this->gate_still_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "NthGateStillSesnsor", s);
 | 
			
		||||
  }
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Sensors:");
 | 
			
		||||
  LOG_SENSOR("  ", "Light", this->light_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "DetectionDistance", this->detection_distance_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "MovingTargetDistance", this->moving_target_distance_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "MovingTargetEnergy", this->moving_target_energy_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "StillTargetDistance", this->still_target_distance_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "StillTargetEnergy", this->still_target_energy_sensor_);
 | 
			
		||||
  for (sensor::Sensor *s : this->gate_move_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "NthGateMoveSesnsor", s);
 | 
			
		||||
    LOG_SENSOR("  ", "GateMove", s);
 | 
			
		||||
  }
 | 
			
		||||
  for (sensor::Sensor *s : this->gate_still_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "GateStill", s);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
  LOG_TEXT_SENSOR("  ", "VersionTextSensor", this->version_text_sensor_);
 | 
			
		||||
  LOG_TEXT_SENSOR("  ", "MacTextSensor", this->mac_text_sensor_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
  LOG_SELECT("  ", "LightFunctionSelect", this->light_function_select_);
 | 
			
		||||
  LOG_SELECT("  ", "OutPinLevelSelect", this->out_pin_level_select_);
 | 
			
		||||
  LOG_SELECT("  ", "DistanceResolutionSelect", this->distance_resolution_select_);
 | 
			
		||||
  LOG_SELECT("  ", "BaudRateSelect", this->baud_rate_select_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Text Sensors:");
 | 
			
		||||
  LOG_TEXT_SENSOR("  ", "Mac", this->mac_text_sensor_);
 | 
			
		||||
  LOG_TEXT_SENSOR("  ", "Version", this->version_text_sensor_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
  LOG_NUMBER("  ", "LightThresholdNumber", this->light_threshold_number_);
 | 
			
		||||
  LOG_NUMBER("  ", "MaxStillDistanceGateNumber", this->max_still_distance_gate_number_);
 | 
			
		||||
  LOG_NUMBER("  ", "MaxMoveDistanceGateNumber", this->max_move_distance_gate_number_);
 | 
			
		||||
  LOG_NUMBER("  ", "TimeoutNumber", this->timeout_number_);
 | 
			
		||||
  for (number::Number *n : this->gate_still_threshold_numbers_) {
 | 
			
		||||
    LOG_NUMBER("  ", "Still Thresholds Number", n);
 | 
			
		||||
  }
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Numbers:");
 | 
			
		||||
  LOG_NUMBER("  ", "LightThreshold", this->light_threshold_number_);
 | 
			
		||||
  LOG_NUMBER("  ", "MaxMoveDistanceGate", this->max_move_distance_gate_number_);
 | 
			
		||||
  LOG_NUMBER("  ", "MaxStillDistanceGate", this->max_still_distance_gate_number_);
 | 
			
		||||
  LOG_NUMBER("  ", "Timeout", this->timeout_number_);
 | 
			
		||||
  for (number::Number *n : this->gate_move_threshold_numbers_) {
 | 
			
		||||
    LOG_NUMBER("  ", "Move Thresholds Number", n);
 | 
			
		||||
    LOG_NUMBER("  ", "MoveThreshold", n);
 | 
			
		||||
  }
 | 
			
		||||
  for (number::Number *n : this->gate_still_threshold_numbers_) {
 | 
			
		||||
    LOG_NUMBER("  ", "StillThreshold", n);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
  this->read_all_info();
 | 
			
		||||
  ESP_LOGCONFIG(TAG,
 | 
			
		||||
                "  Throttle: %ums\n"
 | 
			
		||||
                "  MAC address: %s\n"
 | 
			
		||||
                "  Firmware version: %s",
 | 
			
		||||
                this->throttle_, this->mac_ == NO_MAC ? UNKNOWN_MAC : this->mac_.c_str(), this->version_.c_str());
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Selects:");
 | 
			
		||||
  LOG_SELECT("  ", "BaudRate", this->baud_rate_select_);
 | 
			
		||||
  LOG_SELECT("  ", "DistanceResolution", this->distance_resolution_select_);
 | 
			
		||||
  LOG_SELECT("  ", "LightFunction", this->light_function_select_);
 | 
			
		||||
  LOG_SELECT("  ", "OutPinLevel", this->out_pin_level_select_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Switches:");
 | 
			
		||||
  LOG_SWITCH("  ", "Bluetooth", this->bluetooth_switch_);
 | 
			
		||||
  LOG_SWITCH("  ", "EngineeringMode", this->engineering_mode_switch_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_BUTTON
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Buttons:");
 | 
			
		||||
  LOG_BUTTON("  ", "FactoryReset", this->factory_reset_button_);
 | 
			
		||||
  LOG_BUTTON("  ", "Query", this->query_button_);
 | 
			
		||||
  LOG_BUTTON("  ", "Restart", this->restart_button_);
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::setup() {
 | 
			
		||||
@@ -246,12 +263,12 @@ void LD2410Component::read_all_info() {
 | 
			
		||||
  this->get_version_();
 | 
			
		||||
  this->get_mac_();
 | 
			
		||||
  this->get_distance_resolution_();
 | 
			
		||||
  this->get_light_control_();
 | 
			
		||||
  this->query_light_control_();
 | 
			
		||||
  this->query_parameters_();
 | 
			
		||||
  this->set_config_mode_(false);
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
  const auto baud_rate = std::to_string(this->parent_->get_baud_rate());
 | 
			
		||||
  if (this->baud_rate_select_ != nullptr && this->baud_rate_select_->state != baud_rate) {
 | 
			
		||||
  if (this->baud_rate_select_ != nullptr) {
 | 
			
		||||
    this->baud_rate_select_->publish_state(baud_rate);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
@@ -264,66 +281,57 @@ void LD2410Component::restart_and_read_all_info() {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::loop() {
 | 
			
		||||
  const int max_line_length = 80;
 | 
			
		||||
  static uint8_t buffer[max_line_length];
 | 
			
		||||
 | 
			
		||||
  while (available()) {
 | 
			
		||||
    this->readline_(read(), buffer, max_line_length);
 | 
			
		||||
  while (this->available()) {
 | 
			
		||||
    this->readline_(this->read());
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::send_command_(uint8_t command, const uint8_t *command_value, int command_value_len) {
 | 
			
		||||
void LD2410Component::send_command_(uint8_t command, const uint8_t *command_value, uint8_t command_value_len) {
 | 
			
		||||
  ESP_LOGV(TAG, "Sending COMMAND %02X", command);
 | 
			
		||||
  // frame start bytes
 | 
			
		||||
  this->write_array(CMD_FRAME_HEADER, 4);
 | 
			
		||||
  // frame header bytes
 | 
			
		||||
  this->write_array(CMD_FRAME_HEADER, sizeof(CMD_FRAME_HEADER));
 | 
			
		||||
  // length bytes
 | 
			
		||||
  int len = 2;
 | 
			
		||||
  if (command_value != nullptr)
 | 
			
		||||
  uint8_t len = 2;
 | 
			
		||||
  if (command_value != nullptr) {
 | 
			
		||||
    len += command_value_len;
 | 
			
		||||
  this->write_byte(lowbyte(len));
 | 
			
		||||
  this->write_byte(highbyte(len));
 | 
			
		||||
 | 
			
		||||
  // command
 | 
			
		||||
  this->write_byte(lowbyte(command));
 | 
			
		||||
  this->write_byte(highbyte(command));
 | 
			
		||||
 | 
			
		||||
  }
 | 
			
		||||
  // 2 length bytes (low, high) + 2 command bytes (low, high)
 | 
			
		||||
  uint8_t len_cmd[] = {len, 0x00, command, 0x00};
 | 
			
		||||
  this->write_array(len_cmd, sizeof(len_cmd));
 | 
			
		||||
  // command value bytes
 | 
			
		||||
  if (command_value != nullptr) {
 | 
			
		||||
    for (int i = 0; i < command_value_len; i++) {
 | 
			
		||||
      this->write_byte(command_value[i]);
 | 
			
		||||
    }
 | 
			
		||||
    this->write_array(command_value, command_value_len);
 | 
			
		||||
  }
 | 
			
		||||
  // frame end bytes
 | 
			
		||||
  this->write_array(CMD_FRAME_END, 4);
 | 
			
		||||
  // frame footer bytes
 | 
			
		||||
  this->write_array(CMD_FRAME_FOOTER, sizeof(CMD_FRAME_FOOTER));
 | 
			
		||||
  // FIXME to remove
 | 
			
		||||
  delay(50);  // NOLINT
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::handle_periodic_data_(uint8_t *buffer, int len) {
 | 
			
		||||
  if (len < 12)
 | 
			
		||||
    return;  // 4 frame start bytes + 2 length bytes + 1 data end byte + 1 crc byte + 4 frame end bytes
 | 
			
		||||
  if (buffer[0] != 0xF4 || buffer[1] != 0xF3 || buffer[2] != 0xF2 || buffer[3] != 0xF1)  // check 4 frame start bytes
 | 
			
		||||
void LD2410Component::handle_periodic_data_() {
 | 
			
		||||
  // Reduce data update rate to reduce home assistant database growth
 | 
			
		||||
  // Check this first to prevent unnecessary processing done in later checks/parsing
 | 
			
		||||
  if (App.get_loop_component_start_time() - this->last_periodic_millis_ < this->throttle_) {
 | 
			
		||||
    return;
 | 
			
		||||
  if (buffer[7] != HEAD || buffer[len - 6] != END || buffer[len - 5] != CHECK)  // Check constant values
 | 
			
		||||
    return;  // data head=0xAA, data end=0x55, crc=0x00
 | 
			
		||||
 | 
			
		||||
  /*
 | 
			
		||||
    Reduce data update rate to prevent home assistant database size grow fast
 | 
			
		||||
  */
 | 
			
		||||
  int32_t current_millis = App.get_loop_component_start_time();
 | 
			
		||||
  if (current_millis - last_periodic_millis_ < this->throttle_)
 | 
			
		||||
  }
 | 
			
		||||
  // 4 frame header bytes + 2 length bytes + 1 data end byte + 1 crc byte + 4 frame footer bytes
 | 
			
		||||
  // data header=0xAA, data footer=0x55, crc=0x00
 | 
			
		||||
  if (this->buffer_pos_ < 12 || !ld2410::validate_header_footer(DATA_FRAME_HEADER, this->buffer_data_) ||
 | 
			
		||||
      this->buffer_data_[7] != HEADER || this->buffer_data_[this->buffer_pos_ - 6] != FOOTER ||
 | 
			
		||||
      this->buffer_data_[this->buffer_pos_ - 5] != CHECK) {
 | 
			
		||||
    return;
 | 
			
		||||
  last_periodic_millis_ = current_millis;
 | 
			
		||||
  }
 | 
			
		||||
  // Save the timestamp after validating the frame so, if invalid, we'll take the next frame immediately
 | 
			
		||||
  this->last_periodic_millis_ = App.get_loop_component_start_time();
 | 
			
		||||
 | 
			
		||||
  /*
 | 
			
		||||
    Data Type: 7th
 | 
			
		||||
    0x01: Engineering mode
 | 
			
		||||
    0x02: Normal mode
 | 
			
		||||
  */
 | 
			
		||||
  bool engineering_mode = buffer[DATA_TYPES] == 0x01;
 | 
			
		||||
  bool engineering_mode = this->buffer_data_[DATA_TYPES] == 0x01;
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
  if (this->engineering_mode_switch_ != nullptr &&
 | 
			
		||||
      current_millis - last_engineering_mode_change_millis_ > this->throttle_) {
 | 
			
		||||
  if (this->engineering_mode_switch_ != nullptr) {
 | 
			
		||||
    this->engineering_mode_switch_->publish_state(engineering_mode);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
@@ -335,7 +343,7 @@ void LD2410Component::handle_periodic_data_(uint8_t *buffer, int len) {
 | 
			
		||||
    0x02 = Still targets
 | 
			
		||||
    0x03 = Moving+Still targets
 | 
			
		||||
  */
 | 
			
		||||
  char target_state = buffer[TARGET_STATES];
 | 
			
		||||
  char target_state = this->buffer_data_[TARGET_STATES];
 | 
			
		||||
  if (this->target_binary_sensor_ != nullptr) {
 | 
			
		||||
    this->target_binary_sensor_->publish_state(target_state != 0x00);
 | 
			
		||||
  }
 | 
			
		||||
@@ -355,27 +363,30 @@ void LD2410Component::handle_periodic_data_(uint8_t *buffer, int len) {
 | 
			
		||||
  */
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
  if (this->moving_target_distance_sensor_ != nullptr) {
 | 
			
		||||
    int new_moving_target_distance = ld2410::two_byte_to_int(buffer[MOVING_TARGET_LOW], buffer[MOVING_TARGET_HIGH]);
 | 
			
		||||
    int new_moving_target_distance =
 | 
			
		||||
        ld2410::two_byte_to_int(this->buffer_data_[MOVING_TARGET_LOW], this->buffer_data_[MOVING_TARGET_HIGH]);
 | 
			
		||||
    if (this->moving_target_distance_sensor_->get_state() != new_moving_target_distance)
 | 
			
		||||
      this->moving_target_distance_sensor_->publish_state(new_moving_target_distance);
 | 
			
		||||
  }
 | 
			
		||||
  if (this->moving_target_energy_sensor_ != nullptr) {
 | 
			
		||||
    int new_moving_target_energy = buffer[MOVING_ENERGY];
 | 
			
		||||
    int new_moving_target_energy = this->buffer_data_[MOVING_ENERGY];
 | 
			
		||||
    if (this->moving_target_energy_sensor_->get_state() != new_moving_target_energy)
 | 
			
		||||
      this->moving_target_energy_sensor_->publish_state(new_moving_target_energy);
 | 
			
		||||
  }
 | 
			
		||||
  if (this->still_target_distance_sensor_ != nullptr) {
 | 
			
		||||
    int new_still_target_distance = ld2410::two_byte_to_int(buffer[STILL_TARGET_LOW], buffer[STILL_TARGET_HIGH]);
 | 
			
		||||
    int new_still_target_distance =
 | 
			
		||||
        ld2410::two_byte_to_int(this->buffer_data_[STILL_TARGET_LOW], this->buffer_data_[STILL_TARGET_HIGH]);
 | 
			
		||||
    if (this->still_target_distance_sensor_->get_state() != new_still_target_distance)
 | 
			
		||||
      this->still_target_distance_sensor_->publish_state(new_still_target_distance);
 | 
			
		||||
  }
 | 
			
		||||
  if (this->still_target_energy_sensor_ != nullptr) {
 | 
			
		||||
    int new_still_target_energy = buffer[STILL_ENERGY];
 | 
			
		||||
    int new_still_target_energy = this->buffer_data_[STILL_ENERGY];
 | 
			
		||||
    if (this->still_target_energy_sensor_->get_state() != new_still_target_energy)
 | 
			
		||||
      this->still_target_energy_sensor_->publish_state(new_still_target_energy);
 | 
			
		||||
  }
 | 
			
		||||
  if (this->detection_distance_sensor_ != nullptr) {
 | 
			
		||||
    int new_detect_distance = ld2410::two_byte_to_int(buffer[DETECT_DISTANCE_LOW], buffer[DETECT_DISTANCE_HIGH]);
 | 
			
		||||
    int new_detect_distance =
 | 
			
		||||
        ld2410::two_byte_to_int(this->buffer_data_[DETECT_DISTANCE_LOW], this->buffer_data_[DETECT_DISTANCE_HIGH]);
 | 
			
		||||
    if (this->detection_distance_sensor_->get_state() != new_detect_distance)
 | 
			
		||||
      this->detection_distance_sensor_->publish_state(new_detect_distance);
 | 
			
		||||
  }
 | 
			
		||||
@@ -383,12 +394,12 @@ void LD2410Component::handle_periodic_data_(uint8_t *buffer, int len) {
 | 
			
		||||
    /*
 | 
			
		||||
      Moving distance range: 18th byte
 | 
			
		||||
      Still distance range: 19th byte
 | 
			
		||||
      Moving enery: 20~28th bytes
 | 
			
		||||
      Moving energy: 20~28th bytes
 | 
			
		||||
    */
 | 
			
		||||
    for (std::vector<sensor::Sensor *>::size_type i = 0; i != this->gate_move_sensors_.size(); i++) {
 | 
			
		||||
      sensor::Sensor *s = this->gate_move_sensors_[i];
 | 
			
		||||
      if (s != nullptr) {
 | 
			
		||||
        s->publish_state(buffer[MOVING_SENSOR_START + i]);
 | 
			
		||||
        s->publish_state(this->buffer_data_[MOVING_SENSOR_START + i]);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    /*
 | 
			
		||||
@@ -397,16 +408,17 @@ void LD2410Component::handle_periodic_data_(uint8_t *buffer, int len) {
 | 
			
		||||
    for (std::vector<sensor::Sensor *>::size_type i = 0; i != this->gate_still_sensors_.size(); i++) {
 | 
			
		||||
      sensor::Sensor *s = this->gate_still_sensors_[i];
 | 
			
		||||
      if (s != nullptr) {
 | 
			
		||||
        s->publish_state(buffer[STILL_SENSOR_START + i]);
 | 
			
		||||
        s->publish_state(this->buffer_data_[STILL_SENSOR_START + i]);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    /*
 | 
			
		||||
      Light sensor: 38th bytes
 | 
			
		||||
    */
 | 
			
		||||
    if (this->light_sensor_ != nullptr) {
 | 
			
		||||
      int new_light_sensor = buffer[LIGHT_SENSOR];
 | 
			
		||||
      if (this->light_sensor_->get_state() != new_light_sensor)
 | 
			
		||||
      int new_light_sensor = this->buffer_data_[LIGHT_SENSOR];
 | 
			
		||||
      if (this->light_sensor_->get_state() != new_light_sensor) {
 | 
			
		||||
        this->light_sensor_->publish_state(new_light_sensor);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  } else {
 | 
			
		||||
    for (auto *s : this->gate_move_sensors_) {
 | 
			
		||||
@@ -427,7 +439,7 @@ void LD2410Component::handle_periodic_data_(uint8_t *buffer, int len) {
 | 
			
		||||
#ifdef USE_BINARY_SENSOR
 | 
			
		||||
  if (engineering_mode) {
 | 
			
		||||
    if (this->out_pin_presence_status_binary_sensor_ != nullptr) {
 | 
			
		||||
      this->out_pin_presence_status_binary_sensor_->publish_state(buffer[OUT_PIN_SENSOR] == 0x01);
 | 
			
		||||
      this->out_pin_presence_status_binary_sensor_->publish_state(this->buffer_data_[OUT_PIN_SENSOR] == 0x01);
 | 
			
		||||
    }
 | 
			
		||||
  } else {
 | 
			
		||||
    if (this->out_pin_presence_status_binary_sensor_ != nullptr) {
 | 
			
		||||
@@ -439,127 +451,149 @@ void LD2410Component::handle_periodic_data_(uint8_t *buffer, int len) {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
std::function<void(void)> set_number_value(number::Number *n, float value) {
 | 
			
		||||
  float normalized_value = value * 1.0;
 | 
			
		||||
  if (n != nullptr && (!n->has_state() || n->state != normalized_value)) {
 | 
			
		||||
    n->state = normalized_value;
 | 
			
		||||
    return [n, normalized_value]() { n->publish_state(normalized_value); };
 | 
			
		||||
  if (n != nullptr && (!n->has_state() || n->state != value)) {
 | 
			
		||||
    n->state = value;
 | 
			
		||||
    return [n, value]() { n->publish_state(value); };
 | 
			
		||||
  }
 | 
			
		||||
  return []() {};
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
bool LD2410Component::handle_ack_data_(uint8_t *buffer, int len) {
 | 
			
		||||
  ESP_LOGV(TAG, "Handling ACK DATA for COMMAND %02X", buffer[COMMAND]);
 | 
			
		||||
  if (len < 10) {
 | 
			
		||||
bool LD2410Component::handle_ack_data_() {
 | 
			
		||||
  ESP_LOGV(TAG, "Handling ACK DATA for COMMAND %02X", this->buffer_data_[COMMAND]);
 | 
			
		||||
  if (this->buffer_pos_ < 10) {
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid length");
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  if (buffer[0] != 0xFD || buffer[1] != 0xFC || buffer[2] != 0xFB || buffer[3] != 0xFA) {  // check 4 frame start bytes
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid header");
 | 
			
		||||
  if (!ld2410::validate_header_footer(CMD_FRAME_HEADER, this->buffer_data_)) {
 | 
			
		||||
    ESP_LOGW(TAG, "Invalid header: %s", format_hex_pretty(this->buffer_data_, HEADER_FOOTER_SIZE).c_str());
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  if (buffer[COMMAND_STATUS] != 0x01) {
 | 
			
		||||
  if (this->buffer_data_[COMMAND_STATUS] != 0x01) {
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid status");
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  if (ld2410::two_byte_to_int(buffer[8], buffer[9]) != 0x00) {
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid command: %u, %u", buffer[8], buffer[9]);
 | 
			
		||||
  if (this->buffer_data_[8] || this->buffer_data_[9]) {
 | 
			
		||||
    ESP_LOGW(TAG, "Invalid command: %02X, %02X", this->buffer_data_[8], this->buffer_data_[9]);
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  switch (buffer[COMMAND]) {
 | 
			
		||||
    case lowbyte(CMD_ENABLE_CONF):
 | 
			
		||||
  switch (this->buffer_data_[COMMAND]) {
 | 
			
		||||
    case CMD_ENABLE_CONF:
 | 
			
		||||
      ESP_LOGV(TAG, "Enable conf");
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_DISABLE_CONF):
 | 
			
		||||
 | 
			
		||||
    case CMD_DISABLE_CONF:
 | 
			
		||||
      ESP_LOGV(TAG, "Disabled conf");
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_SET_BAUD_RATE):
 | 
			
		||||
 | 
			
		||||
    case CMD_SET_BAUD_RATE:
 | 
			
		||||
      ESP_LOGV(TAG, "Baud rate change");
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
      if (this->baud_rate_select_ != nullptr) {
 | 
			
		||||
        ESP_LOGE(TAG, "Configure baud rate to %s and reinstall", this->baud_rate_select_->state.c_str());
 | 
			
		||||
        ESP_LOGE(TAG, "Change baud rate to %s and reinstall", this->baud_rate_select_->state.c_str());
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_VERSION):
 | 
			
		||||
      this->version_ = str_sprintf(VERSION_FMT, buffer[13], buffer[12], buffer[17], buffer[16], buffer[15], buffer[14]);
 | 
			
		||||
      ESP_LOGV(TAG, "Firmware version: %s", this->version_.c_str());
 | 
			
		||||
 | 
			
		||||
    case CMD_QUERY_VERSION: {
 | 
			
		||||
      std::memcpy(this->version_, &this->buffer_data_[12], sizeof(this->version_));
 | 
			
		||||
      std::string version = str_sprintf(VERSION_FMT, this->version_[1], this->version_[0], this->version_[5],
 | 
			
		||||
                                        this->version_[4], this->version_[3], this->version_[2]);
 | 
			
		||||
      ESP_LOGV(TAG, "Firmware version: %s", version.c_str());
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
      if (this->version_text_sensor_ != nullptr) {
 | 
			
		||||
        this->version_text_sensor_->publish_state(this->version_);
 | 
			
		||||
        this->version_text_sensor_->publish_state(version);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_QUERY_DISTANCE_RESOLUTION): {
 | 
			
		||||
      std::string distance_resolution =
 | 
			
		||||
          find_str(DISTANCE_RESOLUTIONS_BY_UINT, ld2410::two_byte_to_int(buffer[10], buffer[11]));
 | 
			
		||||
      ESP_LOGV(TAG, "Distance resolution: %s", distance_resolution.c_str());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    case CMD_QUERY_DISTANCE_RESOLUTION: {
 | 
			
		||||
      const auto *distance_resolution = find_str(DISTANCE_RESOLUTIONS_BY_UINT, this->buffer_data_[10]);
 | 
			
		||||
      ESP_LOGV(TAG, "Distance resolution: %s", distance_resolution);
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
      if (this->distance_resolution_select_ != nullptr &&
 | 
			
		||||
          this->distance_resolution_select_->state != distance_resolution) {
 | 
			
		||||
      if (this->distance_resolution_select_ != nullptr) {
 | 
			
		||||
        this->distance_resolution_select_->publish_state(distance_resolution);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
    } break;
 | 
			
		||||
    case lowbyte(CMD_QUERY_LIGHT_CONTROL): {
 | 
			
		||||
      this->light_function_ = find_str(LIGHT_FUNCTIONS_BY_UINT, buffer[10]);
 | 
			
		||||
      this->light_threshold_ = buffer[11] * 1.0;
 | 
			
		||||
      this->out_pin_level_ = find_str(OUT_PIN_LEVELS_BY_UINT, buffer[12]);
 | 
			
		||||
      ESP_LOGV(TAG, "Light function: %s", const_cast<char *>(this->light_function_.c_str()));
 | 
			
		||||
      ESP_LOGV(TAG, "Light threshold: %f", this->light_threshold_);
 | 
			
		||||
      ESP_LOGV(TAG, "Out pin level: %s", const_cast<char *>(this->out_pin_level_.c_str()));
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    case CMD_QUERY_LIGHT_CONTROL: {
 | 
			
		||||
      this->light_function_ = this->buffer_data_[10];
 | 
			
		||||
      this->light_threshold_ = this->buffer_data_[11];
 | 
			
		||||
      this->out_pin_level_ = this->buffer_data_[12];
 | 
			
		||||
      const auto *light_function_str = find_str(LIGHT_FUNCTIONS_BY_UINT, this->light_function_);
 | 
			
		||||
      const auto *out_pin_level_str = find_str(OUT_PIN_LEVELS_BY_UINT, this->out_pin_level_);
 | 
			
		||||
      ESP_LOGV(TAG,
 | 
			
		||||
               "Light function: %s\n"
 | 
			
		||||
               "Light threshold: %u\n"
 | 
			
		||||
               "Out pin level: %s",
 | 
			
		||||
               light_function_str, this->light_threshold_, out_pin_level_str);
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
      if (this->light_function_select_ != nullptr && this->light_function_select_->state != this->light_function_) {
 | 
			
		||||
        this->light_function_select_->publish_state(this->light_function_);
 | 
			
		||||
      if (this->light_function_select_ != nullptr) {
 | 
			
		||||
        this->light_function_select_->publish_state(light_function_str);
 | 
			
		||||
      }
 | 
			
		||||
      if (this->out_pin_level_select_ != nullptr && this->out_pin_level_select_->state != this->out_pin_level_) {
 | 
			
		||||
        this->out_pin_level_select_->publish_state(this->out_pin_level_);
 | 
			
		||||
      if (this->out_pin_level_select_ != nullptr) {
 | 
			
		||||
        this->out_pin_level_select_->publish_state(out_pin_level_str);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
      if (this->light_threshold_number_ != nullptr &&
 | 
			
		||||
          (!this->light_threshold_number_->has_state() ||
 | 
			
		||||
           this->light_threshold_number_->state != this->light_threshold_)) {
 | 
			
		||||
        this->light_threshold_number_->publish_state(this->light_threshold_);
 | 
			
		||||
      if (this->light_threshold_number_ != nullptr) {
 | 
			
		||||
        this->light_threshold_number_->publish_state(static_cast<float>(this->light_threshold_));
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
    } break;
 | 
			
		||||
    case lowbyte(CMD_MAC):
 | 
			
		||||
      if (len < 20) {
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
    case CMD_QUERY_MAC_ADDRESS: {
 | 
			
		||||
      if (this->buffer_pos_ < 20) {
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
      this->mac_ = format_mac_address_pretty(&buffer[10]);
 | 
			
		||||
      ESP_LOGV(TAG, "MAC address: %s", this->mac_.c_str());
 | 
			
		||||
 | 
			
		||||
      this->bluetooth_on_ = std::memcmp(&this->buffer_data_[10], NO_MAC, sizeof(NO_MAC)) != 0;
 | 
			
		||||
      if (this->bluetooth_on_) {
 | 
			
		||||
        std::memcpy(this->mac_address_, &this->buffer_data_[10], sizeof(this->mac_address_));
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      std::string mac_str =
 | 
			
		||||
          mac_address_is_valid(this->mac_address_) ? format_mac_address_pretty(this->mac_address_) : UNKNOWN_MAC;
 | 
			
		||||
      ESP_LOGV(TAG, "MAC address: %s", mac_str.c_str());
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
      if (this->mac_text_sensor_ != nullptr) {
 | 
			
		||||
        this->mac_text_sensor_->publish_state(this->mac_ == NO_MAC ? UNKNOWN_MAC : this->mac_);
 | 
			
		||||
        this->mac_text_sensor_->publish_state(mac_str);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
      if (this->bluetooth_switch_ != nullptr) {
 | 
			
		||||
        this->bluetooth_switch_->publish_state(this->mac_ != NO_MAC);
 | 
			
		||||
        this->bluetooth_switch_->publish_state(this->bluetooth_on_);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_GATE_SENS):
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    case CMD_GATE_SENS:
 | 
			
		||||
      ESP_LOGV(TAG, "Sensitivity");
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_BLUETOOTH):
 | 
			
		||||
 | 
			
		||||
    case CMD_BLUETOOTH:
 | 
			
		||||
      ESP_LOGV(TAG, "Bluetooth");
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_SET_DISTANCE_RESOLUTION):
 | 
			
		||||
 | 
			
		||||
    case CMD_SET_DISTANCE_RESOLUTION:
 | 
			
		||||
      ESP_LOGV(TAG, "Set distance resolution");
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_SET_LIGHT_CONTROL):
 | 
			
		||||
 | 
			
		||||
    case CMD_SET_LIGHT_CONTROL:
 | 
			
		||||
      ESP_LOGV(TAG, "Set light control");
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_BT_PASSWORD):
 | 
			
		||||
 | 
			
		||||
    case CMD_BT_PASSWORD:
 | 
			
		||||
      ESP_LOGV(TAG, "Set bluetooth password");
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_QUERY):  // Query parameters response
 | 
			
		||||
    {
 | 
			
		||||
      if (buffer[10] != 0xAA)
 | 
			
		||||
 | 
			
		||||
    case CMD_QUERY: {  // Query parameters response
 | 
			
		||||
      if (this->buffer_data_[10] != HEADER)
 | 
			
		||||
        return true;  // value head=0xAA
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
      /*
 | 
			
		||||
@@ -567,29 +601,31 @@ bool LD2410Component::handle_ack_data_(uint8_t *buffer, int len) {
 | 
			
		||||
        Still distance range: 14th byte
 | 
			
		||||
      */
 | 
			
		||||
      std::vector<std::function<void(void)>> updates;
 | 
			
		||||
      updates.push_back(set_number_value(this->max_move_distance_gate_number_, buffer[12]));
 | 
			
		||||
      updates.push_back(set_number_value(this->max_still_distance_gate_number_, buffer[13]));
 | 
			
		||||
      updates.push_back(set_number_value(this->max_move_distance_gate_number_, this->buffer_data_[12]));
 | 
			
		||||
      updates.push_back(set_number_value(this->max_still_distance_gate_number_, this->buffer_data_[13]));
 | 
			
		||||
      /*
 | 
			
		||||
        Moving Sensitivities: 15~23th bytes
 | 
			
		||||
      */
 | 
			
		||||
      for (std::vector<number::Number *>::size_type i = 0; i != this->gate_move_threshold_numbers_.size(); i++) {
 | 
			
		||||
        updates.push_back(set_number_value(this->gate_move_threshold_numbers_[i], buffer[14 + i]));
 | 
			
		||||
        updates.push_back(set_number_value(this->gate_move_threshold_numbers_[i], this->buffer_data_[14 + i]));
 | 
			
		||||
      }
 | 
			
		||||
      /*
 | 
			
		||||
        Still Sensitivities: 24~32th bytes
 | 
			
		||||
      */
 | 
			
		||||
      for (std::vector<number::Number *>::size_type i = 0; i != this->gate_still_threshold_numbers_.size(); i++) {
 | 
			
		||||
        updates.push_back(set_number_value(this->gate_still_threshold_numbers_[i], buffer[23 + i]));
 | 
			
		||||
        updates.push_back(set_number_value(this->gate_still_threshold_numbers_[i], this->buffer_data_[23 + i]));
 | 
			
		||||
      }
 | 
			
		||||
      /*
 | 
			
		||||
        None Duration: 33~34th bytes
 | 
			
		||||
      */
 | 
			
		||||
      updates.push_back(set_number_value(this->timeout_number_, ld2410::two_byte_to_int(buffer[32], buffer[33])));
 | 
			
		||||
      updates.push_back(set_number_value(this->timeout_number_,
 | 
			
		||||
                                         ld2410::two_byte_to_int(this->buffer_data_[32], this->buffer_data_[33])));
 | 
			
		||||
      for (auto &update : updates) {
 | 
			
		||||
        update();
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
    } break;
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
    default:
 | 
			
		||||
      break;
 | 
			
		||||
  }
 | 
			
		||||
@@ -597,59 +633,60 @@ bool LD2410Component::handle_ack_data_(uint8_t *buffer, int len) {
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::readline_(int readch, uint8_t *buffer, int len) {
 | 
			
		||||
  static int pos = 0;
 | 
			
		||||
void LD2410Component::readline_(int readch) {
 | 
			
		||||
  if (readch < 0) {
 | 
			
		||||
    return;  // No data available
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (readch >= 0) {
 | 
			
		||||
    if (pos < len - 1) {
 | 
			
		||||
      buffer[pos++] = readch;
 | 
			
		||||
      buffer[pos] = 0;
 | 
			
		||||
  if (this->buffer_pos_ < MAX_LINE_LENGTH - 1) {
 | 
			
		||||
    this->buffer_data_[this->buffer_pos_++] = readch;
 | 
			
		||||
    this->buffer_data_[this->buffer_pos_] = 0;
 | 
			
		||||
  } else {
 | 
			
		||||
    // We should never get here, but just in case...
 | 
			
		||||
    ESP_LOGW(TAG, "Max command length exceeded; ignoring");
 | 
			
		||||
    this->buffer_pos_ = 0;
 | 
			
		||||
  }
 | 
			
		||||
  if (this->buffer_pos_ < 4) {
 | 
			
		||||
    return;  // Not enough data to process yet
 | 
			
		||||
  }
 | 
			
		||||
  if (ld2410::validate_header_footer(DATA_FRAME_FOOTER, &this->buffer_data_[this->buffer_pos_ - 4])) {
 | 
			
		||||
    ESP_LOGV(TAG, "Handling Periodic Data: %s", format_hex_pretty(this->buffer_data_, this->buffer_pos_).c_str());
 | 
			
		||||
    this->handle_periodic_data_();
 | 
			
		||||
    this->buffer_pos_ = 0;  // Reset position index for next message
 | 
			
		||||
  } else if (ld2410::validate_header_footer(CMD_FRAME_FOOTER, &this->buffer_data_[this->buffer_pos_ - 4])) {
 | 
			
		||||
    ESP_LOGV(TAG, "Handling Ack Data: %s", format_hex_pretty(this->buffer_data_, this->buffer_pos_).c_str());
 | 
			
		||||
    if (this->handle_ack_data_()) {
 | 
			
		||||
      this->buffer_pos_ = 0;  // Reset position index for next message
 | 
			
		||||
    } else {
 | 
			
		||||
      pos = 0;
 | 
			
		||||
    }
 | 
			
		||||
    if (pos >= 4) {
 | 
			
		||||
      if (buffer[pos - 4] == 0xF8 && buffer[pos - 3] == 0xF7 && buffer[pos - 2] == 0xF6 && buffer[pos - 1] == 0xF5) {
 | 
			
		||||
        ESP_LOGV(TAG, "Will handle Periodic Data");
 | 
			
		||||
        this->handle_periodic_data_(buffer, pos);
 | 
			
		||||
        pos = 0;  // Reset position index ready for next time
 | 
			
		||||
      } else if (buffer[pos - 4] == 0x04 && buffer[pos - 3] == 0x03 && buffer[pos - 2] == 0x02 &&
 | 
			
		||||
                 buffer[pos - 1] == 0x01) {
 | 
			
		||||
        ESP_LOGV(TAG, "Will handle ACK Data");
 | 
			
		||||
        if (this->handle_ack_data_(buffer, pos)) {
 | 
			
		||||
          pos = 0;  // Reset position index ready for next time
 | 
			
		||||
        } else {
 | 
			
		||||
          ESP_LOGV(TAG, "ACK Data incomplete");
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      ESP_LOGV(TAG, "Ack Data incomplete");
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::set_config_mode_(bool enable) {
 | 
			
		||||
  uint8_t cmd = enable ? CMD_ENABLE_CONF : CMD_DISABLE_CONF;
 | 
			
		||||
  uint8_t cmd_value[2] = {0x01, 0x00};
 | 
			
		||||
  this->send_command_(cmd, enable ? cmd_value : nullptr, 2);
 | 
			
		||||
  const uint8_t cmd = enable ? CMD_ENABLE_CONF : CMD_DISABLE_CONF;
 | 
			
		||||
  const uint8_t cmd_value[2] = {0x01, 0x00};
 | 
			
		||||
  this->send_command_(cmd, enable ? cmd_value : nullptr, sizeof(cmd_value));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::set_bluetooth(bool enable) {
 | 
			
		||||
  this->set_config_mode_(true);
 | 
			
		||||
  uint8_t enable_cmd_value[2] = {0x01, 0x00};
 | 
			
		||||
  uint8_t disable_cmd_value[2] = {0x00, 0x00};
 | 
			
		||||
  this->send_command_(CMD_BLUETOOTH, enable ? enable_cmd_value : disable_cmd_value, 2);
 | 
			
		||||
  const uint8_t cmd_value[2] = {enable ? (uint8_t) 0x01 : (uint8_t) 0x00, 0x00};
 | 
			
		||||
  this->send_command_(CMD_BLUETOOTH, cmd_value, sizeof(cmd_value));
 | 
			
		||||
  this->set_timeout(200, [this]() { this->restart_and_read_all_info(); });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::set_distance_resolution(const std::string &state) {
 | 
			
		||||
  this->set_config_mode_(true);
 | 
			
		||||
  uint8_t cmd_value[2] = {find_uint8(DISTANCE_RESOLUTIONS_BY_STR, state), 0x00};
 | 
			
		||||
  this->send_command_(CMD_SET_DISTANCE_RESOLUTION, cmd_value, 2);
 | 
			
		||||
  const uint8_t cmd_value[2] = {find_uint8(DISTANCE_RESOLUTIONS_BY_STR, state), 0x00};
 | 
			
		||||
  this->send_command_(CMD_SET_DISTANCE_RESOLUTION, cmd_value, sizeof(cmd_value));
 | 
			
		||||
  this->set_timeout(200, [this]() { this->restart_and_read_all_info(); });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::set_baud_rate(const std::string &state) {
 | 
			
		||||
  this->set_config_mode_(true);
 | 
			
		||||
  uint8_t cmd_value[2] = {find_uint8(BAUD_RATES_BY_STR, state), 0x00};
 | 
			
		||||
  this->send_command_(CMD_SET_BAUD_RATE, cmd_value, 2);
 | 
			
		||||
  const uint8_t cmd_value[2] = {find_uint8(BAUD_RATES_BY_STR, state), 0x00};
 | 
			
		||||
  this->send_command_(CMD_SET_BAUD_RATE, cmd_value, sizeof(cmd_value));
 | 
			
		||||
  this->set_timeout(200, [this]() { this->restart_(); });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -661,14 +698,13 @@ void LD2410Component::set_bluetooth_password(const std::string &password) {
 | 
			
		||||
  this->set_config_mode_(true);
 | 
			
		||||
  uint8_t cmd_value[6];
 | 
			
		||||
  std::copy(password.begin(), password.end(), std::begin(cmd_value));
 | 
			
		||||
  this->send_command_(CMD_BT_PASSWORD, cmd_value, 6);
 | 
			
		||||
  this->send_command_(CMD_BT_PASSWORD, cmd_value, sizeof(cmd_value));
 | 
			
		||||
  this->set_config_mode_(false);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::set_engineering_mode(bool enable) {
 | 
			
		||||
  const uint8_t cmd = enable ? CMD_ENABLE_ENG : CMD_DISABLE_ENG;
 | 
			
		||||
  this->set_config_mode_(true);
 | 
			
		||||
  last_engineering_mode_change_millis_ = App.get_loop_component_start_time();
 | 
			
		||||
  uint8_t cmd = enable ? CMD_ENABLE_ENG : CMD_DISABLE_ENG;
 | 
			
		||||
  this->send_command_(cmd, nullptr, 0);
 | 
			
		||||
  this->set_config_mode_(false);
 | 
			
		||||
}
 | 
			
		||||
@@ -682,14 +718,17 @@ void LD2410Component::factory_reset() {
 | 
			
		||||
void LD2410Component::restart_() { this->send_command_(CMD_RESTART, nullptr, 0); }
 | 
			
		||||
 | 
			
		||||
void LD2410Component::query_parameters_() { this->send_command_(CMD_QUERY, nullptr, 0); }
 | 
			
		||||
void LD2410Component::get_version_() { this->send_command_(CMD_VERSION, nullptr, 0); }
 | 
			
		||||
 | 
			
		||||
void LD2410Component::get_version_() { this->send_command_(CMD_QUERY_VERSION, nullptr, 0); }
 | 
			
		||||
 | 
			
		||||
void LD2410Component::get_mac_() {
 | 
			
		||||
  uint8_t cmd_value[2] = {0x01, 0x00};
 | 
			
		||||
  this->send_command_(CMD_MAC, cmd_value, 2);
 | 
			
		||||
  const uint8_t cmd_value[2] = {0x01, 0x00};
 | 
			
		||||
  this->send_command_(CMD_QUERY_MAC_ADDRESS, cmd_value, sizeof(cmd_value));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::get_distance_resolution_() { this->send_command_(CMD_QUERY_DISTANCE_RESOLUTION, nullptr, 0); }
 | 
			
		||||
 | 
			
		||||
void LD2410Component::get_light_control_() { this->send_command_(CMD_QUERY_LIGHT_CONTROL, nullptr, 0); }
 | 
			
		||||
void LD2410Component::query_light_control_() { this->send_command_(CMD_QUERY_LIGHT_CONTROL, nullptr, 0); }
 | 
			
		||||
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
void LD2410Component::set_max_distances_timeout() {
 | 
			
		||||
@@ -719,8 +758,7 @@ void LD2410Component::set_max_distances_timeout() {
 | 
			
		||||
                       0x00,
 | 
			
		||||
                       0x00};
 | 
			
		||||
  this->set_config_mode_(true);
 | 
			
		||||
  this->send_command_(CMD_MAXDIST_DURATION, value, 18);
 | 
			
		||||
  delay(50);  // NOLINT
 | 
			
		||||
  this->send_command_(CMD_MAXDIST_DURATION, value, sizeof(value));
 | 
			
		||||
  this->query_parameters_();
 | 
			
		||||
  this->set_timeout(200, [this]() { this->restart_and_read_all_info(); });
 | 
			
		||||
  this->set_config_mode_(false);
 | 
			
		||||
@@ -749,17 +787,16 @@ void LD2410Component::set_gate_threshold(uint8_t gate) {
 | 
			
		||||
  uint8_t value[18] = {0x00, 0x00, lowbyte(gate),   highbyte(gate),   0x00, 0x00,
 | 
			
		||||
                       0x01, 0x00, lowbyte(motion), highbyte(motion), 0x00, 0x00,
 | 
			
		||||
                       0x02, 0x00, lowbyte(still),  highbyte(still),  0x00, 0x00};
 | 
			
		||||
  this->send_command_(CMD_GATE_SENS, value, 18);
 | 
			
		||||
  delay(50);  // NOLINT
 | 
			
		||||
  this->send_command_(CMD_GATE_SENS, value, sizeof(value));
 | 
			
		||||
  this->query_parameters_();
 | 
			
		||||
  this->set_config_mode_(false);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::set_gate_still_threshold_number(int gate, number::Number *n) {
 | 
			
		||||
void LD2410Component::set_gate_still_threshold_number(uint8_t gate, number::Number *n) {
 | 
			
		||||
  this->gate_still_threshold_numbers_[gate] = n;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2410Component::set_gate_move_threshold_number(int gate, number::Number *n) {
 | 
			
		||||
void LD2410Component::set_gate_move_threshold_number(uint8_t gate, number::Number *n) {
 | 
			
		||||
  this->gate_move_threshold_numbers_[gate] = n;
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
@@ -767,35 +804,28 @@ void LD2410Component::set_gate_move_threshold_number(int gate, number::Number *n
 | 
			
		||||
void LD2410Component::set_light_out_control() {
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
  if (this->light_threshold_number_ != nullptr && this->light_threshold_number_->has_state()) {
 | 
			
		||||
    this->light_threshold_ = this->light_threshold_number_->state;
 | 
			
		||||
    this->light_threshold_ = static_cast<uint8_t>(this->light_threshold_number_->state);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
  if (this->light_function_select_ != nullptr && this->light_function_select_->has_state()) {
 | 
			
		||||
    this->light_function_ = this->light_function_select_->state;
 | 
			
		||||
    this->light_function_ = find_uint8(LIGHT_FUNCTIONS_BY_STR, this->light_function_select_->state);
 | 
			
		||||
  }
 | 
			
		||||
  if (this->out_pin_level_select_ != nullptr && this->out_pin_level_select_->has_state()) {
 | 
			
		||||
    this->out_pin_level_ = this->out_pin_level_select_->state;
 | 
			
		||||
    this->out_pin_level_ = find_uint8(OUT_PIN_LEVELS_BY_STR, this->out_pin_level_select_->state);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
  if (this->light_function_.empty() || this->out_pin_level_.empty() || this->light_threshold_ < 0) {
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  this->set_config_mode_(true);
 | 
			
		||||
  uint8_t light_function = find_uint8(LIGHT_FUNCTIONS_BY_STR, this->light_function_);
 | 
			
		||||
  uint8_t light_threshold = static_cast<uint8_t>(this->light_threshold_);
 | 
			
		||||
  uint8_t out_pin_level = find_uint8(OUT_PIN_LEVELS_BY_STR, this->out_pin_level_);
 | 
			
		||||
  uint8_t value[4] = {light_function, light_threshold, out_pin_level, 0x00};
 | 
			
		||||
  this->send_command_(CMD_SET_LIGHT_CONTROL, value, 4);
 | 
			
		||||
  delay(50);  // NOLINT
 | 
			
		||||
  this->get_light_control_();
 | 
			
		||||
  uint8_t value[4] = {this->light_function_, this->light_threshold_, this->out_pin_level_, 0x00};
 | 
			
		||||
  this->send_command_(CMD_SET_LIGHT_CONTROL, value, sizeof(value));
 | 
			
		||||
  this->query_light_control_();
 | 
			
		||||
  this->set_timeout(200, [this]() { this->restart_and_read_all_info(); });
 | 
			
		||||
  this->set_config_mode_(false);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
void LD2410Component::set_gate_move_sensor(int gate, sensor::Sensor *s) { this->gate_move_sensors_[gate] = s; }
 | 
			
		||||
void LD2410Component::set_gate_still_sensor(int gate, sensor::Sensor *s) { this->gate_still_sensors_[gate] = s; }
 | 
			
		||||
void LD2410Component::set_gate_move_sensor(uint8_t gate, sensor::Sensor *s) { this->gate_move_sensors_[gate] = s; }
 | 
			
		||||
void LD2410Component::set_gate_still_sensor(uint8_t gate, sensor::Sensor *s) { this->gate_still_sensors_[gate] = s; }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
}  // namespace ld2410
 | 
			
		||||
 
 | 
			
		||||
@@ -29,45 +29,48 @@
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2410 {
 | 
			
		||||
 | 
			
		||||
static const uint8_t MAX_LINE_LENGTH = 46;  // Max characters for serial buffer
 | 
			
		||||
static const uint8_t TOTAL_GATES = 9;       // Total number of gates supported by the LD2410
 | 
			
		||||
 | 
			
		||||
class LD2410Component : public Component, public uart::UARTDevice {
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
  SUB_SENSOR(moving_target_distance)
 | 
			
		||||
  SUB_SENSOR(still_target_distance)
 | 
			
		||||
  SUB_SENSOR(moving_target_energy)
 | 
			
		||||
  SUB_SENSOR(still_target_energy)
 | 
			
		||||
  SUB_SENSOR(light)
 | 
			
		||||
  SUB_SENSOR(detection_distance)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_BINARY_SENSOR
 | 
			
		||||
  SUB_BINARY_SENSOR(target)
 | 
			
		||||
  SUB_BINARY_SENSOR(out_pin_presence_status)
 | 
			
		||||
  SUB_BINARY_SENSOR(moving_target)
 | 
			
		||||
  SUB_BINARY_SENSOR(still_target)
 | 
			
		||||
  SUB_BINARY_SENSOR(out_pin_presence_status)
 | 
			
		||||
  SUB_BINARY_SENSOR(target)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
  SUB_SENSOR(light)
 | 
			
		||||
  SUB_SENSOR(detection_distance)
 | 
			
		||||
  SUB_SENSOR(moving_target_distance)
 | 
			
		||||
  SUB_SENSOR(moving_target_energy)
 | 
			
		||||
  SUB_SENSOR(still_target_distance)
 | 
			
		||||
  SUB_SENSOR(still_target_energy)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
  SUB_TEXT_SENSOR(version)
 | 
			
		||||
  SUB_TEXT_SENSOR(mac)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
  SUB_NUMBER(light_threshold)
 | 
			
		||||
  SUB_NUMBER(max_move_distance_gate)
 | 
			
		||||
  SUB_NUMBER(max_still_distance_gate)
 | 
			
		||||
  SUB_NUMBER(timeout)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
  SUB_SELECT(distance_resolution)
 | 
			
		||||
  SUB_SELECT(baud_rate)
 | 
			
		||||
  SUB_SELECT(distance_resolution)
 | 
			
		||||
  SUB_SELECT(light_function)
 | 
			
		||||
  SUB_SELECT(out_pin_level)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
  SUB_SWITCH(engineering_mode)
 | 
			
		||||
  SUB_SWITCH(bluetooth)
 | 
			
		||||
  SUB_SWITCH(engineering_mode)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_BUTTON
 | 
			
		||||
  SUB_BUTTON(reset)
 | 
			
		||||
  SUB_BUTTON(restart)
 | 
			
		||||
  SUB_BUTTON(factory_reset)
 | 
			
		||||
  SUB_BUTTON(query)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
  SUB_NUMBER(max_still_distance_gate)
 | 
			
		||||
  SUB_NUMBER(max_move_distance_gate)
 | 
			
		||||
  SUB_NUMBER(timeout)
 | 
			
		||||
  SUB_NUMBER(light_threshold)
 | 
			
		||||
  SUB_BUTTON(restart)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
 public:
 | 
			
		||||
@@ -76,14 +79,14 @@ class LD2410Component : public Component, public uart::UARTDevice {
 | 
			
		||||
  void loop() override;
 | 
			
		||||
  void set_light_out_control();
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
  void set_gate_still_threshold_number(int gate, number::Number *n);
 | 
			
		||||
  void set_gate_move_threshold_number(int gate, number::Number *n);
 | 
			
		||||
  void set_gate_still_threshold_number(uint8_t gate, number::Number *n);
 | 
			
		||||
  void set_gate_move_threshold_number(uint8_t gate, number::Number *n);
 | 
			
		||||
  void set_max_distances_timeout();
 | 
			
		||||
  void set_gate_threshold(uint8_t gate);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
  void set_gate_move_sensor(int gate, sensor::Sensor *s);
 | 
			
		||||
  void set_gate_still_sensor(int gate, sensor::Sensor *s);
 | 
			
		||||
  void set_gate_move_sensor(uint8_t gate, sensor::Sensor *s);
 | 
			
		||||
  void set_gate_still_sensor(uint8_t gate, sensor::Sensor *s);
 | 
			
		||||
#endif
 | 
			
		||||
  void set_throttle(uint16_t value) { this->throttle_ = value; };
 | 
			
		||||
  void set_bluetooth_password(const std::string &password);
 | 
			
		||||
@@ -96,33 +99,35 @@ class LD2410Component : public Component, public uart::UARTDevice {
 | 
			
		||||
  void factory_reset();
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  void send_command_(uint8_t command_str, const uint8_t *command_value, int command_value_len);
 | 
			
		||||
  void send_command_(uint8_t command_str, const uint8_t *command_value, uint8_t command_value_len);
 | 
			
		||||
  void set_config_mode_(bool enable);
 | 
			
		||||
  void handle_periodic_data_(uint8_t *buffer, int len);
 | 
			
		||||
  bool handle_ack_data_(uint8_t *buffer, int len);
 | 
			
		||||
  void readline_(int readch, uint8_t *buffer, int len);
 | 
			
		||||
  void handle_periodic_data_();
 | 
			
		||||
  bool handle_ack_data_();
 | 
			
		||||
  void readline_(int readch);
 | 
			
		||||
  void query_parameters_();
 | 
			
		||||
  void get_version_();
 | 
			
		||||
  void get_mac_();
 | 
			
		||||
  void get_distance_resolution_();
 | 
			
		||||
  void get_light_control_();
 | 
			
		||||
  void query_light_control_();
 | 
			
		||||
  void restart_();
 | 
			
		||||
 | 
			
		||||
  int32_t last_periodic_millis_ = 0;
 | 
			
		||||
  int32_t last_engineering_mode_change_millis_ = 0;
 | 
			
		||||
  uint16_t throttle_;
 | 
			
		||||
  float light_threshold_ = -1;
 | 
			
		||||
  std::string version_;
 | 
			
		||||
  std::string mac_;
 | 
			
		||||
  std::string out_pin_level_;
 | 
			
		||||
  std::string light_function_;
 | 
			
		||||
  uint32_t last_periodic_millis_ = 0;
 | 
			
		||||
  uint16_t throttle_ = 0;
 | 
			
		||||
  uint8_t light_function_ = 0;
 | 
			
		||||
  uint8_t light_threshold_ = 0;
 | 
			
		||||
  uint8_t out_pin_level_ = 0;
 | 
			
		||||
  uint8_t buffer_pos_ = 0;  // where to resume processing/populating buffer
 | 
			
		||||
  uint8_t buffer_data_[MAX_LINE_LENGTH];
 | 
			
		||||
  uint8_t mac_address_[6] = {0, 0, 0, 0, 0, 0};
 | 
			
		||||
  uint8_t version_[6] = {0, 0, 0, 0, 0, 0};
 | 
			
		||||
  bool bluetooth_on_{false};
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
  std::vector<number::Number *> gate_still_threshold_numbers_ = std::vector<number::Number *>(9);
 | 
			
		||||
  std::vector<number::Number *> gate_move_threshold_numbers_ = std::vector<number::Number *>(9);
 | 
			
		||||
  std::vector<number::Number *> gate_move_threshold_numbers_ = std::vector<number::Number *>(TOTAL_GATES);
 | 
			
		||||
  std::vector<number::Number *> gate_still_threshold_numbers_ = std::vector<number::Number *>(TOTAL_GATES);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
  std::vector<sensor::Sensor *> gate_still_sensors_ = std::vector<sensor::Sensor *>(9);
 | 
			
		||||
  std::vector<sensor::Sensor *> gate_move_sensors_ = std::vector<sensor::Sensor *>(9);
 | 
			
		||||
  std::vector<sensor::Sensor *> gate_move_sensors_ = std::vector<sensor::Sensor *>(TOTAL_GATES);
 | 
			
		||||
  std::vector<sensor::Sensor *> gate_still_sensors_ = std::vector<sensor::Sensor *>(TOTAL_GATES);
 | 
			
		||||
#endif
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -5,10 +5,10 @@
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2420 {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "LD2420.binary_sensor";
 | 
			
		||||
static const char *const TAG = "ld2420.binary_sensor";
 | 
			
		||||
 | 
			
		||||
void LD2420BinarySensor::dump_config() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "LD2420 BinarySensor:");
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Binary Sensor:");
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "Presence", this->presence_bsensor_);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -2,7 +2,7 @@
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "LD2420.button";
 | 
			
		||||
static const char *const TAG = "ld2420.button";
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2420 {
 | 
			
		||||
 
 | 
			
		||||
@@ -137,7 +137,7 @@ static const std::string OP_SIMPLE_MODE_STRING = "Simple";
 | 
			
		||||
// Memory-efficient lookup tables
 | 
			
		||||
struct StringToUint8 {
 | 
			
		||||
  const char *str;
 | 
			
		||||
  uint8_t value;
 | 
			
		||||
  const uint8_t value;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static constexpr StringToUint8 OP_MODE_BY_STR[] = {
 | 
			
		||||
@@ -155,8 +155,9 @@ static constexpr const char *ERR_MESSAGE[] = {
 | 
			
		||||
// Helper function for lookups
 | 
			
		||||
template<size_t N> uint8_t find_uint8(const StringToUint8 (&arr)[N], const std::string &str) {
 | 
			
		||||
  for (const auto &entry : arr) {
 | 
			
		||||
    if (str == entry.str)
 | 
			
		||||
    if (str == entry.str) {
 | 
			
		||||
      return entry.value;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  return 0xFF;  // Not found
 | 
			
		||||
}
 | 
			
		||||
@@ -326,15 +327,8 @@ void LD2420Component::revert_config_action() {
 | 
			
		||||
 | 
			
		||||
void LD2420Component::loop() {
 | 
			
		||||
  // If there is a active send command do not process it here, the send command call will handle it.
 | 
			
		||||
  if (!this->get_cmd_active_()) {
 | 
			
		||||
    if (!this->available())
 | 
			
		||||
      return;
 | 
			
		||||
    static uint8_t buffer[2048];
 | 
			
		||||
    static uint8_t rx_data;
 | 
			
		||||
    while (this->available()) {
 | 
			
		||||
      rx_data = this->read();
 | 
			
		||||
      this->readline_(rx_data, buffer, sizeof(buffer));
 | 
			
		||||
    }
 | 
			
		||||
  while (!this->cmd_active_ && this->available()) {
 | 
			
		||||
    this->readline_(this->read(), this->buffer_data_, MAX_LINE_LENGTH);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -365,8 +359,9 @@ void LD2420Component::auto_calibrate_sensitivity() {
 | 
			
		||||
 | 
			
		||||
    // Store average and peak values
 | 
			
		||||
    this->gate_avg[gate] = sum / CALIBRATE_SAMPLES;
 | 
			
		||||
    if (this->gate_peak[gate] < peak)
 | 
			
		||||
    if (this->gate_peak[gate] < peak) {
 | 
			
		||||
      this->gate_peak[gate] = peak;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    uint32_t calculated_value =
 | 
			
		||||
        (static_cast<uint32_t>(this->gate_peak[gate]) + (move_factor * static_cast<uint32_t>(this->gate_peak[gate])));
 | 
			
		||||
@@ -403,8 +398,9 @@ void LD2420Component::set_operating_mode(const std::string &state) {
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
      // Set the current data back so we don't have new data that can be applied in error.
 | 
			
		||||
      if (this->get_calibration_())
 | 
			
		||||
      if (this->get_calibration_()) {
 | 
			
		||||
        memcpy(&this->new_config, &this->current_config, sizeof(this->current_config));
 | 
			
		||||
      }
 | 
			
		||||
      this->set_calibration_(false);
 | 
			
		||||
    }
 | 
			
		||||
  } else {
 | 
			
		||||
@@ -414,30 +410,32 @@ void LD2420Component::set_operating_mode(const std::string &state) {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2420Component::readline_(int rx_data, uint8_t *buffer, int len) {
 | 
			
		||||
  static int pos = 0;
 | 
			
		||||
 | 
			
		||||
  if (rx_data >= 0) {
 | 
			
		||||
    if (pos < len - 1) {
 | 
			
		||||
      buffer[pos++] = rx_data;
 | 
			
		||||
      buffer[pos] = 0;
 | 
			
		||||
    } else {
 | 
			
		||||
      pos = 0;
 | 
			
		||||
    }
 | 
			
		||||
    if (pos >= 4) {
 | 
			
		||||
      if (memcmp(&buffer[pos - 4], &CMD_FRAME_FOOTER, sizeof(CMD_FRAME_FOOTER)) == 0) {
 | 
			
		||||
        this->set_cmd_active_(false);  // Set command state to inactive after responce.
 | 
			
		||||
        this->handle_ack_data_(buffer, pos);
 | 
			
		||||
        pos = 0;
 | 
			
		||||
      } else if ((buffer[pos - 2] == 0x0D && buffer[pos - 1] == 0x0A) &&
 | 
			
		||||
                 (this->get_mode_() == CMD_SYSTEM_MODE_SIMPLE)) {
 | 
			
		||||
        this->handle_simple_mode_(buffer, pos);
 | 
			
		||||
        pos = 0;
 | 
			
		||||
      } else if ((memcmp(&buffer[pos - 4], &ENERGY_FRAME_FOOTER, sizeof(ENERGY_FRAME_FOOTER)) == 0) &&
 | 
			
		||||
                 (this->get_mode_() == CMD_SYSTEM_MODE_ENERGY)) {
 | 
			
		||||
        this->handle_energy_mode_(buffer, pos);
 | 
			
		||||
        pos = 0;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  if (rx_data < 0) {
 | 
			
		||||
    return;  // No data available
 | 
			
		||||
  }
 | 
			
		||||
  if (this->buffer_pos_ < len - 1) {
 | 
			
		||||
    buffer[this->buffer_pos_++] = rx_data;
 | 
			
		||||
    buffer[this->buffer_pos_] = 0;
 | 
			
		||||
  } else {
 | 
			
		||||
    // We should never get here, but just in case...
 | 
			
		||||
    ESP_LOGW(TAG, "Max command length exceeded; ignoring");
 | 
			
		||||
    this->buffer_pos_ = 0;
 | 
			
		||||
  }
 | 
			
		||||
  if (this->buffer_pos_ < 4) {
 | 
			
		||||
    return;  // Not enough data to process yet
 | 
			
		||||
  }
 | 
			
		||||
  if (memcmp(&buffer[this->buffer_pos_ - 4], &CMD_FRAME_FOOTER, sizeof(CMD_FRAME_FOOTER)) == 0) {
 | 
			
		||||
    this->cmd_active_ = false;  // Set command state to inactive after response
 | 
			
		||||
    this->handle_ack_data_(buffer, this->buffer_pos_);
 | 
			
		||||
    this->buffer_pos_ = 0;
 | 
			
		||||
  } else if ((buffer[this->buffer_pos_ - 2] == 0x0D && buffer[this->buffer_pos_ - 1] == 0x0A) &&
 | 
			
		||||
             (this->get_mode_() == CMD_SYSTEM_MODE_SIMPLE)) {
 | 
			
		||||
    this->handle_simple_mode_(buffer, this->buffer_pos_);
 | 
			
		||||
    this->buffer_pos_ = 0;
 | 
			
		||||
  } else if ((memcmp(&buffer[this->buffer_pos_ - 4], &ENERGY_FRAME_FOOTER, sizeof(ENERGY_FRAME_FOOTER)) == 0) &&
 | 
			
		||||
             (this->get_mode_() == CMD_SYSTEM_MODE_ENERGY)) {
 | 
			
		||||
    this->handle_energy_mode_(buffer, this->buffer_pos_);
 | 
			
		||||
    this->buffer_pos_ = 0;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -462,8 +460,9 @@ void LD2420Component::handle_energy_mode_(uint8_t *buffer, int len) {
 | 
			
		||||
 | 
			
		||||
  // Resonable refresh rate for home assistant database size health
 | 
			
		||||
  const int32_t current_millis = App.get_loop_component_start_time();
 | 
			
		||||
  if (current_millis - this->last_periodic_millis < REFRESH_RATE_MS)
 | 
			
		||||
  if (current_millis - this->last_periodic_millis < REFRESH_RATE_MS) {
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  this->last_periodic_millis = current_millis;
 | 
			
		||||
  for (auto &listener : this->listeners_) {
 | 
			
		||||
    listener->on_distance(this->get_distance_());
 | 
			
		||||
@@ -506,14 +505,16 @@ void LD2420Component::handle_simple_mode_(const uint8_t *inbuf, int len) {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  outbuf[index] = '\0';
 | 
			
		||||
  if (index > 1)
 | 
			
		||||
  if (index > 1) {
 | 
			
		||||
    this->set_distance_(strtol(outbuf, &endptr, 10));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (this->get_mode_() == CMD_SYSTEM_MODE_SIMPLE) {
 | 
			
		||||
    // Resonable refresh rate for home assistant database size health
 | 
			
		||||
    const int32_t current_millis = App.get_loop_component_start_time();
 | 
			
		||||
    if (current_millis - this->last_normal_periodic_millis < REFRESH_RATE_MS)
 | 
			
		||||
    if (current_millis - this->last_normal_periodic_millis < REFRESH_RATE_MS) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    this->last_normal_periodic_millis = current_millis;
 | 
			
		||||
    for (auto &listener : this->listeners_)
 | 
			
		||||
      listener->on_distance(this->get_distance_());
 | 
			
		||||
@@ -593,11 +594,12 @@ void LD2420Component::handle_ack_data_(uint8_t *buffer, int len) {
 | 
			
		||||
int LD2420Component::send_cmd_from_array(CmdFrameT frame) {
 | 
			
		||||
  uint32_t start_millis = millis();
 | 
			
		||||
  uint8_t error = 0;
 | 
			
		||||
  uint8_t ack_buffer[64];
 | 
			
		||||
  uint8_t cmd_buffer[64];
 | 
			
		||||
  uint8_t ack_buffer[MAX_LINE_LENGTH];
 | 
			
		||||
  uint8_t cmd_buffer[MAX_LINE_LENGTH];
 | 
			
		||||
  this->cmd_reply_.ack = false;
 | 
			
		||||
  if (frame.command != CMD_RESTART)
 | 
			
		||||
    this->set_cmd_active_(true);  // Restart does not reply, thus no ack state required.
 | 
			
		||||
  if (frame.command != CMD_RESTART) {
 | 
			
		||||
    this->cmd_active_ = true;
 | 
			
		||||
  }  // Restart does not reply, thus no ack state required
 | 
			
		||||
  uint8_t retry = 3;
 | 
			
		||||
  while (retry) {
 | 
			
		||||
    frame.length = 0;
 | 
			
		||||
@@ -619,9 +621,7 @@ int LD2420Component::send_cmd_from_array(CmdFrameT frame) {
 | 
			
		||||
 | 
			
		||||
    memcpy(cmd_buffer + frame.length, &frame.footer, sizeof(frame.footer));
 | 
			
		||||
    frame.length += sizeof(frame.footer);
 | 
			
		||||
    for (uint16_t index = 0; index < frame.length; index++) {
 | 
			
		||||
      this->write_byte(cmd_buffer[index]);
 | 
			
		||||
    }
 | 
			
		||||
    this->write_array(cmd_buffer, frame.length);
 | 
			
		||||
 | 
			
		||||
    error = 0;
 | 
			
		||||
    if (frame.command == CMD_RESTART) {
 | 
			
		||||
@@ -630,7 +630,7 @@ int LD2420Component::send_cmd_from_array(CmdFrameT frame) {
 | 
			
		||||
 | 
			
		||||
    while (!this->cmd_reply_.ack) {
 | 
			
		||||
      while (this->available()) {
 | 
			
		||||
        this->readline_(read(), ack_buffer, sizeof(ack_buffer));
 | 
			
		||||
        this->readline_(this->read(), ack_buffer, sizeof(ack_buffer));
 | 
			
		||||
      }
 | 
			
		||||
      delay_microseconds_safe(1450);
 | 
			
		||||
      // Wait on an Rx from the LD2420 for up to 3 1 second loops, otherwise it could trigger a WDT.
 | 
			
		||||
@@ -641,10 +641,12 @@ int LD2420Component::send_cmd_from_array(CmdFrameT frame) {
 | 
			
		||||
        break;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    if (this->cmd_reply_.ack)
 | 
			
		||||
    if (this->cmd_reply_.ack) {
 | 
			
		||||
      retry = 0;
 | 
			
		||||
    if (this->cmd_reply_.error > 0)
 | 
			
		||||
    }
 | 
			
		||||
    if (this->cmd_reply_.error > 0) {
 | 
			
		||||
      this->handle_cmd_error(error);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  return error;
 | 
			
		||||
}
 | 
			
		||||
@@ -764,8 +766,9 @@ void LD2420Component::set_system_mode(uint16_t mode) {
 | 
			
		||||
  cmd_frame.data_length += sizeof(unknown_parm);
 | 
			
		||||
  cmd_frame.footer = CMD_FRAME_FOOTER;
 | 
			
		||||
  ESP_LOGV(TAG, "Sending write system mode command: %2X", cmd_frame.command);
 | 
			
		||||
  if (this->send_cmd_from_array(cmd_frame) == 0)
 | 
			
		||||
  if (this->send_cmd_from_array(cmd_frame) == 0) {
 | 
			
		||||
    this->set_mode_(mode);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2420Component::get_firmware_version_() {
 | 
			
		||||
@@ -840,18 +843,24 @@ void LD2420Component::set_gate_threshold(uint8_t gate) {
 | 
			
		||||
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
void LD2420Component::init_gate_config_numbers() {
 | 
			
		||||
  if (this->gate_timeout_number_ != nullptr)
 | 
			
		||||
  if (this->gate_timeout_number_ != nullptr) {
 | 
			
		||||
    this->gate_timeout_number_->publish_state(static_cast<uint16_t>(this->current_config.timeout));
 | 
			
		||||
  if (this->gate_select_number_ != nullptr)
 | 
			
		||||
  }
 | 
			
		||||
  if (this->gate_select_number_ != nullptr) {
 | 
			
		||||
    this->gate_select_number_->publish_state(0);
 | 
			
		||||
  if (this->min_gate_distance_number_ != nullptr)
 | 
			
		||||
  }
 | 
			
		||||
  if (this->min_gate_distance_number_ != nullptr) {
 | 
			
		||||
    this->min_gate_distance_number_->publish_state(static_cast<uint16_t>(this->current_config.min_gate));
 | 
			
		||||
  if (this->max_gate_distance_number_ != nullptr)
 | 
			
		||||
  }
 | 
			
		||||
  if (this->max_gate_distance_number_ != nullptr) {
 | 
			
		||||
    this->max_gate_distance_number_->publish_state(static_cast<uint16_t>(this->current_config.max_gate));
 | 
			
		||||
  if (this->gate_move_sensitivity_factor_number_ != nullptr)
 | 
			
		||||
  }
 | 
			
		||||
  if (this->gate_move_sensitivity_factor_number_ != nullptr) {
 | 
			
		||||
    this->gate_move_sensitivity_factor_number_->publish_state(this->gate_move_sensitivity_factor);
 | 
			
		||||
  if (this->gate_still_sensitivity_factor_number_ != nullptr)
 | 
			
		||||
  }
 | 
			
		||||
  if (this->gate_still_sensitivity_factor_number_ != nullptr) {
 | 
			
		||||
    this->gate_still_sensitivity_factor_number_->publish_state(this->gate_still_sensitivity_factor);
 | 
			
		||||
  }
 | 
			
		||||
  for (uint8_t gate = 0; gate < TOTAL_GATES; gate++) {
 | 
			
		||||
    if (this->gate_still_threshold_numbers_[gate] != nullptr) {
 | 
			
		||||
      this->gate_still_threshold_numbers_[gate]->publish_state(
 | 
			
		||||
 
 | 
			
		||||
@@ -20,8 +20,9 @@
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2420 {
 | 
			
		||||
 | 
			
		||||
static const uint8_t TOTAL_GATES = 16;
 | 
			
		||||
static const uint8_t CALIBRATE_SAMPLES = 64;
 | 
			
		||||
static const uint8_t MAX_LINE_LENGTH = 46;  // Max characters for serial buffer
 | 
			
		||||
static const uint8_t TOTAL_GATES = 16;
 | 
			
		||||
 | 
			
		||||
enum OpMode : uint8_t {
 | 
			
		||||
  OP_NORMAL_MODE = 1,
 | 
			
		||||
@@ -118,10 +119,10 @@ class LD2420Component : public Component, public uart::UARTDevice {
 | 
			
		||||
 | 
			
		||||
  float gate_move_sensitivity_factor{0.5};
 | 
			
		||||
  float gate_still_sensitivity_factor{0.5};
 | 
			
		||||
  int32_t last_periodic_millis = millis();
 | 
			
		||||
  int32_t report_periodic_millis = millis();
 | 
			
		||||
  int32_t monitor_periodic_millis = millis();
 | 
			
		||||
  int32_t last_normal_periodic_millis = millis();
 | 
			
		||||
  int32_t last_periodic_millis{0};
 | 
			
		||||
  int32_t report_periodic_millis{0};
 | 
			
		||||
  int32_t monitor_periodic_millis{0};
 | 
			
		||||
  int32_t last_normal_periodic_millis{0};
 | 
			
		||||
  uint16_t radar_data[TOTAL_GATES][CALIBRATE_SAMPLES];
 | 
			
		||||
  uint16_t gate_avg[TOTAL_GATES];
 | 
			
		||||
  uint16_t gate_peak[TOTAL_GATES];
 | 
			
		||||
@@ -161,8 +162,6 @@ class LD2420Component : public Component, public uart::UARTDevice {
 | 
			
		||||
  void set_presence_(bool presence) { this->presence_ = presence; };
 | 
			
		||||
  uint16_t get_distance_() { return this->distance_; };
 | 
			
		||||
  void set_distance_(uint16_t distance) { this->distance_ = distance; };
 | 
			
		||||
  bool get_cmd_active_() { return this->cmd_active_; };
 | 
			
		||||
  void set_cmd_active_(bool active) { this->cmd_active_ = active; };
 | 
			
		||||
  void handle_simple_mode_(const uint8_t *inbuf, int len);
 | 
			
		||||
  void handle_energy_mode_(uint8_t *buffer, int len);
 | 
			
		||||
  void handle_ack_data_(uint8_t *buffer, int len);
 | 
			
		||||
@@ -181,12 +180,11 @@ class LD2420Component : public Component, public uart::UARTDevice {
 | 
			
		||||
  std::vector<number::Number *> gate_move_threshold_numbers_ = std::vector<number::Number *>(16);
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
  uint32_t max_distance_gate_;
 | 
			
		||||
  uint32_t min_distance_gate_;
 | 
			
		||||
  uint16_t distance_{0};
 | 
			
		||||
  uint16_t system_mode_;
 | 
			
		||||
  uint16_t gate_energy_[TOTAL_GATES];
 | 
			
		||||
  uint16_t distance_{0};
 | 
			
		||||
  uint8_t config_checksum_{0};
 | 
			
		||||
  uint8_t buffer_pos_{0};  // where to resume processing/populating buffer
 | 
			
		||||
  uint8_t buffer_data_[MAX_LINE_LENGTH];
 | 
			
		||||
  char firmware_ver_[8]{"v0.0.0"};
 | 
			
		||||
  bool cmd_active_{false};
 | 
			
		||||
  bool presence_{false};
 | 
			
		||||
 
 | 
			
		||||
@@ -2,7 +2,7 @@
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "LD2420.number";
 | 
			
		||||
static const char *const TAG = "ld2420.number";
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2420 {
 | 
			
		||||
 
 | 
			
		||||
@@ -5,7 +5,7 @@
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2420 {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "LD2420.select";
 | 
			
		||||
static const char *const TAG = "ld2420.select";
 | 
			
		||||
 | 
			
		||||
void LD2420Select::control(const std::string &value) {
 | 
			
		||||
  this->publish_state(value);
 | 
			
		||||
 
 | 
			
		||||
@@ -5,10 +5,10 @@
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2420 {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "LD2420.sensor";
 | 
			
		||||
static const char *const TAG = "ld2420.sensor";
 | 
			
		||||
 | 
			
		||||
void LD2420Sensor::dump_config() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "LD2420 Sensor:");
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Sensor:");
 | 
			
		||||
  LOG_SENSOR("  ", "Distance", this->distance_sensor_);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -5,10 +5,10 @@
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2420 {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "LD2420.text_sensor";
 | 
			
		||||
static const char *const TAG = "ld2420.text_sensor";
 | 
			
		||||
 | 
			
		||||
void LD2420TextSensor::dump_config() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "LD2420 TextSensor:");
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Text Sensor:");
 | 
			
		||||
  LOG_TEXT_SENSOR("  ", "Firmware", this->fw_version_text_sensor_);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -13,13 +13,13 @@ from esphome.const import (
 | 
			
		||||
 | 
			
		||||
from .. import CONF_LD2450_ID, LD2450Component, ld2450_ns
 | 
			
		||||
 | 
			
		||||
ResetButton = ld2450_ns.class_("ResetButton", button.Button)
 | 
			
		||||
FactoryResetButton = ld2450_ns.class_("FactoryResetButton", button.Button)
 | 
			
		||||
RestartButton = ld2450_ns.class_("RestartButton", button.Button)
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = {
 | 
			
		||||
    cv.GenerateID(CONF_LD2450_ID): cv.use_id(LD2450Component),
 | 
			
		||||
    cv.Optional(CONF_FACTORY_RESET): button.button_schema(
 | 
			
		||||
        ResetButton,
 | 
			
		||||
        FactoryResetButton,
 | 
			
		||||
        device_class=DEVICE_CLASS_RESTART,
 | 
			
		||||
        entity_category=ENTITY_CATEGORY_CONFIG,
 | 
			
		||||
        icon=ICON_RESTART_ALERT,
 | 
			
		||||
@@ -38,7 +38,7 @@ async def to_code(config):
 | 
			
		||||
    if factory_reset_config := config.get(CONF_FACTORY_RESET):
 | 
			
		||||
        b = await button.new_button(factory_reset_config)
 | 
			
		||||
        await cg.register_parented(b, config[CONF_LD2450_ID])
 | 
			
		||||
        cg.add(ld2450_component.set_reset_button(b))
 | 
			
		||||
        cg.add(ld2450_component.set_factory_reset_button(b))
 | 
			
		||||
    if restart_config := config.get(CONF_RESTART):
 | 
			
		||||
        b = await button.new_button(restart_config)
 | 
			
		||||
        await cg.register_parented(b, config[CONF_LD2450_ID])
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,9 @@
 | 
			
		||||
#include "factory_reset_button.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2450 {
 | 
			
		||||
 | 
			
		||||
void FactoryResetButton::press_action() { this->parent_->factory_reset(); }
 | 
			
		||||
 | 
			
		||||
}  // namespace ld2450
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -6,9 +6,9 @@
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2450 {
 | 
			
		||||
 | 
			
		||||
class ResetButton : public button::Button, public Parented<LD2450Component> {
 | 
			
		||||
class FactoryResetButton : public button::Button, public Parented<LD2450Component> {
 | 
			
		||||
 public:
 | 
			
		||||
  ResetButton() = default;
 | 
			
		||||
  FactoryResetButton() = default;
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  void press_action() override;
 | 
			
		||||
@@ -1,9 +0,0 @@
 | 
			
		||||
#include "reset_button.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace ld2450 {
 | 
			
		||||
 | 
			
		||||
void ResetButton::press_action() { this->parent_->factory_reset(); }
 | 
			
		||||
 | 
			
		||||
}  // namespace ld2450
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
@@ -1,5 +1,6 @@
 | 
			
		||||
#include "ld2450.h"
 | 
			
		||||
#include <utility>
 | 
			
		||||
#include <cmath>
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
#include "esphome/components/number/number.h"
 | 
			
		||||
#endif
 | 
			
		||||
@@ -17,11 +18,10 @@ namespace esphome {
 | 
			
		||||
namespace ld2450 {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "ld2450";
 | 
			
		||||
static const char *const NO_MAC = "08:05:04:03:02:01";
 | 
			
		||||
static const char *const UNKNOWN_MAC = "unknown";
 | 
			
		||||
static const char *const VERSION_FMT = "%u.%02X.%02X%02X%02X%02X";
 | 
			
		||||
 | 
			
		||||
enum BaudRateStructure : uint8_t {
 | 
			
		||||
enum BaudRate : uint8_t {
 | 
			
		||||
  BAUD_RATE_9600 = 1,
 | 
			
		||||
  BAUD_RATE_19200 = 2,
 | 
			
		||||
  BAUD_RATE_38400 = 3,
 | 
			
		||||
@@ -32,14 +32,13 @@ enum BaudRateStructure : uint8_t {
 | 
			
		||||
  BAUD_RATE_460800 = 8
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// Zone type struct
 | 
			
		||||
enum ZoneTypeStructure : uint8_t {
 | 
			
		||||
enum ZoneType : uint8_t {
 | 
			
		||||
  ZONE_DISABLED = 0,
 | 
			
		||||
  ZONE_DETECTION = 1,
 | 
			
		||||
  ZONE_FILTER = 2,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum PeriodicDataStructure : uint8_t {
 | 
			
		||||
enum PeriodicData : uint8_t {
 | 
			
		||||
  TARGET_X = 4,
 | 
			
		||||
  TARGET_Y = 6,
 | 
			
		||||
  TARGET_SPEED = 8,
 | 
			
		||||
@@ -47,12 +46,12 @@ enum PeriodicDataStructure : uint8_t {
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum PeriodicDataValue : uint8_t {
 | 
			
		||||
  HEAD = 0xAA,
 | 
			
		||||
  END = 0x55,
 | 
			
		||||
  HEADER = 0xAA,
 | 
			
		||||
  FOOTER = 0x55,
 | 
			
		||||
  CHECK = 0x00,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum AckDataStructure : uint8_t {
 | 
			
		||||
enum AckData : uint8_t {
 | 
			
		||||
  COMMAND = 6,
 | 
			
		||||
  COMMAND_STATUS = 7,
 | 
			
		||||
};
 | 
			
		||||
@@ -60,11 +59,11 @@ enum AckDataStructure : uint8_t {
 | 
			
		||||
// Memory-efficient lookup tables
 | 
			
		||||
struct StringToUint8 {
 | 
			
		||||
  const char *str;
 | 
			
		||||
  uint8_t value;
 | 
			
		||||
  const uint8_t value;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct Uint8ToString {
 | 
			
		||||
  uint8_t value;
 | 
			
		||||
  const uint8_t value;
 | 
			
		||||
  const char *str;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
@@ -74,6 +73,13 @@ constexpr StringToUint8 BAUD_RATES_BY_STR[] = {
 | 
			
		||||
    {"256000", BAUD_RATE_256000}, {"460800", BAUD_RATE_460800},
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
constexpr Uint8ToString DIRECTION_BY_UINT[] = {
 | 
			
		||||
    {DIRECTION_APPROACHING, "Approaching"},
 | 
			
		||||
    {DIRECTION_MOVING_AWAY, "Moving away"},
 | 
			
		||||
    {DIRECTION_STATIONARY, "Stationary"},
 | 
			
		||||
    {DIRECTION_NA, "NA"},
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
constexpr Uint8ToString ZONE_TYPE_BY_UINT[] = {
 | 
			
		||||
    {ZONE_DISABLED, "Disabled"},
 | 
			
		||||
    {ZONE_DETECTION, "Detection"},
 | 
			
		||||
@@ -103,36 +109,38 @@ template<size_t N> const char *find_str(const Uint8ToString (&arr)[N], uint8_t v
 | 
			
		||||
  return "";  // Not found
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// LD2450 serial command header & footer
 | 
			
		||||
static const uint8_t CMD_FRAME_HEADER[4] = {0xFD, 0xFC, 0xFB, 0xFA};
 | 
			
		||||
static const uint8_t CMD_FRAME_END[4] = {0x04, 0x03, 0x02, 0x01};
 | 
			
		||||
// LD2450 UART Serial Commands
 | 
			
		||||
static const uint8_t CMD_ENABLE_CONF = 0xFF;
 | 
			
		||||
static const uint8_t CMD_DISABLE_CONF = 0xFE;
 | 
			
		||||
static const uint8_t CMD_VERSION = 0xA0;
 | 
			
		||||
static const uint8_t CMD_MAC = 0xA5;
 | 
			
		||||
static const uint8_t CMD_RESET = 0xA2;
 | 
			
		||||
static const uint8_t CMD_RESTART = 0xA3;
 | 
			
		||||
static const uint8_t CMD_BLUETOOTH = 0xA4;
 | 
			
		||||
static const uint8_t CMD_SINGLE_TARGET_MODE = 0x80;
 | 
			
		||||
static const uint8_t CMD_MULTI_TARGET_MODE = 0x90;
 | 
			
		||||
static const uint8_t CMD_QUERY_TARGET_MODE = 0x91;
 | 
			
		||||
static const uint8_t CMD_SET_BAUD_RATE = 0xA1;
 | 
			
		||||
static const uint8_t CMD_QUERY_ZONE = 0xC1;
 | 
			
		||||
static const uint8_t CMD_SET_ZONE = 0xC2;
 | 
			
		||||
static constexpr uint8_t CMD_ENABLE_CONF = 0xFF;
 | 
			
		||||
static constexpr uint8_t CMD_DISABLE_CONF = 0xFE;
 | 
			
		||||
static constexpr uint8_t CMD_QUERY_VERSION = 0xA0;
 | 
			
		||||
static constexpr uint8_t CMD_QUERY_MAC_ADDRESS = 0xA5;
 | 
			
		||||
static constexpr uint8_t CMD_RESET = 0xA2;
 | 
			
		||||
static constexpr uint8_t CMD_RESTART = 0xA3;
 | 
			
		||||
static constexpr uint8_t CMD_BLUETOOTH = 0xA4;
 | 
			
		||||
static constexpr uint8_t CMD_SINGLE_TARGET_MODE = 0x80;
 | 
			
		||||
static constexpr uint8_t CMD_MULTI_TARGET_MODE = 0x90;
 | 
			
		||||
static constexpr uint8_t CMD_QUERY_TARGET_MODE = 0x91;
 | 
			
		||||
static constexpr uint8_t CMD_SET_BAUD_RATE = 0xA1;
 | 
			
		||||
static constexpr uint8_t CMD_QUERY_ZONE = 0xC1;
 | 
			
		||||
static constexpr uint8_t CMD_SET_ZONE = 0xC2;
 | 
			
		||||
// Header & Footer size
 | 
			
		||||
static constexpr uint8_t HEADER_FOOTER_SIZE = 4;
 | 
			
		||||
// Command Header & Footer
 | 
			
		||||
static constexpr uint8_t CMD_FRAME_HEADER[HEADER_FOOTER_SIZE] = {0xFD, 0xFC, 0xFB, 0xFA};
 | 
			
		||||
static constexpr uint8_t CMD_FRAME_FOOTER[HEADER_FOOTER_SIZE] = {0x04, 0x03, 0x02, 0x01};
 | 
			
		||||
// Data Header & Footer
 | 
			
		||||
static constexpr uint8_t DATA_FRAME_HEADER[HEADER_FOOTER_SIZE] = {0xAA, 0xFF, 0x03, 0x00};
 | 
			
		||||
static constexpr uint8_t DATA_FRAME_FOOTER[2] = {0x55, 0xCC};
 | 
			
		||||
// MAC address the module uses when Bluetooth is disabled
 | 
			
		||||
static constexpr uint8_t NO_MAC[] = {0x08, 0x05, 0x04, 0x03, 0x02, 0x01};
 | 
			
		||||
 | 
			
		||||
static inline uint16_t convert_seconds_to_ms(uint16_t value) { return value * 1000; };
 | 
			
		||||
 | 
			
		||||
static inline std::string convert_signed_int_to_hex(int value) {
 | 
			
		||||
  auto value_as_str = str_snprintf("%04x", 4, value & 0xFFFF);
 | 
			
		||||
  return value_as_str;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void convert_int_values_to_hex(const int *values, uint8_t *bytes) {
 | 
			
		||||
  for (int i = 0; i < 4; i++) {
 | 
			
		||||
    std::string temp_hex = convert_signed_int_to_hex(values[i]);
 | 
			
		||||
    bytes[i * 2] = std::stoi(temp_hex.substr(2, 2), nullptr, 16);      // Store high byte
 | 
			
		||||
    bytes[i * 2 + 1] = std::stoi(temp_hex.substr(0, 2), nullptr, 16);  // Store low byte
 | 
			
		||||
  for (uint8_t i = 0; i < 4; i++) {
 | 
			
		||||
    uint16_t val = values[i] & 0xFFFF;
 | 
			
		||||
    bytes[i * 2] = val & 0xFF;             // Store low byte first (little-endian)
 | 
			
		||||
    bytes[i * 2 + 1] = (val >> 8) & 0xFF;  // Store high byte second
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -170,18 +178,13 @@ static inline float calculate_angle(float base, float hypotenuse) {
 | 
			
		||||
  return angle_degrees;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline std::string get_direction(int16_t speed) {
 | 
			
		||||
  static const char *const APPROACHING = "Approaching";
 | 
			
		||||
  static const char *const MOVING_AWAY = "Moving away";
 | 
			
		||||
  static const char *const STATIONARY = "Stationary";
 | 
			
		||||
 | 
			
		||||
  if (speed > 0) {
 | 
			
		||||
    return MOVING_AWAY;
 | 
			
		||||
static bool validate_header_footer(const uint8_t *header_footer, const uint8_t *buffer) {
 | 
			
		||||
  for (uint8_t i = 0; i < HEADER_FOOTER_SIZE; i++) {
 | 
			
		||||
    if (header_footer[i] != buffer[i]) {
 | 
			
		||||
      return false;  // Mismatch in header/footer
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  if (speed < 0) {
 | 
			
		||||
    return APPROACHING;
 | 
			
		||||
  }
 | 
			
		||||
  return STATIONARY;
 | 
			
		||||
  return true;  // Valid header/footer
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2450Component::setup() {
 | 
			
		||||
@@ -196,84 +199,93 @@ void LD2450Component::setup() {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2450Component::dump_config() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "LD2450:");
 | 
			
		||||
  std::string mac_str =
 | 
			
		||||
      mac_address_is_valid(this->mac_address_) ? format_mac_address_pretty(this->mac_address_) : UNKNOWN_MAC;
 | 
			
		||||
  std::string version = str_sprintf(VERSION_FMT, this->version_[1], this->version_[0], this->version_[5],
 | 
			
		||||
                                    this->version_[4], this->version_[3], this->version_[2]);
 | 
			
		||||
  ESP_LOGCONFIG(TAG,
 | 
			
		||||
                "LD2450:\n"
 | 
			
		||||
                "  Firmware version: %s\n"
 | 
			
		||||
                "  MAC address: %s\n"
 | 
			
		||||
                "  Throttle: %u ms",
 | 
			
		||||
                version.c_str(), mac_str.c_str(), this->throttle_);
 | 
			
		||||
#ifdef USE_BINARY_SENSOR
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "TargetBinarySensor", this->target_binary_sensor_);
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "MovingTargetBinarySensor", this->moving_target_binary_sensor_);
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "StillTargetBinarySensor", this->still_target_binary_sensor_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
  LOG_SWITCH("  ", "BluetoothSwitch", this->bluetooth_switch_);
 | 
			
		||||
  LOG_SWITCH("  ", "MultiTargetSwitch", this->multi_target_switch_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_BUTTON
 | 
			
		||||
  LOG_BUTTON("  ", "ResetButton", this->reset_button_);
 | 
			
		||||
  LOG_BUTTON("  ", "RestartButton", this->restart_button_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Binary Sensors:");
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "MovingTarget", this->moving_target_binary_sensor_);
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "StillTarget", this->still_target_binary_sensor_);
 | 
			
		||||
  LOG_BINARY_SENSOR("  ", "Target", this->target_binary_sensor_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
  LOG_SENSOR("  ", "TargetCountSensor", this->target_count_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "StillTargetCountSensor", this->still_target_count_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "MovingTargetCountSensor", this->moving_target_count_sensor_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Sensors:");
 | 
			
		||||
  LOG_SENSOR("  ", "MovingTargetCount", this->moving_target_count_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "StillTargetCount", this->still_target_count_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "TargetCount", this->target_count_sensor_);
 | 
			
		||||
  for (sensor::Sensor *s : this->move_x_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "NthTargetXSensor", s);
 | 
			
		||||
    LOG_SENSOR("  ", "TargetX", s);
 | 
			
		||||
  }
 | 
			
		||||
  for (sensor::Sensor *s : this->move_y_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "NthTargetYSensor", s);
 | 
			
		||||
  }
 | 
			
		||||
  for (sensor::Sensor *s : this->move_speed_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "NthTargetSpeedSensor", s);
 | 
			
		||||
    LOG_SENSOR("  ", "TargetY", s);
 | 
			
		||||
  }
 | 
			
		||||
  for (sensor::Sensor *s : this->move_angle_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "NthTargetAngleSensor", s);
 | 
			
		||||
    LOG_SENSOR("  ", "TargetAngle", s);
 | 
			
		||||
  }
 | 
			
		||||
  for (sensor::Sensor *s : this->move_distance_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "NthTargetDistanceSensor", s);
 | 
			
		||||
    LOG_SENSOR("  ", "TargetDistance", s);
 | 
			
		||||
  }
 | 
			
		||||
  for (sensor::Sensor *s : this->move_resolution_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "NthTargetResolutionSensor", s);
 | 
			
		||||
    LOG_SENSOR("  ", "TargetResolution", s);
 | 
			
		||||
  }
 | 
			
		||||
  for (sensor::Sensor *s : this->move_speed_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "TargetSpeed", s);
 | 
			
		||||
  }
 | 
			
		||||
  for (sensor::Sensor *s : this->zone_target_count_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "NthZoneTargetCountSensor", s);
 | 
			
		||||
  }
 | 
			
		||||
  for (sensor::Sensor *s : this->zone_still_target_count_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "NthZoneStillTargetCountSensor", s);
 | 
			
		||||
    LOG_SENSOR("  ", "ZoneTargetCount", s);
 | 
			
		||||
  }
 | 
			
		||||
  for (sensor::Sensor *s : this->zone_moving_target_count_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "NthZoneMovingTargetCountSensor", s);
 | 
			
		||||
    LOG_SENSOR("  ", "ZoneMovingTargetCount", s);
 | 
			
		||||
  }
 | 
			
		||||
  for (sensor::Sensor *s : this->zone_still_target_count_sensors_) {
 | 
			
		||||
    LOG_SENSOR("  ", "ZoneStillTargetCount", s);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
  LOG_TEXT_SENSOR("  ", "VersionTextSensor", this->version_text_sensor_);
 | 
			
		||||
  LOG_TEXT_SENSOR("  ", "MacTextSensor", this->mac_text_sensor_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Text Sensors:");
 | 
			
		||||
  LOG_TEXT_SENSOR("  ", "Version", this->version_text_sensor_);
 | 
			
		||||
  LOG_TEXT_SENSOR("  ", "Mac", this->mac_text_sensor_);
 | 
			
		||||
  for (text_sensor::TextSensor *s : this->direction_text_sensors_) {
 | 
			
		||||
    LOG_TEXT_SENSOR("  ", "NthDirectionTextSensor", s);
 | 
			
		||||
    LOG_TEXT_SENSOR("  ", "Direction", s);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Numbers:");
 | 
			
		||||
  LOG_NUMBER("  ", "PresenceTimeout", this->presence_timeout_number_);
 | 
			
		||||
  for (auto n : this->zone_numbers_) {
 | 
			
		||||
    LOG_NUMBER("  ", "ZoneX1Number", n.x1);
 | 
			
		||||
    LOG_NUMBER("  ", "ZoneY1Number", n.y1);
 | 
			
		||||
    LOG_NUMBER("  ", "ZoneX2Number", n.x2);
 | 
			
		||||
    LOG_NUMBER("  ", "ZoneY2Number", n.y2);
 | 
			
		||||
    LOG_NUMBER("  ", "ZoneX1", n.x1);
 | 
			
		||||
    LOG_NUMBER("  ", "ZoneY1", n.y1);
 | 
			
		||||
    LOG_NUMBER("  ", "ZoneX2", n.x2);
 | 
			
		||||
    LOG_NUMBER("  ", "ZoneY2", n.y2);
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
  LOG_SELECT("  ", "BaudRateSelect", this->baud_rate_select_);
 | 
			
		||||
  LOG_SELECT("  ", "ZoneTypeSelect", this->zone_type_select_);
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Selects:");
 | 
			
		||||
  LOG_SELECT("  ", "BaudRate", this->baud_rate_select_);
 | 
			
		||||
  LOG_SELECT("  ", "ZoneType", this->zone_type_select_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
  LOG_NUMBER("  ", "PresenceTimeoutNumber", this->presence_timeout_number_);
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Switches:");
 | 
			
		||||
  LOG_SWITCH("  ", "Bluetooth", this->bluetooth_switch_);
 | 
			
		||||
  LOG_SWITCH("  ", "MultiTarget", this->multi_target_switch_);
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_BUTTON
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "Buttons:");
 | 
			
		||||
  LOG_BUTTON("  ", "FactoryReset", this->factory_reset_button_);
 | 
			
		||||
  LOG_BUTTON("  ", "Restart", this->restart_button_);
 | 
			
		||||
#endif
 | 
			
		||||
  ESP_LOGCONFIG(TAG,
 | 
			
		||||
                "  Throttle: %ums\n"
 | 
			
		||||
                "  MAC Address: %s\n"
 | 
			
		||||
                "  Firmware version: %s",
 | 
			
		||||
                this->throttle_, this->mac_ == NO_MAC ? UNKNOWN_MAC : this->mac_.c_str(), this->version_.c_str());
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LD2450Component::loop() {
 | 
			
		||||
  while (this->available()) {
 | 
			
		||||
    this->readline_(read(), this->buffer_data_, MAX_LINE_LENGTH);
 | 
			
		||||
    this->readline_(this->read());
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -308,7 +320,7 @@ void LD2450Component::set_radar_zone(int32_t zone_type, int32_t zone1_x1, int32_
 | 
			
		||||
  this->zone_type_ = zone_type;
 | 
			
		||||
  int zone_parameters[12] = {zone1_x1, zone1_y1, zone1_x2, zone1_y2, zone2_x1, zone2_y1,
 | 
			
		||||
                             zone2_x2, zone2_y2, zone3_x1, zone3_y1, zone3_x2, zone3_y2};
 | 
			
		||||
  for (int i = 0; i < MAX_ZONES; i++) {
 | 
			
		||||
  for (uint8_t i = 0; i < MAX_ZONES; i++) {
 | 
			
		||||
    this->zone_config_[i].x1 = zone_parameters[i * 4];
 | 
			
		||||
    this->zone_config_[i].y1 = zone_parameters[i * 4 + 1];
 | 
			
		||||
    this->zone_config_[i].x2 = zone_parameters[i * 4 + 2];
 | 
			
		||||
@@ -322,15 +334,15 @@ void LD2450Component::send_set_zone_command_() {
 | 
			
		||||
  uint8_t cmd_value[26] = {};
 | 
			
		||||
  uint8_t zone_type_bytes[2] = {static_cast<uint8_t>(this->zone_type_), 0x00};
 | 
			
		||||
  uint8_t area_config[24] = {};
 | 
			
		||||
  for (int i = 0; i < MAX_ZONES; i++) {
 | 
			
		||||
  for (uint8_t i = 0; i < MAX_ZONES; i++) {
 | 
			
		||||
    int values[4] = {this->zone_config_[i].x1, this->zone_config_[i].y1, this->zone_config_[i].x2,
 | 
			
		||||
                     this->zone_config_[i].y2};
 | 
			
		||||
    ld2450::convert_int_values_to_hex(values, area_config + (i * 8));
 | 
			
		||||
  }
 | 
			
		||||
  std::memcpy(cmd_value, zone_type_bytes, 2);
 | 
			
		||||
  std::memcpy(cmd_value + 2, area_config, 24);
 | 
			
		||||
  std::memcpy(cmd_value, zone_type_bytes, sizeof(zone_type_bytes));
 | 
			
		||||
  std::memcpy(cmd_value + 2, area_config, sizeof(area_config));
 | 
			
		||||
  this->set_config_mode_(true);
 | 
			
		||||
  this->send_command_(CMD_SET_ZONE, cmd_value, 26);
 | 
			
		||||
  this->send_command_(CMD_SET_ZONE, cmd_value, sizeof(cmd_value));
 | 
			
		||||
  this->set_config_mode_(false);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -346,14 +358,14 @@ bool LD2450Component::get_timeout_status_(uint32_t check_millis) {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Extract, store and publish zone details LD2450 buffer
 | 
			
		||||
void LD2450Component::process_zone_(uint8_t *buffer) {
 | 
			
		||||
void LD2450Component::process_zone_() {
 | 
			
		||||
  uint8_t index, start;
 | 
			
		||||
  for (index = 0; index < MAX_ZONES; index++) {
 | 
			
		||||
    start = 12 + index * 8;
 | 
			
		||||
    this->zone_config_[index].x1 = ld2450::hex_to_signed_int(buffer, start);
 | 
			
		||||
    this->zone_config_[index].y1 = ld2450::hex_to_signed_int(buffer, start + 2);
 | 
			
		||||
    this->zone_config_[index].x2 = ld2450::hex_to_signed_int(buffer, start + 4);
 | 
			
		||||
    this->zone_config_[index].y2 = ld2450::hex_to_signed_int(buffer, start + 6);
 | 
			
		||||
    this->zone_config_[index].x1 = ld2450::hex_to_signed_int(this->buffer_data_, start);
 | 
			
		||||
    this->zone_config_[index].y1 = ld2450::hex_to_signed_int(this->buffer_data_, start + 2);
 | 
			
		||||
    this->zone_config_[index].x2 = ld2450::hex_to_signed_int(this->buffer_data_, start + 4);
 | 
			
		||||
    this->zone_config_[index].y2 = ld2450::hex_to_signed_int(this->buffer_data_, start + 6);
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
    // only one null check as all coordinates are required for a single zone
 | 
			
		||||
    if (this->zone_numbers_[index].x1 != nullptr) {
 | 
			
		||||
@@ -399,27 +411,25 @@ void LD2450Component::restart_and_read_all_info() {
 | 
			
		||||
 | 
			
		||||
// Send command with values to LD2450
 | 
			
		||||
void LD2450Component::send_command_(uint8_t command, const uint8_t *command_value, uint8_t command_value_len) {
 | 
			
		||||
  ESP_LOGV(TAG, "Sending command %02X", command);
 | 
			
		||||
  // frame header
 | 
			
		||||
  this->write_array(CMD_FRAME_HEADER, 4);
 | 
			
		||||
  ESP_LOGV(TAG, "Sending COMMAND %02X", command);
 | 
			
		||||
  // frame header bytes
 | 
			
		||||
  this->write_array(CMD_FRAME_HEADER, sizeof(CMD_FRAME_HEADER));
 | 
			
		||||
  // length bytes
 | 
			
		||||
  int len = 2;
 | 
			
		||||
  uint8_t len = 2;
 | 
			
		||||
  if (command_value != nullptr) {
 | 
			
		||||
    len += command_value_len;
 | 
			
		||||
  }
 | 
			
		||||
  this->write_byte(lowbyte(len));
 | 
			
		||||
  this->write_byte(highbyte(len));
 | 
			
		||||
  // command
 | 
			
		||||
  this->write_byte(lowbyte(command));
 | 
			
		||||
  this->write_byte(highbyte(command));
 | 
			
		||||
  uint8_t len_cmd[] = {lowbyte(len), highbyte(len), command, 0x00};
 | 
			
		||||
  this->write_array(len_cmd, sizeof(len_cmd));
 | 
			
		||||
 | 
			
		||||
  // command value bytes
 | 
			
		||||
  if (command_value != nullptr) {
 | 
			
		||||
    for (int i = 0; i < command_value_len; i++) {
 | 
			
		||||
    for (uint8_t i = 0; i < command_value_len; i++) {
 | 
			
		||||
      this->write_byte(command_value[i]);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  // footer
 | 
			
		||||
  this->write_array(CMD_FRAME_END, 4);
 | 
			
		||||
  // frame footer bytes
 | 
			
		||||
  this->write_array(CMD_FRAME_FOOTER, sizeof(CMD_FRAME_FOOTER));
 | 
			
		||||
  // FIXME to remove
 | 
			
		||||
  delay(50);  // NOLINT
 | 
			
		||||
}
 | 
			
		||||
@@ -427,25 +437,23 @@ void LD2450Component::send_command_(uint8_t command, const uint8_t *command_valu
 | 
			
		||||
// LD2450 Radar data message:
 | 
			
		||||
//  [AA FF 03 00] [0E 03 B1 86 10 00 40 01] [00 00 00 00 00 00 00 00] [00 00 00 00 00 00 00 00] [55 CC]
 | 
			
		||||
//   Header       Target 1                  Target 2                  Target 3                  End
 | 
			
		||||
void LD2450Component::handle_periodic_data_(uint8_t *buffer, uint8_t len) {
 | 
			
		||||
  if (len < 29) {  // header (4 bytes) + 8 x 3 target data + footer (2 bytes)
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid message length");
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  if (buffer[0] != 0xAA || buffer[1] != 0xFF || buffer[2] != 0x03 || buffer[3] != 0x00) {  // header
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid message header");
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  if (buffer[len - 2] != 0x55 || buffer[len - 1] != 0xCC) {  // footer
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid message footer");
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
void LD2450Component::handle_periodic_data_() {
 | 
			
		||||
  // Early throttle check - moved before any processing to save CPU cycles
 | 
			
		||||
  if (App.get_loop_component_start_time() - this->last_periodic_millis_ < this->throttle_) {
 | 
			
		||||
    ESP_LOGV(TAG, "Throttling: %d", this->throttle_);
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (this->buffer_pos_ < 29) {  // header (4 bytes) + 8 x 3 target data + footer (2 bytes)
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid length");
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  if (!ld2450::validate_header_footer(DATA_FRAME_HEADER, this->buffer_data_) ||
 | 
			
		||||
      this->buffer_data_[this->buffer_pos_ - 2] != DATA_FRAME_FOOTER[0] ||
 | 
			
		||||
      this->buffer_data_[this->buffer_pos_ - 1] != DATA_FRAME_FOOTER[1]) {
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid header/footer");
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  // Save the timestamp after validating the frame so, if invalid, we'll take the next frame immediately
 | 
			
		||||
  this->last_periodic_millis_ = App.get_loop_component_start_time();
 | 
			
		||||
 | 
			
		||||
  int16_t target_count = 0;
 | 
			
		||||
@@ -453,13 +461,13 @@ void LD2450Component::handle_periodic_data_(uint8_t *buffer, uint8_t len) {
 | 
			
		||||
  int16_t moving_target_count = 0;
 | 
			
		||||
  int16_t start = 0;
 | 
			
		||||
  int16_t val = 0;
 | 
			
		||||
  uint8_t index = 0;
 | 
			
		||||
  int16_t tx = 0;
 | 
			
		||||
  int16_t ty = 0;
 | 
			
		||||
  int16_t td = 0;
 | 
			
		||||
  int16_t ts = 0;
 | 
			
		||||
  int16_t angle = 0;
 | 
			
		||||
  std::string direction{};
 | 
			
		||||
  uint8_t index = 0;
 | 
			
		||||
  Direction direction{DIRECTION_UNDEFINED};
 | 
			
		||||
  bool is_moving = false;
 | 
			
		||||
 | 
			
		||||
#if defined(USE_BINARY_SENSOR) || defined(USE_SENSOR) || defined(USE_TEXT_SENSOR)
 | 
			
		||||
@@ -471,29 +479,38 @@ void LD2450Component::handle_periodic_data_(uint8_t *buffer, uint8_t len) {
 | 
			
		||||
    is_moving = false;
 | 
			
		||||
    sensor::Sensor *sx = this->move_x_sensors_[index];
 | 
			
		||||
    if (sx != nullptr) {
 | 
			
		||||
      val = ld2450::decode_coordinate(buffer[start], buffer[start + 1]);
 | 
			
		||||
      val = ld2450::decode_coordinate(this->buffer_data_[start], this->buffer_data_[start + 1]);
 | 
			
		||||
      tx = val;
 | 
			
		||||
      sx->publish_state(val);
 | 
			
		||||
      if (this->cached_target_data_[index].x != val) {
 | 
			
		||||
        sx->publish_state(val);
 | 
			
		||||
        this->cached_target_data_[index].x = val;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    // Y
 | 
			
		||||
    start = TARGET_Y + index * 8;
 | 
			
		||||
    sensor::Sensor *sy = this->move_y_sensors_[index];
 | 
			
		||||
    if (sy != nullptr) {
 | 
			
		||||
      val = ld2450::decode_coordinate(buffer[start], buffer[start + 1]);
 | 
			
		||||
      val = ld2450::decode_coordinate(this->buffer_data_[start], this->buffer_data_[start + 1]);
 | 
			
		||||
      ty = val;
 | 
			
		||||
      sy->publish_state(val);
 | 
			
		||||
      if (this->cached_target_data_[index].y != val) {
 | 
			
		||||
        sy->publish_state(val);
 | 
			
		||||
        this->cached_target_data_[index].y = val;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    // RESOLUTION
 | 
			
		||||
    start = TARGET_RESOLUTION + index * 8;
 | 
			
		||||
    sensor::Sensor *sr = this->move_resolution_sensors_[index];
 | 
			
		||||
    if (sr != nullptr) {
 | 
			
		||||
      val = (buffer[start + 1] << 8) | buffer[start];
 | 
			
		||||
      sr->publish_state(val);
 | 
			
		||||
      val = (this->buffer_data_[start + 1] << 8) | this->buffer_data_[start];
 | 
			
		||||
      if (this->cached_target_data_[index].resolution != val) {
 | 
			
		||||
        sr->publish_state(val);
 | 
			
		||||
        this->cached_target_data_[index].resolution = val;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
#endif
 | 
			
		||||
    // SPEED
 | 
			
		||||
    start = TARGET_SPEED + index * 8;
 | 
			
		||||
    val = ld2450::decode_speed(buffer[start], buffer[start + 1]);
 | 
			
		||||
    val = ld2450::decode_speed(this->buffer_data_[start], this->buffer_data_[start + 1]);
 | 
			
		||||
    ts = val;
 | 
			
		||||
    if (val) {
 | 
			
		||||
      is_moving = true;
 | 
			
		||||
@@ -502,13 +519,17 @@ void LD2450Component::handle_periodic_data_(uint8_t *buffer, uint8_t len) {
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
    sensor::Sensor *ss = this->move_speed_sensors_[index];
 | 
			
		||||
    if (ss != nullptr) {
 | 
			
		||||
      ss->publish_state(val);
 | 
			
		||||
      if (this->cached_target_data_[index].speed != val) {
 | 
			
		||||
        ss->publish_state(val);
 | 
			
		||||
        this->cached_target_data_[index].speed = val;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
#endif
 | 
			
		||||
    // DISTANCE
 | 
			
		||||
    val = (uint16_t) sqrt(
 | 
			
		||||
        pow(ld2450::decode_coordinate(buffer[TARGET_X + index * 8], buffer[(TARGET_X + index * 8) + 1]), 2) +
 | 
			
		||||
        pow(ld2450::decode_coordinate(buffer[TARGET_Y + index * 8], buffer[(TARGET_Y + index * 8) + 1]), 2));
 | 
			
		||||
    // Optimized: use already decoded tx and ty values, replace pow() with multiplication
 | 
			
		||||
    int32_t x_squared = (int32_t) tx * tx;
 | 
			
		||||
    int32_t y_squared = (int32_t) ty * ty;
 | 
			
		||||
    val = (uint16_t) sqrt(x_squared + y_squared);
 | 
			
		||||
    td = val;
 | 
			
		||||
    if (val > 0) {
 | 
			
		||||
      target_count++;
 | 
			
		||||
@@ -516,27 +537,42 @@ void LD2450Component::handle_periodic_data_(uint8_t *buffer, uint8_t len) {
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
    sensor::Sensor *sd = this->move_distance_sensors_[index];
 | 
			
		||||
    if (sd != nullptr) {
 | 
			
		||||
      sd->publish_state(val);
 | 
			
		||||
      if (this->cached_target_data_[index].distance != val) {
 | 
			
		||||
        sd->publish_state(val);
 | 
			
		||||
        this->cached_target_data_[index].distance = val;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    // ANGLE
 | 
			
		||||
    angle = calculate_angle(static_cast<float>(ty), static_cast<float>(td));
 | 
			
		||||
    angle = ld2450::calculate_angle(static_cast<float>(ty), static_cast<float>(td));
 | 
			
		||||
    if (tx > 0) {
 | 
			
		||||
      angle = angle * -1;
 | 
			
		||||
    }
 | 
			
		||||
    sensor::Sensor *sa = this->move_angle_sensors_[index];
 | 
			
		||||
    if (sa != nullptr) {
 | 
			
		||||
      sa->publish_state(angle);
 | 
			
		||||
      if (std::isnan(this->cached_target_data_[index].angle) ||
 | 
			
		||||
          std::abs(this->cached_target_data_[index].angle - angle) > 0.1f) {
 | 
			
		||||
        sa->publish_state(angle);
 | 
			
		||||
        this->cached_target_data_[index].angle = angle;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
    // DIRECTION
 | 
			
		||||
    direction = get_direction(ts);
 | 
			
		||||
    if (td == 0) {
 | 
			
		||||
      direction = "NA";
 | 
			
		||||
      direction = DIRECTION_NA;
 | 
			
		||||
    } else if (ts > 0) {
 | 
			
		||||
      direction = DIRECTION_MOVING_AWAY;
 | 
			
		||||
    } else if (ts < 0) {
 | 
			
		||||
      direction = DIRECTION_APPROACHING;
 | 
			
		||||
    } else {
 | 
			
		||||
      direction = DIRECTION_STATIONARY;
 | 
			
		||||
    }
 | 
			
		||||
    text_sensor::TextSensor *tsd = this->direction_text_sensors_[index];
 | 
			
		||||
    if (tsd != nullptr) {
 | 
			
		||||
      tsd->publish_state(direction);
 | 
			
		||||
      if (this->cached_target_data_[index].direction != direction) {
 | 
			
		||||
        tsd->publish_state(find_str(ld2450::DIRECTION_BY_UINT, direction));
 | 
			
		||||
        this->cached_target_data_[index].direction = direction;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
@@ -563,32 +599,50 @@ void LD2450Component::handle_periodic_data_(uint8_t *buffer, uint8_t len) {
 | 
			
		||||
    // Publish Still Target Count in Zones
 | 
			
		||||
    sensor::Sensor *szstc = this->zone_still_target_count_sensors_[index];
 | 
			
		||||
    if (szstc != nullptr) {
 | 
			
		||||
      szstc->publish_state(zone_still_targets);
 | 
			
		||||
      if (this->cached_zone_data_[index].still_count != zone_still_targets) {
 | 
			
		||||
        szstc->publish_state(zone_still_targets);
 | 
			
		||||
        this->cached_zone_data_[index].still_count = zone_still_targets;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    // Publish Moving Target Count in Zones
 | 
			
		||||
    sensor::Sensor *szmtc = this->zone_moving_target_count_sensors_[index];
 | 
			
		||||
    if (szmtc != nullptr) {
 | 
			
		||||
      szmtc->publish_state(zone_moving_targets);
 | 
			
		||||
      if (this->cached_zone_data_[index].moving_count != zone_moving_targets) {
 | 
			
		||||
        szmtc->publish_state(zone_moving_targets);
 | 
			
		||||
        this->cached_zone_data_[index].moving_count = zone_moving_targets;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    // Publish All Target Count in Zones
 | 
			
		||||
    sensor::Sensor *sztc = this->zone_target_count_sensors_[index];
 | 
			
		||||
    if (sztc != nullptr) {
 | 
			
		||||
      sztc->publish_state(zone_all_targets);
 | 
			
		||||
      if (this->cached_zone_data_[index].total_count != zone_all_targets) {
 | 
			
		||||
        sztc->publish_state(zone_all_targets);
 | 
			
		||||
        this->cached_zone_data_[index].total_count = zone_all_targets;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
  }  // End loop thru zones
 | 
			
		||||
 | 
			
		||||
  // Target Count
 | 
			
		||||
  if (this->target_count_sensor_ != nullptr) {
 | 
			
		||||
    this->target_count_sensor_->publish_state(target_count);
 | 
			
		||||
    if (this->cached_global_data_.target_count != target_count) {
 | 
			
		||||
      this->target_count_sensor_->publish_state(target_count);
 | 
			
		||||
      this->cached_global_data_.target_count = target_count;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  // Still Target Count
 | 
			
		||||
  if (this->still_target_count_sensor_ != nullptr) {
 | 
			
		||||
    this->still_target_count_sensor_->publish_state(still_target_count);
 | 
			
		||||
    if (this->cached_global_data_.still_count != still_target_count) {
 | 
			
		||||
      this->still_target_count_sensor_->publish_state(still_target_count);
 | 
			
		||||
      this->cached_global_data_.still_count = still_target_count;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  // Moving Target Count
 | 
			
		||||
  if (this->moving_target_count_sensor_ != nullptr) {
 | 
			
		||||
    this->moving_target_count_sensor_->publish_state(moving_target_count);
 | 
			
		||||
    if (this->cached_global_data_.moving_count != moving_target_count) {
 | 
			
		||||
      this->moving_target_count_sensor_->publish_state(moving_target_count);
 | 
			
		||||
      this->cached_global_data_.moving_count = moving_target_count;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
@@ -640,117 +694,139 @@ void LD2450Component::handle_periodic_data_(uint8_t *buffer, uint8_t len) {
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool LD2450Component::handle_ack_data_(uint8_t *buffer, uint8_t len) {
 | 
			
		||||
  ESP_LOGV(TAG, "Handling ack data for command %02X", buffer[COMMAND]);
 | 
			
		||||
  if (len < 10) {
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid ack length");
 | 
			
		||||
bool LD2450Component::handle_ack_data_() {
 | 
			
		||||
  ESP_LOGV(TAG, "Handling ACK DATA for COMMAND %02X", this->buffer_data_[COMMAND]);
 | 
			
		||||
  if (this->buffer_pos_ < 10) {
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid length");
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  if (buffer[0] != 0xFD || buffer[1] != 0xFC || buffer[2] != 0xFB || buffer[3] != 0xFA) {  // frame header
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid ack header (command %02X)", buffer[COMMAND]);
 | 
			
		||||
  if (!ld2450::validate_header_footer(CMD_FRAME_HEADER, this->buffer_data_)) {
 | 
			
		||||
    ESP_LOGW(TAG, "Invalid header: %s", format_hex_pretty(this->buffer_data_, HEADER_FOOTER_SIZE).c_str());
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  if (buffer[COMMAND_STATUS] != 0x01) {
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid ack status");
 | 
			
		||||
  if (this->buffer_data_[COMMAND_STATUS] != 0x01) {
 | 
			
		||||
    ESP_LOGE(TAG, "Invalid status");
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  if (buffer[8] || buffer[9]) {
 | 
			
		||||
    ESP_LOGE(TAG, "Last buffer was %u, %u", buffer[8], buffer[9]);
 | 
			
		||||
  if (this->buffer_data_[8] || this->buffer_data_[9]) {
 | 
			
		||||
    ESP_LOGW(TAG, "Invalid command: %02X, %02X", this->buffer_data_[8], this->buffer_data_[9]);
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  switch (buffer[COMMAND]) {
 | 
			
		||||
    case lowbyte(CMD_ENABLE_CONF):
 | 
			
		||||
      ESP_LOGV(TAG, "Enable conf command");
 | 
			
		||||
  switch (this->buffer_data_[COMMAND]) {
 | 
			
		||||
    case CMD_ENABLE_CONF:
 | 
			
		||||
      ESP_LOGV(TAG, "Enable conf");
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_DISABLE_CONF):
 | 
			
		||||
      ESP_LOGV(TAG, "Disable conf command");
 | 
			
		||||
 | 
			
		||||
    case CMD_DISABLE_CONF:
 | 
			
		||||
      ESP_LOGV(TAG, "Disabled conf");
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_SET_BAUD_RATE):
 | 
			
		||||
      ESP_LOGV(TAG, "Baud rate change command");
 | 
			
		||||
 | 
			
		||||
    case CMD_SET_BAUD_RATE:
 | 
			
		||||
      ESP_LOGV(TAG, "Baud rate change");
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
      if (this->baud_rate_select_ != nullptr) {
 | 
			
		||||
        ESP_LOGV(TAG, "Change baud rate to %s", this->baud_rate_select_->state.c_str());
 | 
			
		||||
        ESP_LOGE(TAG, "Change baud rate to %s and reinstall", this->baud_rate_select_->state.c_str());
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_VERSION):
 | 
			
		||||
      this->version_ = str_sprintf(VERSION_FMT, buffer[13], buffer[12], buffer[17], buffer[16], buffer[15], buffer[14]);
 | 
			
		||||
      ESP_LOGV(TAG, "Firmware version: %s", this->version_.c_str());
 | 
			
		||||
 | 
			
		||||
    case CMD_QUERY_VERSION: {
 | 
			
		||||
      std::memcpy(this->version_, &this->buffer_data_[12], sizeof(this->version_));
 | 
			
		||||
      std::string version = str_sprintf(VERSION_FMT, this->version_[1], this->version_[0], this->version_[5],
 | 
			
		||||
                                        this->version_[4], this->version_[3], this->version_[2]);
 | 
			
		||||
      ESP_LOGV(TAG, "Firmware version: %s", version.c_str());
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
      if (this->version_text_sensor_ != nullptr) {
 | 
			
		||||
        this->version_text_sensor_->publish_state(this->version_);
 | 
			
		||||
        this->version_text_sensor_->publish_state(version);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_MAC):
 | 
			
		||||
      if (len < 20) {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    case CMD_QUERY_MAC_ADDRESS: {
 | 
			
		||||
      if (this->buffer_pos_ < 20) {
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
      this->mac_ = format_mac_address_pretty(&buffer[10]);
 | 
			
		||||
      ESP_LOGV(TAG, "MAC address: %s", this->mac_.c_str());
 | 
			
		||||
 | 
			
		||||
      this->bluetooth_on_ = std::memcmp(&this->buffer_data_[10], NO_MAC, sizeof(NO_MAC)) != 0;
 | 
			
		||||
      if (this->bluetooth_on_) {
 | 
			
		||||
        std::memcpy(this->mac_address_, &this->buffer_data_[10], sizeof(this->mac_address_));
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      std::string mac_str =
 | 
			
		||||
          mac_address_is_valid(this->mac_address_) ? format_mac_address_pretty(this->mac_address_) : UNKNOWN_MAC;
 | 
			
		||||
      ESP_LOGV(TAG, "MAC address: %s", mac_str.c_str());
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
      if (this->mac_text_sensor_ != nullptr) {
 | 
			
		||||
        this->mac_text_sensor_->publish_state(this->mac_ == NO_MAC ? UNKNOWN_MAC : this->mac_);
 | 
			
		||||
        this->mac_text_sensor_->publish_state(mac_str);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
      if (this->bluetooth_switch_ != nullptr) {
 | 
			
		||||
        this->bluetooth_switch_->publish_state(this->mac_ != NO_MAC);
 | 
			
		||||
        this->bluetooth_switch_->publish_state(this->bluetooth_on_);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_BLUETOOTH):
 | 
			
		||||
      ESP_LOGV(TAG, "Bluetooth command");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    case CMD_BLUETOOTH:
 | 
			
		||||
      ESP_LOGV(TAG, "Bluetooth");
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_SINGLE_TARGET_MODE):
 | 
			
		||||
      ESP_LOGV(TAG, "Single target conf command");
 | 
			
		||||
 | 
			
		||||
    case CMD_SINGLE_TARGET_MODE:
 | 
			
		||||
      ESP_LOGV(TAG, "Single target conf");
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
      if (this->multi_target_switch_ != nullptr) {
 | 
			
		||||
        this->multi_target_switch_->publish_state(false);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_MULTI_TARGET_MODE):
 | 
			
		||||
      ESP_LOGV(TAG, "Multi target conf command");
 | 
			
		||||
 | 
			
		||||
    case CMD_MULTI_TARGET_MODE:
 | 
			
		||||
      ESP_LOGV(TAG, "Multi target conf");
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
      if (this->multi_target_switch_ != nullptr) {
 | 
			
		||||
        this->multi_target_switch_->publish_state(true);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_QUERY_TARGET_MODE):
 | 
			
		||||
      ESP_LOGV(TAG, "Query target tracking mode command");
 | 
			
		||||
 | 
			
		||||
    case CMD_QUERY_TARGET_MODE:
 | 
			
		||||
      ESP_LOGV(TAG, "Query target tracking mode");
 | 
			
		||||
#ifdef USE_SWITCH
 | 
			
		||||
      if (this->multi_target_switch_ != nullptr) {
 | 
			
		||||
        this->multi_target_switch_->publish_state(buffer[10] == 0x02);
 | 
			
		||||
        this->multi_target_switch_->publish_state(this->buffer_data_[10] == 0x02);
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_QUERY_ZONE):
 | 
			
		||||
      ESP_LOGV(TAG, "Query zone conf command");
 | 
			
		||||
      this->zone_type_ = std::stoi(std::to_string(buffer[10]), nullptr, 16);
 | 
			
		||||
 | 
			
		||||
    case CMD_QUERY_ZONE:
 | 
			
		||||
      ESP_LOGV(TAG, "Query zone conf");
 | 
			
		||||
      this->zone_type_ = std::stoi(std::to_string(this->buffer_data_[10]), nullptr, 16);
 | 
			
		||||
      this->publish_zone_type();
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
      if (this->zone_type_select_ != nullptr) {
 | 
			
		||||
        ESP_LOGV(TAG, "Change zone type to: %s", this->zone_type_select_->state.c_str());
 | 
			
		||||
      }
 | 
			
		||||
#endif
 | 
			
		||||
      if (buffer[10] == 0x00) {
 | 
			
		||||
      if (this->buffer_data_[10] == 0x00) {
 | 
			
		||||
        ESP_LOGV(TAG, "Zone: Disabled");
 | 
			
		||||
      }
 | 
			
		||||
      if (buffer[10] == 0x01) {
 | 
			
		||||
      if (this->buffer_data_[10] == 0x01) {
 | 
			
		||||
        ESP_LOGV(TAG, "Zone: Area detection");
 | 
			
		||||
      }
 | 
			
		||||
      if (buffer[10] == 0x02) {
 | 
			
		||||
      if (this->buffer_data_[10] == 0x02) {
 | 
			
		||||
        ESP_LOGV(TAG, "Zone: Area filter");
 | 
			
		||||
      }
 | 
			
		||||
      this->process_zone_(buffer);
 | 
			
		||||
      this->process_zone_();
 | 
			
		||||
      break;
 | 
			
		||||
    case lowbyte(CMD_SET_ZONE):
 | 
			
		||||
      ESP_LOGV(TAG, "Set zone conf command");
 | 
			
		||||
 | 
			
		||||
    case CMD_SET_ZONE:
 | 
			
		||||
      ESP_LOGV(TAG, "Set zone conf");
 | 
			
		||||
      this->query_zone_info();
 | 
			
		||||
      break;
 | 
			
		||||
 | 
			
		||||
    default:
 | 
			
		||||
      break;
 | 
			
		||||
  }
 | 
			
		||||
@@ -758,55 +834,57 @@ bool LD2450Component::handle_ack_data_(uint8_t *buffer, uint8_t len) {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Read LD2450 buffer data
 | 
			
		||||
void LD2450Component::readline_(int readch, uint8_t *buffer, uint8_t len) {
 | 
			
		||||
void LD2450Component::readline_(int readch) {
 | 
			
		||||
  if (readch < 0) {
 | 
			
		||||
    return;
 | 
			
		||||
    return;  // No data available
 | 
			
		||||
  }
 | 
			
		||||
  if (this->buffer_pos_ < len - 1) {
 | 
			
		||||
    buffer[this->buffer_pos_++] = readch;
 | 
			
		||||
    buffer[this->buffer_pos_] = 0;
 | 
			
		||||
 | 
			
		||||
  if (this->buffer_pos_ < MAX_LINE_LENGTH - 1) {
 | 
			
		||||
    this->buffer_data_[this->buffer_pos_++] = readch;
 | 
			
		||||
    this->buffer_data_[this->buffer_pos_] = 0;
 | 
			
		||||
  } else {
 | 
			
		||||
    // We should never get here, but just in case...
 | 
			
		||||
    ESP_LOGW(TAG, "Max command length exceeded; ignoring");
 | 
			
		||||
    this->buffer_pos_ = 0;
 | 
			
		||||
  }
 | 
			
		||||
  if (this->buffer_pos_ < 4) {
 | 
			
		||||
    return;
 | 
			
		||||
    return;  // Not enough data to process yet
 | 
			
		||||
  }
 | 
			
		||||
  if (buffer[this->buffer_pos_ - 2] == 0x55 && buffer[this->buffer_pos_ - 1] == 0xCC) {
 | 
			
		||||
    ESP_LOGV(TAG, "Handle periodic radar data");
 | 
			
		||||
    this->handle_periodic_data_(buffer, this->buffer_pos_);
 | 
			
		||||
  if (this->buffer_data_[this->buffer_pos_ - 2] == DATA_FRAME_FOOTER[0] &&
 | 
			
		||||
      this->buffer_data_[this->buffer_pos_ - 1] == DATA_FRAME_FOOTER[1]) {
 | 
			
		||||
    ESP_LOGV(TAG, "Handling Periodic Data: %s", format_hex_pretty(this->buffer_data_, this->buffer_pos_).c_str());
 | 
			
		||||
    this->handle_periodic_data_();
 | 
			
		||||
    this->buffer_pos_ = 0;  // Reset position index for next frame
 | 
			
		||||
  } else if (buffer[this->buffer_pos_ - 4] == 0x04 && buffer[this->buffer_pos_ - 3] == 0x03 &&
 | 
			
		||||
             buffer[this->buffer_pos_ - 2] == 0x02 && buffer[this->buffer_pos_ - 1] == 0x01) {
 | 
			
		||||
    ESP_LOGV(TAG, "Handle command ack data");
 | 
			
		||||
    if (this->handle_ack_data_(buffer, this->buffer_pos_)) {
 | 
			
		||||
      this->buffer_pos_ = 0;  // Reset position index for next frame
 | 
			
		||||
  } else if (ld2450::validate_header_footer(CMD_FRAME_FOOTER, &this->buffer_data_[this->buffer_pos_ - 4])) {
 | 
			
		||||
    ESP_LOGV(TAG, "Handling Ack Data: %s", format_hex_pretty(this->buffer_data_, this->buffer_pos_).c_str());
 | 
			
		||||
    if (this->handle_ack_data_()) {
 | 
			
		||||
      this->buffer_pos_ = 0;  // Reset position index for next message
 | 
			
		||||
    } else {
 | 
			
		||||
      ESP_LOGV(TAG, "Command ack data invalid");
 | 
			
		||||
      ESP_LOGV(TAG, "Ack Data incomplete");
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Set Config Mode - Pre-requisite sending commands
 | 
			
		||||
void LD2450Component::set_config_mode_(bool enable) {
 | 
			
		||||
  uint8_t cmd = enable ? CMD_ENABLE_CONF : CMD_DISABLE_CONF;
 | 
			
		||||
  uint8_t cmd_value[2] = {0x01, 0x00};
 | 
			
		||||
  this->send_command_(cmd, enable ? cmd_value : nullptr, 2);
 | 
			
		||||
  const uint8_t cmd = enable ? CMD_ENABLE_CONF : CMD_DISABLE_CONF;
 | 
			
		||||
  const uint8_t cmd_value[2] = {0x01, 0x00};
 | 
			
		||||
  this->send_command_(cmd, enable ? cmd_value : nullptr, sizeof(cmd_value));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Set Bluetooth Enable/Disable
 | 
			
		||||
void LD2450Component::set_bluetooth(bool enable) {
 | 
			
		||||
  this->set_config_mode_(true);
 | 
			
		||||
  uint8_t enable_cmd_value[2] = {0x01, 0x00};
 | 
			
		||||
  uint8_t disable_cmd_value[2] = {0x00, 0x00};
 | 
			
		||||
  this->send_command_(CMD_BLUETOOTH, enable ? enable_cmd_value : disable_cmd_value, 2);
 | 
			
		||||
  const uint8_t cmd_value[2] = {enable ? (uint8_t) 0x01 : (uint8_t) 0x00, 0x00};
 | 
			
		||||
  this->send_command_(CMD_BLUETOOTH, cmd_value, sizeof(cmd_value));
 | 
			
		||||
  this->set_timeout(200, [this]() { this->restart_and_read_all_info(); });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Set Baud rate
 | 
			
		||||
void LD2450Component::set_baud_rate(const std::string &state) {
 | 
			
		||||
  this->set_config_mode_(true);
 | 
			
		||||
  uint8_t cmd_value[2] = {find_uint8(BAUD_RATES_BY_STR, state), 0x00};
 | 
			
		||||
  this->send_command_(CMD_SET_BAUD_RATE, cmd_value, 2);
 | 
			
		||||
  const uint8_t cmd_value[2] = {find_uint8(BAUD_RATES_BY_STR, state), 0x00};
 | 
			
		||||
  this->send_command_(CMD_SET_BAUD_RATE, cmd_value, sizeof(cmd_value));
 | 
			
		||||
  this->set_timeout(200, [this]() { this->restart_(); });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -847,12 +925,12 @@ void LD2450Component::factory_reset() {
 | 
			
		||||
void LD2450Component::restart_() { this->send_command_(CMD_RESTART, nullptr, 0); }
 | 
			
		||||
 | 
			
		||||
// Get LD2450 firmware version
 | 
			
		||||
void LD2450Component::get_version_() { this->send_command_(CMD_VERSION, nullptr, 0); }
 | 
			
		||||
void LD2450Component::get_version_() { this->send_command_(CMD_QUERY_VERSION, nullptr, 0); }
 | 
			
		||||
 | 
			
		||||
// Get LD2450 mac address
 | 
			
		||||
void LD2450Component::get_mac_() {
 | 
			
		||||
  uint8_t cmd_value[2] = {0x01, 0x00};
 | 
			
		||||
  this->send_command_(CMD_MAC, cmd_value, 2);
 | 
			
		||||
  this->send_command_(CMD_QUERY_MAC_ADDRESS, cmd_value, 2);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Query for target tracking mode
 | 
			
		||||
 
 | 
			
		||||
@@ -5,6 +5,8 @@
 | 
			
		||||
#include "esphome/core/defines.h"
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
#include "esphome/core/preferences.h"
 | 
			
		||||
#include <limits>
 | 
			
		||||
#include <cmath>
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
#include "esphome/components/sensor/sensor.h"
 | 
			
		||||
#endif
 | 
			
		||||
@@ -36,10 +38,18 @@ namespace ld2450 {
 | 
			
		||||
 | 
			
		||||
// Constants
 | 
			
		||||
static const uint8_t DEFAULT_PRESENCE_TIMEOUT = 5;  // Timeout to reset presense status 5 sec.
 | 
			
		||||
static const uint8_t MAX_LINE_LENGTH = 60;          // Max characters for serial buffer
 | 
			
		||||
static const uint8_t MAX_LINE_LENGTH = 41;          // Max characters for serial buffer
 | 
			
		||||
static const uint8_t MAX_TARGETS = 3;               // Max 3 Targets in LD2450
 | 
			
		||||
static const uint8_t MAX_ZONES = 3;                 // Max 3 Zones in LD2450
 | 
			
		||||
 | 
			
		||||
enum Direction : uint8_t {
 | 
			
		||||
  DIRECTION_APPROACHING = 0,
 | 
			
		||||
  DIRECTION_MOVING_AWAY = 1,
 | 
			
		||||
  DIRECTION_STATIONARY = 2,
 | 
			
		||||
  DIRECTION_NA = 3,
 | 
			
		||||
  DIRECTION_UNDEFINED = 4,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// Target coordinate struct
 | 
			
		||||
struct Target {
 | 
			
		||||
  int16_t x;
 | 
			
		||||
@@ -65,19 +75,22 @@ struct ZoneOfNumbers {
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
class LD2450Component : public Component, public uart::UARTDevice {
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
  SUB_SENSOR(target_count)
 | 
			
		||||
  SUB_SENSOR(still_target_count)
 | 
			
		||||
  SUB_SENSOR(moving_target_count)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_BINARY_SENSOR
 | 
			
		||||
  SUB_BINARY_SENSOR(target)
 | 
			
		||||
  SUB_BINARY_SENSOR(moving_target)
 | 
			
		||||
  SUB_BINARY_SENSOR(still_target)
 | 
			
		||||
  SUB_BINARY_SENSOR(target)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SENSOR
 | 
			
		||||
  SUB_SENSOR(moving_target_count)
 | 
			
		||||
  SUB_SENSOR(still_target_count)
 | 
			
		||||
  SUB_SENSOR(target_count)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_TEXT_SENSOR
 | 
			
		||||
  SUB_TEXT_SENSOR(version)
 | 
			
		||||
  SUB_TEXT_SENSOR(mac)
 | 
			
		||||
  SUB_TEXT_SENSOR(version)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
  SUB_NUMBER(presence_timeout)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_SELECT
 | 
			
		||||
  SUB_SELECT(baud_rate)
 | 
			
		||||
@@ -88,19 +101,16 @@ class LD2450Component : public Component, public uart::UARTDevice {
 | 
			
		||||
  SUB_SWITCH(multi_target)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_BUTTON
 | 
			
		||||
  SUB_BUTTON(reset)
 | 
			
		||||
  SUB_BUTTON(factory_reset)
 | 
			
		||||
  SUB_BUTTON(restart)
 | 
			
		||||
#endif
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
  SUB_NUMBER(presence_timeout)
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
 public:
 | 
			
		||||
  void setup() override;
 | 
			
		||||
  void dump_config() override;
 | 
			
		||||
  void loop() override;
 | 
			
		||||
  void set_presence_timeout();
 | 
			
		||||
  void set_throttle(uint16_t value) { this->throttle_ = value; };
 | 
			
		||||
  void set_throttle(uint16_t value) { this->throttle_ = value; }
 | 
			
		||||
  void read_all_info();
 | 
			
		||||
  void query_zone_info();
 | 
			
		||||
  void restart_and_read_all_info();
 | 
			
		||||
@@ -136,10 +146,10 @@ class LD2450Component : public Component, public uart::UARTDevice {
 | 
			
		||||
 protected:
 | 
			
		||||
  void send_command_(uint8_t command_str, const uint8_t *command_value, uint8_t command_value_len);
 | 
			
		||||
  void set_config_mode_(bool enable);
 | 
			
		||||
  void handle_periodic_data_(uint8_t *buffer, uint8_t len);
 | 
			
		||||
  bool handle_ack_data_(uint8_t *buffer, uint8_t len);
 | 
			
		||||
  void process_zone_(uint8_t *buffer);
 | 
			
		||||
  void readline_(int readch, uint8_t *buffer, uint8_t len);
 | 
			
		||||
  void handle_periodic_data_();
 | 
			
		||||
  bool handle_ack_data_();
 | 
			
		||||
  void process_zone_();
 | 
			
		||||
  void readline_(int readch);
 | 
			
		||||
  void get_version_();
 | 
			
		||||
  void get_mac_();
 | 
			
		||||
  void query_target_tracking_mode_();
 | 
			
		||||
@@ -157,13 +167,40 @@ class LD2450Component : public Component, public uart::UARTDevice {
 | 
			
		||||
  uint32_t moving_presence_millis_ = 0;
 | 
			
		||||
  uint16_t throttle_ = 0;
 | 
			
		||||
  uint16_t timeout_ = 5;
 | 
			
		||||
  uint8_t buffer_pos_ = 0;  // where to resume processing/populating buffer
 | 
			
		||||
  uint8_t buffer_data_[MAX_LINE_LENGTH];
 | 
			
		||||
  uint8_t mac_address_[6] = {0, 0, 0, 0, 0, 0};
 | 
			
		||||
  uint8_t version_[6] = {0, 0, 0, 0, 0, 0};
 | 
			
		||||
  uint8_t buffer_pos_ = 0;  // where to resume processing/populating buffer
 | 
			
		||||
  uint8_t zone_type_ = 0;
 | 
			
		||||
  bool bluetooth_on_{false};
 | 
			
		||||
  Target target_info_[MAX_TARGETS];
 | 
			
		||||
  Zone zone_config_[MAX_ZONES];
 | 
			
		||||
  std::string version_{};
 | 
			
		||||
  std::string mac_{};
 | 
			
		||||
 | 
			
		||||
  // Change detection - cache previous values to avoid redundant publishes
 | 
			
		||||
  // All values are initialized to sentinel values that are outside the valid sensor ranges
 | 
			
		||||
  // to ensure the first real measurement is always published
 | 
			
		||||
  struct CachedTargetData {
 | 
			
		||||
    int16_t x = std::numeric_limits<int16_t>::min();             // -32768, outside range of -4860 to 4860
 | 
			
		||||
    int16_t y = std::numeric_limits<int16_t>::min();             // -32768, outside range of 0 to 7560
 | 
			
		||||
    int16_t speed = std::numeric_limits<int16_t>::min();         // -32768, outside practical sensor range
 | 
			
		||||
    uint16_t resolution = std::numeric_limits<uint16_t>::max();  // 65535, unlikely resolution value
 | 
			
		||||
    uint16_t distance = std::numeric_limits<uint16_t>::max();    // 65535, outside range of 0 to ~8990
 | 
			
		||||
    Direction direction = DIRECTION_UNDEFINED;                   // Undefined, will differ from any real direction
 | 
			
		||||
    float angle = NAN;                                           // NAN, safe sentinel for floats
 | 
			
		||||
  } cached_target_data_[MAX_TARGETS];
 | 
			
		||||
 | 
			
		||||
  struct CachedZoneData {
 | 
			
		||||
    uint8_t still_count = std::numeric_limits<uint8_t>::max();   // 255, unlikely zone count
 | 
			
		||||
    uint8_t moving_count = std::numeric_limits<uint8_t>::max();  // 255, unlikely zone count
 | 
			
		||||
    uint8_t total_count = std::numeric_limits<uint8_t>::max();   // 255, unlikely zone count
 | 
			
		||||
  } cached_zone_data_[MAX_ZONES];
 | 
			
		||||
 | 
			
		||||
  struct CachedGlobalData {
 | 
			
		||||
    uint8_t target_count = std::numeric_limits<uint8_t>::max();  // 255, max 3 targets possible
 | 
			
		||||
    uint8_t still_count = std::numeric_limits<uint8_t>::max();   // 255, max 3 targets possible
 | 
			
		||||
    uint8_t moving_count = std::numeric_limits<uint8_t>::max();  // 255, max 3 targets possible
 | 
			
		||||
  } cached_global_data_;
 | 
			
		||||
 | 
			
		||||
#ifdef USE_NUMBER
 | 
			
		||||
  ESPPreferenceObject pref_;  // only used when numbers are in use
 | 
			
		||||
  ZoneOfNumbers zone_numbers_[MAX_ZONES];
 | 
			
		||||
 
 | 
			
		||||
@@ -268,6 +268,7 @@ async def component_to_code(config):
 | 
			
		||||
 | 
			
		||||
    # disable library compatibility checks
 | 
			
		||||
    cg.add_platformio_option("lib_ldf_mode", "off")
 | 
			
		||||
    cg.add_platformio_option("lib_compat_mode", "soft")
 | 
			
		||||
    # include <Arduino.h> in every file
 | 
			
		||||
    cg.add_platformio_option("build_src_flags", "-include Arduino.h")
 | 
			
		||||
    # dummy version code
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										35
									
								
								esphome/components/libretiny/helpers.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								esphome/components/libretiny/helpers.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,35 @@
 | 
			
		||||
#include "esphome/core/helpers.h"
 | 
			
		||||
 | 
			
		||||
#ifdef USE_LIBRETINY
 | 
			
		||||
 | 
			
		||||
#include "esphome/core/hal.h"
 | 
			
		||||
 | 
			
		||||
#include <WiFi.h>  // for macAddress()
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
 | 
			
		||||
uint32_t random_uint32() { return rand(); }
 | 
			
		||||
 | 
			
		||||
bool random_bytes(uint8_t *data, size_t len) {
 | 
			
		||||
  lt_rand_bytes(data, len);
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Mutex::Mutex() { handle_ = xSemaphoreCreateMutex(); }
 | 
			
		||||
Mutex::~Mutex() {}
 | 
			
		||||
void Mutex::lock() { xSemaphoreTake(this->handle_, portMAX_DELAY); }
 | 
			
		||||
bool Mutex::try_lock() { return xSemaphoreTake(this->handle_, 0) == pdTRUE; }
 | 
			
		||||
void Mutex::unlock() { xSemaphoreGive(this->handle_); }
 | 
			
		||||
 | 
			
		||||
// only affects the executing core
 | 
			
		||||
// so should not be used as a mutex lock, only to get accurate timing
 | 
			
		||||
IRAM_ATTR InterruptLock::InterruptLock() { portDISABLE_INTERRUPTS(); }
 | 
			
		||||
IRAM_ATTR InterruptLock::~InterruptLock() { portENABLE_INTERRUPTS(); }
 | 
			
		||||
 | 
			
		||||
void get_mac_address_raw(uint8_t *mac) {  // NOLINT(readability-non-const-parameter)
 | 
			
		||||
  WiFi.macAddress(mac);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
 | 
			
		||||
#endif  // USE_LIBRETINY
 | 
			
		||||
@@ -97,12 +97,12 @@ class AddressableLight : public LightOutput, public Component {
 | 
			
		||||
  }
 | 
			
		||||
  virtual ESPColorView get_view_internal(int32_t index) const = 0;
 | 
			
		||||
 | 
			
		||||
  bool effect_active_{false};
 | 
			
		||||
  ESPColorCorrection correction_{};
 | 
			
		||||
  LightState *state_parent_{nullptr};
 | 
			
		||||
#ifdef USE_POWER_SUPPLY
 | 
			
		||||
  power_supply::PowerSupplyRequester power_;
 | 
			
		||||
#endif
 | 
			
		||||
  LightState *state_parent_{nullptr};
 | 
			
		||||
  bool effect_active_{false};
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
class AddressableLightTransformer : public LightTransitionTransformer {
 | 
			
		||||
@@ -114,9 +114,9 @@ class AddressableLightTransformer : public LightTransitionTransformer {
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  AddressableLight &light_;
 | 
			
		||||
  Color target_color_{};
 | 
			
		||||
  float last_transition_progress_{0.0f};
 | 
			
		||||
  float accumulated_alpha_{0.0f};
 | 
			
		||||
  Color target_color_{};
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace light
 | 
			
		||||
 
 | 
			
		||||
@@ -69,8 +69,8 @@ class ESPColorCorrection {
 | 
			
		||||
 protected:
 | 
			
		||||
  uint8_t gamma_table_[256];
 | 
			
		||||
  uint8_t gamma_reverse_table_[256];
 | 
			
		||||
  uint8_t local_brightness_{255};
 | 
			
		||||
  Color max_brightness_;
 | 
			
		||||
  uint8_t local_brightness_{255};
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace light
 | 
			
		||||
 
 | 
			
		||||
@@ -2,12 +2,28 @@
 | 
			
		||||
#include "light_call.h"
 | 
			
		||||
#include "light_state.h"
 | 
			
		||||
#include "esphome/core/log.h"
 | 
			
		||||
#include "esphome/core/optional.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace light {
 | 
			
		||||
 | 
			
		||||
static const char *const TAG = "light";
 | 
			
		||||
 | 
			
		||||
// Macro to reduce repetitive setter code
 | 
			
		||||
#define IMPLEMENT_LIGHT_CALL_SETTER(name, type, flag) \
 | 
			
		||||
  LightCall &LightCall::set_##name(optional<type>(name)) { \
 | 
			
		||||
    if ((name).has_value()) { \
 | 
			
		||||
      this->name##_ = (name).value(); \
 | 
			
		||||
    } \
 | 
			
		||||
    this->set_flag_(flag, (name).has_value()); \
 | 
			
		||||
    return *this; \
 | 
			
		||||
  } \
 | 
			
		||||
  LightCall &LightCall::set_##name(type name) { \
 | 
			
		||||
    this->name##_ = name; \
 | 
			
		||||
    this->set_flag_(flag, true); \
 | 
			
		||||
    return *this; \
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
static const LogString *color_mode_to_human(ColorMode color_mode) {
 | 
			
		||||
  if (color_mode == ColorMode::UNKNOWN)
 | 
			
		||||
    return LOG_STR("Unknown");
 | 
			
		||||
@@ -32,41 +48,43 @@ void LightCall::perform() {
 | 
			
		||||
  const char *name = this->parent_->get_name().c_str();
 | 
			
		||||
  LightColorValues v = this->validate_();
 | 
			
		||||
 | 
			
		||||
  if (this->publish_) {
 | 
			
		||||
  if (this->get_publish_()) {
 | 
			
		||||
    ESP_LOGD(TAG, "'%s' Setting:", name);
 | 
			
		||||
 | 
			
		||||
    // Only print color mode when it's being changed
 | 
			
		||||
    ColorMode current_color_mode = this->parent_->remote_values.get_color_mode();
 | 
			
		||||
    if (this->color_mode_.value_or(current_color_mode) != current_color_mode) {
 | 
			
		||||
    ColorMode target_color_mode = this->has_color_mode() ? this->color_mode_ : current_color_mode;
 | 
			
		||||
    if (target_color_mode != current_color_mode) {
 | 
			
		||||
      ESP_LOGD(TAG, "  Color mode: %s", LOG_STR_ARG(color_mode_to_human(v.get_color_mode())));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Only print state when it's being changed
 | 
			
		||||
    bool current_state = this->parent_->remote_values.is_on();
 | 
			
		||||
    if (this->state_.value_or(current_state) != current_state) {
 | 
			
		||||
    bool target_state = this->has_state() ? this->state_ : current_state;
 | 
			
		||||
    if (target_state != current_state) {
 | 
			
		||||
      ESP_LOGD(TAG, "  State: %s", ONOFF(v.is_on()));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (this->brightness_.has_value()) {
 | 
			
		||||
    if (this->has_brightness()) {
 | 
			
		||||
      ESP_LOGD(TAG, "  Brightness: %.0f%%", v.get_brightness() * 100.0f);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (this->color_brightness_.has_value()) {
 | 
			
		||||
    if (this->has_color_brightness()) {
 | 
			
		||||
      ESP_LOGD(TAG, "  Color brightness: %.0f%%", v.get_color_brightness() * 100.0f);
 | 
			
		||||
    }
 | 
			
		||||
    if (this->red_.has_value() || this->green_.has_value() || this->blue_.has_value()) {
 | 
			
		||||
    if (this->has_red() || this->has_green() || this->has_blue()) {
 | 
			
		||||
      ESP_LOGD(TAG, "  Red: %.0f%%, Green: %.0f%%, Blue: %.0f%%", v.get_red() * 100.0f, v.get_green() * 100.0f,
 | 
			
		||||
               v.get_blue() * 100.0f);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (this->white_.has_value()) {
 | 
			
		||||
    if (this->has_white()) {
 | 
			
		||||
      ESP_LOGD(TAG, "  White: %.0f%%", v.get_white() * 100.0f);
 | 
			
		||||
    }
 | 
			
		||||
    if (this->color_temperature_.has_value()) {
 | 
			
		||||
    if (this->has_color_temperature()) {
 | 
			
		||||
      ESP_LOGD(TAG, "  Color temperature: %.1f mireds", v.get_color_temperature());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (this->cold_white_.has_value() || this->warm_white_.has_value()) {
 | 
			
		||||
    if (this->has_cold_white() || this->has_warm_white()) {
 | 
			
		||||
      ESP_LOGD(TAG, "  Cold white: %.0f%%, warm white: %.0f%%", v.get_cold_white() * 100.0f,
 | 
			
		||||
               v.get_warm_white() * 100.0f);
 | 
			
		||||
    }
 | 
			
		||||
@@ -74,58 +92,57 @@ void LightCall::perform() {
 | 
			
		||||
 | 
			
		||||
  if (this->has_flash_()) {
 | 
			
		||||
    // FLASH
 | 
			
		||||
    if (this->publish_) {
 | 
			
		||||
      ESP_LOGD(TAG, "  Flash length: %.1fs", *this->flash_length_ / 1e3f);
 | 
			
		||||
    if (this->get_publish_()) {
 | 
			
		||||
      ESP_LOGD(TAG, "  Flash length: %.1fs", this->flash_length_ / 1e3f);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this->parent_->start_flash_(v, *this->flash_length_, this->publish_);
 | 
			
		||||
    this->parent_->start_flash_(v, this->flash_length_, this->get_publish_());
 | 
			
		||||
  } else if (this->has_transition_()) {
 | 
			
		||||
    // TRANSITION
 | 
			
		||||
    if (this->publish_) {
 | 
			
		||||
      ESP_LOGD(TAG, "  Transition length: %.1fs", *this->transition_length_ / 1e3f);
 | 
			
		||||
    if (this->get_publish_()) {
 | 
			
		||||
      ESP_LOGD(TAG, "  Transition length: %.1fs", this->transition_length_ / 1e3f);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Special case: Transition and effect can be set when turning off
 | 
			
		||||
    if (this->has_effect_()) {
 | 
			
		||||
      if (this->publish_) {
 | 
			
		||||
      if (this->get_publish_()) {
 | 
			
		||||
        ESP_LOGD(TAG, "  Effect: 'None'");
 | 
			
		||||
      }
 | 
			
		||||
      this->parent_->stop_effect_();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this->parent_->start_transition_(v, *this->transition_length_, this->publish_);
 | 
			
		||||
    this->parent_->start_transition_(v, this->transition_length_, this->get_publish_());
 | 
			
		||||
 | 
			
		||||
  } else if (this->has_effect_()) {
 | 
			
		||||
    // EFFECT
 | 
			
		||||
    auto effect = this->effect_;
 | 
			
		||||
    const char *effect_s;
 | 
			
		||||
    if (effect == 0u) {
 | 
			
		||||
    if (this->effect_ == 0u) {
 | 
			
		||||
      effect_s = "None";
 | 
			
		||||
    } else {
 | 
			
		||||
      effect_s = this->parent_->effects_[*this->effect_ - 1]->get_name().c_str();
 | 
			
		||||
      effect_s = this->parent_->effects_[this->effect_ - 1]->get_name().c_str();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (this->publish_) {
 | 
			
		||||
    if (this->get_publish_()) {
 | 
			
		||||
      ESP_LOGD(TAG, "  Effect: '%s'", effect_s);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this->parent_->start_effect_(*this->effect_);
 | 
			
		||||
    this->parent_->start_effect_(this->effect_);
 | 
			
		||||
 | 
			
		||||
    // Also set light color values when starting an effect
 | 
			
		||||
    // For example to turn off the light
 | 
			
		||||
    this->parent_->set_immediately_(v, true);
 | 
			
		||||
  } else {
 | 
			
		||||
    // INSTANT CHANGE
 | 
			
		||||
    this->parent_->set_immediately_(v, this->publish_);
 | 
			
		||||
    this->parent_->set_immediately_(v, this->get_publish_());
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (!this->has_transition_()) {
 | 
			
		||||
    this->parent_->target_state_reached_callback_.call();
 | 
			
		||||
  }
 | 
			
		||||
  if (this->publish_) {
 | 
			
		||||
  if (this->get_publish_()) {
 | 
			
		||||
    this->parent_->publish_state();
 | 
			
		||||
  }
 | 
			
		||||
  if (this->save_) {
 | 
			
		||||
  if (this->get_save_()) {
 | 
			
		||||
    this->parent_->save_remote_values_();
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -135,82 +152,80 @@ LightColorValues LightCall::validate_() {
 | 
			
		||||
  auto traits = this->parent_->get_traits();
 | 
			
		||||
 | 
			
		||||
  // Color mode check
 | 
			
		||||
  if (this->color_mode_.has_value() && !traits.supports_color_mode(this->color_mode_.value())) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s' does not support color mode %s", name,
 | 
			
		||||
             LOG_STR_ARG(color_mode_to_human(this->color_mode_.value())));
 | 
			
		||||
    this->color_mode_.reset();
 | 
			
		||||
  if (this->has_color_mode() && !traits.supports_color_mode(this->color_mode_)) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s' does not support color mode %s", name, LOG_STR_ARG(color_mode_to_human(this->color_mode_)));
 | 
			
		||||
    this->set_flag_(FLAG_HAS_COLOR_MODE, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Ensure there is always a color mode set
 | 
			
		||||
  if (!this->color_mode_.has_value()) {
 | 
			
		||||
  if (!this->has_color_mode()) {
 | 
			
		||||
    this->color_mode_ = this->compute_color_mode_();
 | 
			
		||||
    this->set_flag_(FLAG_HAS_COLOR_MODE, true);
 | 
			
		||||
  }
 | 
			
		||||
  auto color_mode = *this->color_mode_;
 | 
			
		||||
  auto color_mode = this->color_mode_;
 | 
			
		||||
 | 
			
		||||
  // Transform calls that use non-native parameters for the current mode.
 | 
			
		||||
  this->transform_parameters_();
 | 
			
		||||
 | 
			
		||||
  // Brightness exists check
 | 
			
		||||
  if (this->brightness_.has_value() && *this->brightness_ > 0.0f && !(color_mode & ColorCapability::BRIGHTNESS)) {
 | 
			
		||||
  if (this->has_brightness() && this->brightness_ > 0.0f && !(color_mode & ColorCapability::BRIGHTNESS)) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s': setting brightness not supported", name);
 | 
			
		||||
    this->brightness_.reset();
 | 
			
		||||
    this->set_flag_(FLAG_HAS_BRIGHTNESS, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Transition length possible check
 | 
			
		||||
  if (this->transition_length_.has_value() && *this->transition_length_ != 0 &&
 | 
			
		||||
      !(color_mode & ColorCapability::BRIGHTNESS)) {
 | 
			
		||||
  if (this->has_transition_() && this->transition_length_ != 0 && !(color_mode & ColorCapability::BRIGHTNESS)) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s': transitions not supported", name);
 | 
			
		||||
    this->transition_length_.reset();
 | 
			
		||||
    this->set_flag_(FLAG_HAS_TRANSITION, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Color brightness exists check
 | 
			
		||||
  if (this->color_brightness_.has_value() && *this->color_brightness_ > 0.0f && !(color_mode & ColorCapability::RGB)) {
 | 
			
		||||
  if (this->has_color_brightness() && this->color_brightness_ > 0.0f && !(color_mode & ColorCapability::RGB)) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s': color mode does not support setting RGB brightness", name);
 | 
			
		||||
    this->color_brightness_.reset();
 | 
			
		||||
    this->set_flag_(FLAG_HAS_COLOR_BRIGHTNESS, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // RGB exists check
 | 
			
		||||
  if ((this->red_.has_value() && *this->red_ > 0.0f) || (this->green_.has_value() && *this->green_ > 0.0f) ||
 | 
			
		||||
      (this->blue_.has_value() && *this->blue_ > 0.0f)) {
 | 
			
		||||
  if ((this->has_red() && this->red_ > 0.0f) || (this->has_green() && this->green_ > 0.0f) ||
 | 
			
		||||
      (this->has_blue() && this->blue_ > 0.0f)) {
 | 
			
		||||
    if (!(color_mode & ColorCapability::RGB)) {
 | 
			
		||||
      ESP_LOGW(TAG, "'%s': color mode does not support setting RGB color", name);
 | 
			
		||||
      this->red_.reset();
 | 
			
		||||
      this->green_.reset();
 | 
			
		||||
      this->blue_.reset();
 | 
			
		||||
      this->set_flag_(FLAG_HAS_RED, false);
 | 
			
		||||
      this->set_flag_(FLAG_HAS_GREEN, false);
 | 
			
		||||
      this->set_flag_(FLAG_HAS_BLUE, false);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // White value exists check
 | 
			
		||||
  if (this->white_.has_value() && *this->white_ > 0.0f &&
 | 
			
		||||
  if (this->has_white() && this->white_ > 0.0f &&
 | 
			
		||||
      !(color_mode & ColorCapability::WHITE || color_mode & ColorCapability::COLD_WARM_WHITE)) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s': color mode does not support setting white value", name);
 | 
			
		||||
    this->white_.reset();
 | 
			
		||||
    this->set_flag_(FLAG_HAS_WHITE, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Color temperature exists check
 | 
			
		||||
  if (this->color_temperature_.has_value() &&
 | 
			
		||||
  if (this->has_color_temperature() &&
 | 
			
		||||
      !(color_mode & ColorCapability::COLOR_TEMPERATURE || color_mode & ColorCapability::COLD_WARM_WHITE)) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s': color mode does not support setting color temperature", name);
 | 
			
		||||
    this->color_temperature_.reset();
 | 
			
		||||
    this->set_flag_(FLAG_HAS_COLOR_TEMPERATURE, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Cold/warm white value exists check
 | 
			
		||||
  if ((this->cold_white_.has_value() && *this->cold_white_ > 0.0f) ||
 | 
			
		||||
      (this->warm_white_.has_value() && *this->warm_white_ > 0.0f)) {
 | 
			
		||||
  if ((this->has_cold_white() && this->cold_white_ > 0.0f) || (this->has_warm_white() && this->warm_white_ > 0.0f)) {
 | 
			
		||||
    if (!(color_mode & ColorCapability::COLD_WARM_WHITE)) {
 | 
			
		||||
      ESP_LOGW(TAG, "'%s': color mode does not support setting cold/warm white value", name);
 | 
			
		||||
      this->cold_white_.reset();
 | 
			
		||||
      this->warm_white_.reset();
 | 
			
		||||
      this->set_flag_(FLAG_HAS_COLD_WHITE, false);
 | 
			
		||||
      this->set_flag_(FLAG_HAS_WARM_WHITE, false);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
#define VALIDATE_RANGE_(name_, upper_name, min, max) \
 | 
			
		||||
  if (name_##_.has_value()) { \
 | 
			
		||||
    auto val = *name_##_; \
 | 
			
		||||
  if (this->has_##name_()) { \
 | 
			
		||||
    auto val = this->name_##_; \
 | 
			
		||||
    if (val < (min) || val > (max)) { \
 | 
			
		||||
      ESP_LOGW(TAG, "'%s': %s value %.2f is out of range [%.1f - %.1f]", name, LOG_STR_LITERAL(upper_name), val, \
 | 
			
		||||
               (min), (max)); \
 | 
			
		||||
      name_##_ = clamp(val, (min), (max)); \
 | 
			
		||||
      this->name_##_ = clamp(val, (min), (max)); \
 | 
			
		||||
    } \
 | 
			
		||||
  }
 | 
			
		||||
#define VALIDATE_RANGE(name, upper_name) VALIDATE_RANGE_(name, upper_name, 0.0f, 1.0f)
 | 
			
		||||
@@ -227,110 +242,116 @@ LightColorValues LightCall::validate_() {
 | 
			
		||||
  VALIDATE_RANGE_(color_temperature, "Color temperature", traits.get_min_mireds(), traits.get_max_mireds())
 | 
			
		||||
 | 
			
		||||
  // Flag whether an explicit turn off was requested, in which case we'll also stop the effect.
 | 
			
		||||
  bool explicit_turn_off_request = this->state_.has_value() && !*this->state_;
 | 
			
		||||
  bool explicit_turn_off_request = this->has_state() && !this->state_;
 | 
			
		||||
 | 
			
		||||
  // Turn off when brightness is set to zero, and reset brightness (so that it has nonzero brightness when turned on).
 | 
			
		||||
  if (this->brightness_.has_value() && *this->brightness_ == 0.0f) {
 | 
			
		||||
    this->state_ = optional<float>(false);
 | 
			
		||||
    this->brightness_ = optional<float>(1.0f);
 | 
			
		||||
  if (this->has_brightness() && this->brightness_ == 0.0f) {
 | 
			
		||||
    this->state_ = false;
 | 
			
		||||
    this->set_flag_(FLAG_HAS_STATE, true);
 | 
			
		||||
    this->brightness_ = 1.0f;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Set color brightness to 100% if currently zero and a color is set.
 | 
			
		||||
  if (this->red_.has_value() || this->green_.has_value() || this->blue_.has_value()) {
 | 
			
		||||
    if (!this->color_brightness_.has_value() && this->parent_->remote_values.get_color_brightness() == 0.0f)
 | 
			
		||||
      this->color_brightness_ = optional<float>(1.0f);
 | 
			
		||||
  if (this->has_red() || this->has_green() || this->has_blue()) {
 | 
			
		||||
    if (!this->has_color_brightness() && this->parent_->remote_values.get_color_brightness() == 0.0f) {
 | 
			
		||||
      this->color_brightness_ = 1.0f;
 | 
			
		||||
      this->set_flag_(FLAG_HAS_COLOR_BRIGHTNESS, true);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Create color values for the light with this call applied.
 | 
			
		||||
  auto v = this->parent_->remote_values;
 | 
			
		||||
  if (this->color_mode_.has_value())
 | 
			
		||||
    v.set_color_mode(*this->color_mode_);
 | 
			
		||||
  if (this->state_.has_value())
 | 
			
		||||
    v.set_state(*this->state_);
 | 
			
		||||
  if (this->brightness_.has_value())
 | 
			
		||||
    v.set_brightness(*this->brightness_);
 | 
			
		||||
  if (this->color_brightness_.has_value())
 | 
			
		||||
    v.set_color_brightness(*this->color_brightness_);
 | 
			
		||||
  if (this->red_.has_value())
 | 
			
		||||
    v.set_red(*this->red_);
 | 
			
		||||
  if (this->green_.has_value())
 | 
			
		||||
    v.set_green(*this->green_);
 | 
			
		||||
  if (this->blue_.has_value())
 | 
			
		||||
    v.set_blue(*this->blue_);
 | 
			
		||||
  if (this->white_.has_value())
 | 
			
		||||
    v.set_white(*this->white_);
 | 
			
		||||
  if (this->color_temperature_.has_value())
 | 
			
		||||
    v.set_color_temperature(*this->color_temperature_);
 | 
			
		||||
  if (this->cold_white_.has_value())
 | 
			
		||||
    v.set_cold_white(*this->cold_white_);
 | 
			
		||||
  if (this->warm_white_.has_value())
 | 
			
		||||
    v.set_warm_white(*this->warm_white_);
 | 
			
		||||
  if (this->has_color_mode())
 | 
			
		||||
    v.set_color_mode(this->color_mode_);
 | 
			
		||||
  if (this->has_state())
 | 
			
		||||
    v.set_state(this->state_);
 | 
			
		||||
  if (this->has_brightness())
 | 
			
		||||
    v.set_brightness(this->brightness_);
 | 
			
		||||
  if (this->has_color_brightness())
 | 
			
		||||
    v.set_color_brightness(this->color_brightness_);
 | 
			
		||||
  if (this->has_red())
 | 
			
		||||
    v.set_red(this->red_);
 | 
			
		||||
  if (this->has_green())
 | 
			
		||||
    v.set_green(this->green_);
 | 
			
		||||
  if (this->has_blue())
 | 
			
		||||
    v.set_blue(this->blue_);
 | 
			
		||||
  if (this->has_white())
 | 
			
		||||
    v.set_white(this->white_);
 | 
			
		||||
  if (this->has_color_temperature())
 | 
			
		||||
    v.set_color_temperature(this->color_temperature_);
 | 
			
		||||
  if (this->has_cold_white())
 | 
			
		||||
    v.set_cold_white(this->cold_white_);
 | 
			
		||||
  if (this->has_warm_white())
 | 
			
		||||
    v.set_warm_white(this->warm_white_);
 | 
			
		||||
 | 
			
		||||
  v.normalize_color();
 | 
			
		||||
 | 
			
		||||
  // Flash length check
 | 
			
		||||
  if (this->has_flash_() && *this->flash_length_ == 0) {
 | 
			
		||||
  if (this->has_flash_() && this->flash_length_ == 0) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s': flash length must be greater than zero", name);
 | 
			
		||||
    this->flash_length_.reset();
 | 
			
		||||
    this->set_flag_(FLAG_HAS_FLASH, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // validate transition length/flash length/effect not used at the same time
 | 
			
		||||
  bool supports_transition = color_mode & ColorCapability::BRIGHTNESS;
 | 
			
		||||
 | 
			
		||||
  // If effect is already active, remove effect start
 | 
			
		||||
  if (this->has_effect_() && *this->effect_ == this->parent_->active_effect_index_) {
 | 
			
		||||
    this->effect_.reset();
 | 
			
		||||
  if (this->has_effect_() && this->effect_ == this->parent_->active_effect_index_) {
 | 
			
		||||
    this->set_flag_(FLAG_HAS_EFFECT, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // validate effect index
 | 
			
		||||
  if (this->has_effect_() && *this->effect_ > this->parent_->effects_.size()) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s': invalid effect index %" PRIu32, name, *this->effect_);
 | 
			
		||||
    this->effect_.reset();
 | 
			
		||||
  if (this->has_effect_() && this->effect_ > this->parent_->effects_.size()) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s': invalid effect index %" PRIu32, name, this->effect_);
 | 
			
		||||
    this->set_flag_(FLAG_HAS_EFFECT, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (this->has_effect_() && (this->has_transition_() || this->has_flash_())) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s': effect cannot be used with transition/flash", name);
 | 
			
		||||
    this->transition_length_.reset();
 | 
			
		||||
    this->flash_length_.reset();
 | 
			
		||||
    this->set_flag_(FLAG_HAS_TRANSITION, false);
 | 
			
		||||
    this->set_flag_(FLAG_HAS_FLASH, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (this->has_flash_() && this->has_transition_()) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s': flash cannot be used with transition", name);
 | 
			
		||||
    this->transition_length_.reset();
 | 
			
		||||
    this->set_flag_(FLAG_HAS_TRANSITION, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (!this->has_transition_() && !this->has_flash_() && (!this->has_effect_() || *this->effect_ == 0) &&
 | 
			
		||||
  if (!this->has_transition_() && !this->has_flash_() && (!this->has_effect_() || this->effect_ == 0) &&
 | 
			
		||||
      supports_transition) {
 | 
			
		||||
    // nothing specified and light supports transitions, set default transition length
 | 
			
		||||
    this->transition_length_ = this->parent_->default_transition_length_;
 | 
			
		||||
    this->set_flag_(FLAG_HAS_TRANSITION, true);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (this->transition_length_.value_or(0) == 0) {
 | 
			
		||||
  if (this->has_transition_() && this->transition_length_ == 0) {
 | 
			
		||||
    // 0 transition is interpreted as no transition (instant change)
 | 
			
		||||
    this->transition_length_.reset();
 | 
			
		||||
    this->set_flag_(FLAG_HAS_TRANSITION, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (this->has_transition_() && !supports_transition) {
 | 
			
		||||
    ESP_LOGW(TAG, "'%s': transitions not supported", name);
 | 
			
		||||
    this->transition_length_.reset();
 | 
			
		||||
    this->set_flag_(FLAG_HAS_TRANSITION, false);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // If not a flash and turning the light off, then disable the light
 | 
			
		||||
  // Do not use light color values directly, so that effects can set 0% brightness
 | 
			
		||||
  // Reason: When user turns off the light in frontend, the effect should also stop
 | 
			
		||||
  if (!this->has_flash_() && !this->state_.value_or(v.is_on())) {
 | 
			
		||||
  bool target_state = this->has_state() ? this->state_ : v.is_on();
 | 
			
		||||
  if (!this->has_flash_() && !target_state) {
 | 
			
		||||
    if (this->has_effect_()) {
 | 
			
		||||
      ESP_LOGW(TAG, "'%s': cannot start effect when turning off", name);
 | 
			
		||||
      this->effect_.reset();
 | 
			
		||||
      this->set_flag_(FLAG_HAS_EFFECT, false);
 | 
			
		||||
    } else if (this->parent_->active_effect_index_ != 0 && explicit_turn_off_request) {
 | 
			
		||||
      // Auto turn off effect
 | 
			
		||||
      this->effect_ = 0;
 | 
			
		||||
      this->set_flag_(FLAG_HAS_EFFECT, true);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Disable saving for flashes
 | 
			
		||||
  if (this->has_flash_())
 | 
			
		||||
    this->save_ = false;
 | 
			
		||||
    this->set_flag_(FLAG_SAVE, false);
 | 
			
		||||
 | 
			
		||||
  return v;
 | 
			
		||||
}
 | 
			
		||||
@@ -343,24 +364,27 @@ void LightCall::transform_parameters_() {
 | 
			
		||||
  // - RGBWW lights with color_interlock=true, which also sets "brightness" and
 | 
			
		||||
  //   "color_temperature" (without color_interlock, CW/WW are set directly)
 | 
			
		||||
  // - Legacy Home Assistant (pre-colormode), which sets "white" and "color_temperature"
 | 
			
		||||
  if (((this->white_.has_value() && *this->white_ > 0.0f) || this->color_temperature_.has_value()) &&  //
 | 
			
		||||
      (*this->color_mode_ & ColorCapability::COLD_WARM_WHITE) &&                                       //
 | 
			
		||||
      !(*this->color_mode_ & ColorCapability::WHITE) &&                                                //
 | 
			
		||||
      !(*this->color_mode_ & ColorCapability::COLOR_TEMPERATURE) &&                                    //
 | 
			
		||||
  if (((this->has_white() && this->white_ > 0.0f) || this->has_color_temperature()) &&  //
 | 
			
		||||
      (this->color_mode_ & ColorCapability::COLD_WARM_WHITE) &&                         //
 | 
			
		||||
      !(this->color_mode_ & ColorCapability::WHITE) &&                                  //
 | 
			
		||||
      !(this->color_mode_ & ColorCapability::COLOR_TEMPERATURE) &&                      //
 | 
			
		||||
      traits.get_min_mireds() > 0.0f && traits.get_max_mireds() > 0.0f) {
 | 
			
		||||
    ESP_LOGD(TAG, "'%s': setting cold/warm white channels using white/color temperature values",
 | 
			
		||||
             this->parent_->get_name().c_str());
 | 
			
		||||
    if (this->color_temperature_.has_value()) {
 | 
			
		||||
      const float color_temp = clamp(*this->color_temperature_, traits.get_min_mireds(), traits.get_max_mireds());
 | 
			
		||||
    if (this->has_color_temperature()) {
 | 
			
		||||
      const float color_temp = clamp(this->color_temperature_, traits.get_min_mireds(), traits.get_max_mireds());
 | 
			
		||||
      const float ww_fraction =
 | 
			
		||||
          (color_temp - traits.get_min_mireds()) / (traits.get_max_mireds() - traits.get_min_mireds());
 | 
			
		||||
      const float cw_fraction = 1.0f - ww_fraction;
 | 
			
		||||
      const float max_cw_ww = std::max(ww_fraction, cw_fraction);
 | 
			
		||||
      this->cold_white_ = gamma_uncorrect(cw_fraction / max_cw_ww, this->parent_->get_gamma_correct());
 | 
			
		||||
      this->warm_white_ = gamma_uncorrect(ww_fraction / max_cw_ww, this->parent_->get_gamma_correct());
 | 
			
		||||
      this->set_flag_(FLAG_HAS_COLD_WHITE, true);
 | 
			
		||||
      this->set_flag_(FLAG_HAS_WARM_WHITE, true);
 | 
			
		||||
    }
 | 
			
		||||
    if (this->white_.has_value()) {
 | 
			
		||||
      this->brightness_ = *this->white_;
 | 
			
		||||
    if (this->has_white()) {
 | 
			
		||||
      this->brightness_ = this->white_;
 | 
			
		||||
      this->set_flag_(FLAG_HAS_BRIGHTNESS, true);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -378,7 +402,7 @@ ColorMode LightCall::compute_color_mode_() {
 | 
			
		||||
 | 
			
		||||
  // Don't change if the light is being turned off.
 | 
			
		||||
  ColorMode current_mode = this->parent_->remote_values.get_color_mode();
 | 
			
		||||
  if (this->state_.has_value() && !*this->state_)
 | 
			
		||||
  if (this->has_state() && !this->state_)
 | 
			
		||||
    return current_mode;
 | 
			
		||||
 | 
			
		||||
  // If no color mode is specified, we try to guess the color mode. This is needed for backward compatibility to
 | 
			
		||||
@@ -411,12 +435,12 @@ ColorMode LightCall::compute_color_mode_() {
 | 
			
		||||
  return color_mode;
 | 
			
		||||
}
 | 
			
		||||
std::set<ColorMode> LightCall::get_suitable_color_modes_() {
 | 
			
		||||
  bool has_white = this->white_.has_value() && *this->white_ > 0.0f;
 | 
			
		||||
  bool has_ct = this->color_temperature_.has_value();
 | 
			
		||||
  bool has_cwww = (this->cold_white_.has_value() && *this->cold_white_ > 0.0f) ||
 | 
			
		||||
                  (this->warm_white_.has_value() && *this->warm_white_ > 0.0f);
 | 
			
		||||
  bool has_rgb = (this->color_brightness_.has_value() && *this->color_brightness_ > 0.0f) ||
 | 
			
		||||
                 (this->red_.has_value() || this->green_.has_value() || this->blue_.has_value());
 | 
			
		||||
  bool has_white = this->has_white() && this->white_ > 0.0f;
 | 
			
		||||
  bool has_ct = this->has_color_temperature();
 | 
			
		||||
  bool has_cwww =
 | 
			
		||||
      (this->has_cold_white() && this->cold_white_ > 0.0f) || (this->has_warm_white() && this->warm_white_ > 0.0f);
 | 
			
		||||
  bool has_rgb = (this->has_color_brightness() && this->color_brightness_ > 0.0f) ||
 | 
			
		||||
                 (this->has_red() || this->has_green() || this->has_blue());
 | 
			
		||||
 | 
			
		||||
#define KEY(white, ct, cwww, rgb) ((white) << 0 | (ct) << 1 | (cwww) << 2 | (rgb) << 3)
 | 
			
		||||
#define ENTRY(white, ct, cwww, rgb, ...) \
 | 
			
		||||
@@ -491,7 +515,7 @@ LightCall &LightCall::from_light_color_values(const LightColorValues &values) {
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
ColorMode LightCall::get_active_color_mode_() {
 | 
			
		||||
  return this->color_mode_.value_or(this->parent_->remote_values.get_color_mode());
 | 
			
		||||
  return this->has_color_mode() ? this->color_mode_ : this->parent_->remote_values.get_color_mode();
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_transition_length_if_supported(uint32_t transition_length) {
 | 
			
		||||
  if (this->get_active_color_mode_() & ColorCapability::BRIGHTNESS)
 | 
			
		||||
@@ -505,7 +529,7 @@ LightCall &LightCall::set_brightness_if_supported(float brightness) {
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_color_mode_if_supported(ColorMode color_mode) {
 | 
			
		||||
  if (this->parent_->get_traits().supports_color_mode(color_mode))
 | 
			
		||||
    this->color_mode_ = color_mode;
 | 
			
		||||
    this->set_color_mode(color_mode);
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_color_brightness_if_supported(float brightness) {
 | 
			
		||||
@@ -549,110 +573,19 @@ LightCall &LightCall::set_warm_white_if_supported(float warm_white) {
 | 
			
		||||
    this->set_warm_white(warm_white);
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_state(optional<bool> state) {
 | 
			
		||||
  this->state_ = state;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_state(bool state) {
 | 
			
		||||
  this->state_ = state;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_transition_length(optional<uint32_t> transition_length) {
 | 
			
		||||
  this->transition_length_ = transition_length;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_transition_length(uint32_t transition_length) {
 | 
			
		||||
  this->transition_length_ = transition_length;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_flash_length(optional<uint32_t> flash_length) {
 | 
			
		||||
  this->flash_length_ = flash_length;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_flash_length(uint32_t flash_length) {
 | 
			
		||||
  this->flash_length_ = flash_length;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_brightness(optional<float> brightness) {
 | 
			
		||||
  this->brightness_ = brightness;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_brightness(float brightness) {
 | 
			
		||||
  this->brightness_ = brightness;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_color_mode(optional<ColorMode> color_mode) {
 | 
			
		||||
  this->color_mode_ = color_mode;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_color_mode(ColorMode color_mode) {
 | 
			
		||||
  this->color_mode_ = color_mode;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_color_brightness(optional<float> brightness) {
 | 
			
		||||
  this->color_brightness_ = brightness;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_color_brightness(float brightness) {
 | 
			
		||||
  this->color_brightness_ = brightness;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_red(optional<float> red) {
 | 
			
		||||
  this->red_ = red;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_red(float red) {
 | 
			
		||||
  this->red_ = red;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_green(optional<float> green) {
 | 
			
		||||
  this->green_ = green;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_green(float green) {
 | 
			
		||||
  this->green_ = green;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_blue(optional<float> blue) {
 | 
			
		||||
  this->blue_ = blue;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_blue(float blue) {
 | 
			
		||||
  this->blue_ = blue;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_white(optional<float> white) {
 | 
			
		||||
  this->white_ = white;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_white(float white) {
 | 
			
		||||
  this->white_ = white;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_color_temperature(optional<float> color_temperature) {
 | 
			
		||||
  this->color_temperature_ = color_temperature;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_color_temperature(float color_temperature) {
 | 
			
		||||
  this->color_temperature_ = color_temperature;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_cold_white(optional<float> cold_white) {
 | 
			
		||||
  this->cold_white_ = cold_white;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_cold_white(float cold_white) {
 | 
			
		||||
  this->cold_white_ = cold_white;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_warm_white(optional<float> warm_white) {
 | 
			
		||||
  this->warm_white_ = warm_white;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_warm_white(float warm_white) {
 | 
			
		||||
  this->warm_white_ = warm_white;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(state, bool, FLAG_HAS_STATE)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(transition_length, uint32_t, FLAG_HAS_TRANSITION)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(flash_length, uint32_t, FLAG_HAS_FLASH)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(brightness, float, FLAG_HAS_BRIGHTNESS)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(color_mode, ColorMode, FLAG_HAS_COLOR_MODE)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(color_brightness, float, FLAG_HAS_COLOR_BRIGHTNESS)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(red, float, FLAG_HAS_RED)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(green, float, FLAG_HAS_GREEN)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(blue, float, FLAG_HAS_BLUE)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(white, float, FLAG_HAS_WHITE)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(color_temperature, float, FLAG_HAS_COLOR_TEMPERATURE)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(cold_white, float, FLAG_HAS_COLD_WHITE)
 | 
			
		||||
IMPLEMENT_LIGHT_CALL_SETTER(warm_white, float, FLAG_HAS_WARM_WHITE)
 | 
			
		||||
LightCall &LightCall::set_effect(optional<std::string> effect) {
 | 
			
		||||
  if (effect.has_value())
 | 
			
		||||
    this->set_effect(*effect);
 | 
			
		||||
@@ -660,18 +593,22 @@ LightCall &LightCall::set_effect(optional<std::string> effect) {
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_effect(uint32_t effect_number) {
 | 
			
		||||
  this->effect_ = effect_number;
 | 
			
		||||
  this->set_flag_(FLAG_HAS_EFFECT, true);
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_effect(optional<uint32_t> effect_number) {
 | 
			
		||||
  this->effect_ = effect_number;
 | 
			
		||||
  if (effect_number.has_value()) {
 | 
			
		||||
    this->effect_ = effect_number.value();
 | 
			
		||||
  }
 | 
			
		||||
  this->set_flag_(FLAG_HAS_EFFECT, effect_number.has_value());
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_publish(bool publish) {
 | 
			
		||||
  this->publish_ = publish;
 | 
			
		||||
  this->set_flag_(FLAG_PUBLISH, publish);
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_save(bool save) {
 | 
			
		||||
  this->save_ = save;
 | 
			
		||||
  this->set_flag_(FLAG_SAVE, save);
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
LightCall &LightCall::set_rgb(float red, float green, float blue) {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "esphome/core/optional.h"
 | 
			
		||||
#include "light_color_values.h"
 | 
			
		||||
#include <set>
 | 
			
		||||
 | 
			
		||||
@@ -10,6 +9,11 @@ namespace light {
 | 
			
		||||
class LightState;
 | 
			
		||||
 | 
			
		||||
/** This class represents a requested change in a light state.
 | 
			
		||||
 *
 | 
			
		||||
 * Light state changes are tracked using a bitfield flags_ to minimize memory usage.
 | 
			
		||||
 * Each possible light property has a flag indicating whether it has been set.
 | 
			
		||||
 * This design keeps LightCall at ~56 bytes to minimize heap fragmentation on
 | 
			
		||||
 * ESP8266 and other memory-constrained devices.
 | 
			
		||||
 */
 | 
			
		||||
class LightCall {
 | 
			
		||||
 public:
 | 
			
		||||
@@ -131,6 +135,19 @@ class LightCall {
 | 
			
		||||
  /// Set whether this light call should trigger a save state to recover them at startup..
 | 
			
		||||
  LightCall &set_save(bool save);
 | 
			
		||||
 | 
			
		||||
  // Getter methods to check if values are set
 | 
			
		||||
  bool has_state() const { return (flags_ & FLAG_HAS_STATE) != 0; }
 | 
			
		||||
  bool has_brightness() const { return (flags_ & FLAG_HAS_BRIGHTNESS) != 0; }
 | 
			
		||||
  bool has_color_brightness() const { return (flags_ & FLAG_HAS_COLOR_BRIGHTNESS) != 0; }
 | 
			
		||||
  bool has_red() const { return (flags_ & FLAG_HAS_RED) != 0; }
 | 
			
		||||
  bool has_green() const { return (flags_ & FLAG_HAS_GREEN) != 0; }
 | 
			
		||||
  bool has_blue() const { return (flags_ & FLAG_HAS_BLUE) != 0; }
 | 
			
		||||
  bool has_white() const { return (flags_ & FLAG_HAS_WHITE) != 0; }
 | 
			
		||||
  bool has_color_temperature() const { return (flags_ & FLAG_HAS_COLOR_TEMPERATURE) != 0; }
 | 
			
		||||
  bool has_cold_white() const { return (flags_ & FLAG_HAS_COLD_WHITE) != 0; }
 | 
			
		||||
  bool has_warm_white() const { return (flags_ & FLAG_HAS_WARM_WHITE) != 0; }
 | 
			
		||||
  bool has_color_mode() const { return (flags_ & FLAG_HAS_COLOR_MODE) != 0; }
 | 
			
		||||
 | 
			
		||||
  /** Set the RGB color of the light by RGB values.
 | 
			
		||||
   *
 | 
			
		||||
   * Please note that this only changes the color of the light, not the brightness.
 | 
			
		||||
@@ -170,27 +187,62 @@ class LightCall {
 | 
			
		||||
  /// Some color modes also can be set using non-native parameters, transform those calls.
 | 
			
		||||
  void transform_parameters_();
 | 
			
		||||
 | 
			
		||||
  bool has_transition_() { return this->transition_length_.has_value(); }
 | 
			
		||||
  bool has_flash_() { return this->flash_length_.has_value(); }
 | 
			
		||||
  bool has_effect_() { return this->effect_.has_value(); }
 | 
			
		||||
  // Bitfield flags - each flag indicates whether a corresponding value has been set.
 | 
			
		||||
  enum FieldFlags : uint16_t {
 | 
			
		||||
    FLAG_HAS_STATE = 1 << 0,
 | 
			
		||||
    FLAG_HAS_TRANSITION = 1 << 1,
 | 
			
		||||
    FLAG_HAS_FLASH = 1 << 2,
 | 
			
		||||
    FLAG_HAS_EFFECT = 1 << 3,
 | 
			
		||||
    FLAG_HAS_BRIGHTNESS = 1 << 4,
 | 
			
		||||
    FLAG_HAS_COLOR_BRIGHTNESS = 1 << 5,
 | 
			
		||||
    FLAG_HAS_RED = 1 << 6,
 | 
			
		||||
    FLAG_HAS_GREEN = 1 << 7,
 | 
			
		||||
    FLAG_HAS_BLUE = 1 << 8,
 | 
			
		||||
    FLAG_HAS_WHITE = 1 << 9,
 | 
			
		||||
    FLAG_HAS_COLOR_TEMPERATURE = 1 << 10,
 | 
			
		||||
    FLAG_HAS_COLD_WHITE = 1 << 11,
 | 
			
		||||
    FLAG_HAS_WARM_WHITE = 1 << 12,
 | 
			
		||||
    FLAG_HAS_COLOR_MODE = 1 << 13,
 | 
			
		||||
    FLAG_PUBLISH = 1 << 14,
 | 
			
		||||
    FLAG_SAVE = 1 << 15,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  bool has_transition_() { return (this->flags_ & FLAG_HAS_TRANSITION) != 0; }
 | 
			
		||||
  bool has_flash_() { return (this->flags_ & FLAG_HAS_FLASH) != 0; }
 | 
			
		||||
  bool has_effect_() { return (this->flags_ & FLAG_HAS_EFFECT) != 0; }
 | 
			
		||||
  bool get_publish_() { return (this->flags_ & FLAG_PUBLISH) != 0; }
 | 
			
		||||
  bool get_save_() { return (this->flags_ & FLAG_SAVE) != 0; }
 | 
			
		||||
 | 
			
		||||
  // Helper to set flag
 | 
			
		||||
  void set_flag_(FieldFlags flag, bool value) {
 | 
			
		||||
    if (value) {
 | 
			
		||||
      this->flags_ |= flag;
 | 
			
		||||
    } else {
 | 
			
		||||
      this->flags_ &= ~flag;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  LightState *parent_;
 | 
			
		||||
  optional<bool> state_;
 | 
			
		||||
  optional<uint32_t> transition_length_;
 | 
			
		||||
  optional<uint32_t> flash_length_;
 | 
			
		||||
  optional<ColorMode> color_mode_;
 | 
			
		||||
  optional<float> brightness_;
 | 
			
		||||
  optional<float> color_brightness_;
 | 
			
		||||
  optional<float> red_;
 | 
			
		||||
  optional<float> green_;
 | 
			
		||||
  optional<float> blue_;
 | 
			
		||||
  optional<float> white_;
 | 
			
		||||
  optional<float> color_temperature_;
 | 
			
		||||
  optional<float> cold_white_;
 | 
			
		||||
  optional<float> warm_white_;
 | 
			
		||||
  optional<uint32_t> effect_;
 | 
			
		||||
  bool publish_{true};
 | 
			
		||||
  bool save_{true};
 | 
			
		||||
 | 
			
		||||
  // Light state values - use flags_ to check if a value has been set.
 | 
			
		||||
  // Group 4-byte aligned members first
 | 
			
		||||
  uint32_t transition_length_;
 | 
			
		||||
  uint32_t flash_length_;
 | 
			
		||||
  uint32_t effect_;
 | 
			
		||||
  float brightness_;
 | 
			
		||||
  float color_brightness_;
 | 
			
		||||
  float red_;
 | 
			
		||||
  float green_;
 | 
			
		||||
  float blue_;
 | 
			
		||||
  float white_;
 | 
			
		||||
  float color_temperature_;
 | 
			
		||||
  float cold_white_;
 | 
			
		||||
  float warm_white_;
 | 
			
		||||
 | 
			
		||||
  // Smaller members at the end for better packing
 | 
			
		||||
  uint16_t flags_{FLAG_PUBLISH | FLAG_SAVE};  // Tracks which values are set
 | 
			
		||||
  ColorMode color_mode_;
 | 
			
		||||
  bool state_;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace light
 | 
			
		||||
 
 | 
			
		||||
@@ -46,8 +46,7 @@ class LightColorValues {
 | 
			
		||||
 public:
 | 
			
		||||
  /// Construct the LightColorValues with all attributes enabled, but state set to off.
 | 
			
		||||
  LightColorValues()
 | 
			
		||||
      : color_mode_(ColorMode::UNKNOWN),
 | 
			
		||||
        state_(0.0f),
 | 
			
		||||
      : state_(0.0f),
 | 
			
		||||
        brightness_(1.0f),
 | 
			
		||||
        color_brightness_(1.0f),
 | 
			
		||||
        red_(1.0f),
 | 
			
		||||
@@ -56,7 +55,8 @@ class LightColorValues {
 | 
			
		||||
        white_(1.0f),
 | 
			
		||||
        color_temperature_{0.0f},
 | 
			
		||||
        cold_white_{1.0f},
 | 
			
		||||
        warm_white_{1.0f} {}
 | 
			
		||||
        warm_white_{1.0f},
 | 
			
		||||
        color_mode_(ColorMode::UNKNOWN) {}
 | 
			
		||||
 | 
			
		||||
  LightColorValues(ColorMode color_mode, float state, float brightness, float color_brightness, float red, float green,
 | 
			
		||||
                   float blue, float white, float color_temperature, float cold_white, float warm_white) {
 | 
			
		||||
@@ -292,7 +292,6 @@ class LightColorValues {
 | 
			
		||||
  void set_warm_white(float warm_white) { this->warm_white_ = clamp(warm_white, 0.0f, 1.0f); }
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  ColorMode color_mode_;
 | 
			
		||||
  float state_;  ///< ON / OFF, float for transition
 | 
			
		||||
  float brightness_;
 | 
			
		||||
  float color_brightness_;
 | 
			
		||||
@@ -303,6 +302,7 @@ class LightColorValues {
 | 
			
		||||
  float color_temperature_;  ///< Color Temperature in Mired
 | 
			
		||||
  float cold_white_;
 | 
			
		||||
  float warm_white_;
 | 
			
		||||
  ColorMode color_mode_;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace light
 | 
			
		||||
 
 | 
			
		||||
@@ -31,9 +31,7 @@ enum LightRestoreMode : uint8_t {
 | 
			
		||||
struct LightStateRTCState {
 | 
			
		||||
  LightStateRTCState(ColorMode color_mode, bool state, float brightness, float color_brightness, float red, float green,
 | 
			
		||||
                     float blue, float white, float color_temp, float cold_white, float warm_white)
 | 
			
		||||
      : color_mode(color_mode),
 | 
			
		||||
        state(state),
 | 
			
		||||
        brightness(brightness),
 | 
			
		||||
      : brightness(brightness),
 | 
			
		||||
        color_brightness(color_brightness),
 | 
			
		||||
        red(red),
 | 
			
		||||
        green(green),
 | 
			
		||||
@@ -41,10 +39,12 @@ struct LightStateRTCState {
 | 
			
		||||
        white(white),
 | 
			
		||||
        color_temp(color_temp),
 | 
			
		||||
        cold_white(cold_white),
 | 
			
		||||
        warm_white(warm_white) {}
 | 
			
		||||
        warm_white(warm_white),
 | 
			
		||||
        effect(0),
 | 
			
		||||
        color_mode(color_mode),
 | 
			
		||||
        state(state) {}
 | 
			
		||||
  LightStateRTCState() = default;
 | 
			
		||||
  ColorMode color_mode{ColorMode::UNKNOWN};
 | 
			
		||||
  bool state{false};
 | 
			
		||||
  // Group 4-byte aligned members first
 | 
			
		||||
  float brightness{1.0f};
 | 
			
		||||
  float color_brightness{1.0f};
 | 
			
		||||
  float red{1.0f};
 | 
			
		||||
@@ -55,6 +55,9 @@ struct LightStateRTCState {
 | 
			
		||||
  float cold_white{1.0f};
 | 
			
		||||
  float warm_white{1.0f};
 | 
			
		||||
  uint32_t effect{0};
 | 
			
		||||
  // Group smaller members at the end
 | 
			
		||||
  ColorMode color_mode{ColorMode::UNKNOWN};
 | 
			
		||||
  bool state{false};
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/** This class represents the communication layer between the front-end MQTT layer and the
 | 
			
		||||
@@ -216,6 +219,8 @@ class LightState : public EntityBase, public Component {
 | 
			
		||||
  std::unique_ptr<LightTransformer> transformer_{nullptr};
 | 
			
		||||
  /// List of effects for this light.
 | 
			
		||||
  std::vector<LightEffect *> effects_;
 | 
			
		||||
  /// Object used to store the persisted values of the light.
 | 
			
		||||
  ESPPreferenceObject rtc_;
 | 
			
		||||
  /// Value for storing the index of the currently active effect. 0 if no effect is active
 | 
			
		||||
  uint32_t active_effect_index_{};
 | 
			
		||||
  /// Default transition length for all transitions in ms.
 | 
			
		||||
@@ -224,15 +229,11 @@ class LightState : public EntityBase, public Component {
 | 
			
		||||
  uint32_t flash_transition_length_{};
 | 
			
		||||
  /// Gamma correction factor for the light.
 | 
			
		||||
  float gamma_correct_{};
 | 
			
		||||
 | 
			
		||||
  /// Whether the light value should be written in the next cycle.
 | 
			
		||||
  bool next_write_{true};
 | 
			
		||||
  // for effects, true if a transformer (transition) is active.
 | 
			
		||||
  bool is_transformer_active_ = false;
 | 
			
		||||
 | 
			
		||||
  /// Object used to store the persisted values of the light.
 | 
			
		||||
  ESPPreferenceObject rtc_;
 | 
			
		||||
 | 
			
		||||
  /** Callback to call when new values for the frontend are available.
 | 
			
		||||
   *
 | 
			
		||||
   * "Remote values" are light color values that are reported to the frontend and have a lower
 | 
			
		||||
 
 | 
			
		||||
@@ -59,9 +59,9 @@ class LightTransitionTransformer : public LightTransformer {
 | 
			
		||||
  // transition from 0 to 1 on x = [0, 1]
 | 
			
		||||
  static float smoothed_progress(float x) { return x * x * x * (x * (x * 6.0f - 15.0f) + 10.0f); }
 | 
			
		||||
 | 
			
		||||
  bool changing_color_mode_{false};
 | 
			
		||||
  LightColorValues end_values_{};
 | 
			
		||||
  LightColorValues intermediate_values_{};
 | 
			
		||||
  bool changing_color_mode_{false};
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
class LightFlashTransformer : public LightTransformer {
 | 
			
		||||
@@ -117,8 +117,8 @@ class LightFlashTransformer : public LightTransformer {
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  LightState &state_;
 | 
			
		||||
  uint32_t transition_length_;
 | 
			
		||||
  std::unique_ptr<LightTransformer> transformer_{nullptr};
 | 
			
		||||
  uint32_t transition_length_;
 | 
			
		||||
  bool begun_lightstate_restore_;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -21,6 +21,7 @@ from esphome.components.libretiny.const import (
 | 
			
		||||
    COMPONENT_LN882X,
 | 
			
		||||
    COMPONENT_RTL87XX,
 | 
			
		||||
)
 | 
			
		||||
from esphome.config_helpers import filter_source_files_from_platform
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_ARGS,
 | 
			
		||||
@@ -42,6 +43,7 @@ from esphome.const import (
 | 
			
		||||
    PLATFORM_LN882X,
 | 
			
		||||
    PLATFORM_RP2040,
 | 
			
		||||
    PLATFORM_RTL87XX,
 | 
			
		||||
    PlatformFramework,
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import CORE, Lambda, coroutine_with_priority
 | 
			
		||||
 | 
			
		||||
@@ -444,3 +446,25 @@ async def logger_set_level_to_code(config, action_id, template_arg, args):
 | 
			
		||||
 | 
			
		||||
    lambda_ = await cg.process_lambda(Lambda(text), args, return_type=cg.void)
 | 
			
		||||
    return cg.new_Pvariable(action_id, template_arg, lambda_)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FILTER_SOURCE_FILES = filter_source_files_from_platform(
 | 
			
		||||
    {
 | 
			
		||||
        "logger_esp32.cpp": {
 | 
			
		||||
            PlatformFramework.ESP32_ARDUINO,
 | 
			
		||||
            PlatformFramework.ESP32_IDF,
 | 
			
		||||
        },
 | 
			
		||||
        "logger_esp8266.cpp": {PlatformFramework.ESP8266_ARDUINO},
 | 
			
		||||
        "logger_host.cpp": {PlatformFramework.HOST_NATIVE},
 | 
			
		||||
        "logger_rp2040.cpp": {PlatformFramework.RP2040_ARDUINO},
 | 
			
		||||
        "logger_libretiny.cpp": {
 | 
			
		||||
            PlatformFramework.BK72XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.RTL87XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.LN882X_ARDUINO,
 | 
			
		||||
        },
 | 
			
		||||
        "task_log_buffer.cpp": {
 | 
			
		||||
            PlatformFramework.ESP32_ARDUINO,
 | 
			
		||||
            PlatformFramework.ESP32_IDF,
 | 
			
		||||
        },
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
 
 | 
			
		||||
@@ -90,6 +90,25 @@ void HOT Logger::log_vprintf_(uint8_t level, const char *tag, int line, const ch
 | 
			
		||||
#ifdef USE_STORE_LOG_STR_IN_FLASH
 | 
			
		||||
// Implementation for ESP8266 with flash string support.
 | 
			
		||||
// Note: USE_STORE_LOG_STR_IN_FLASH is only defined for ESP8266.
 | 
			
		||||
//
 | 
			
		||||
// This function handles format strings stored in flash memory (PROGMEM) to save RAM.
 | 
			
		||||
// The buffer is used in a special way to avoid allocating extra memory:
 | 
			
		||||
//
 | 
			
		||||
// Memory layout during execution:
 | 
			
		||||
// Step 1: Copy format string from flash to buffer
 | 
			
		||||
//         tx_buffer_: [format_string][null][.....................]
 | 
			
		||||
//         tx_buffer_at_: ------------------^
 | 
			
		||||
//         msg_start: saved here -----------^
 | 
			
		||||
//
 | 
			
		||||
// Step 2: format_log_to_buffer_with_terminator_ reads format string from beginning
 | 
			
		||||
//         and writes formatted output starting at msg_start position
 | 
			
		||||
//         tx_buffer_: [format_string][null][formatted_message][null]
 | 
			
		||||
//         tx_buffer_at_: -------------------------------------^
 | 
			
		||||
//
 | 
			
		||||
// Step 3: Output the formatted message (starting at msg_start)
 | 
			
		||||
//         write_msg_ and callbacks receive: this->tx_buffer_ + msg_start
 | 
			
		||||
//         which points to: [formatted_message][null]
 | 
			
		||||
//
 | 
			
		||||
void Logger::log_vprintf_(uint8_t level, const char *tag, int line, const __FlashStringHelper *format,
 | 
			
		||||
                          va_list args) {  // NOLINT
 | 
			
		||||
  if (level > this->level_for(tag) || global_recursion_guard_)
 | 
			
		||||
@@ -121,7 +140,9 @@ void Logger::log_vprintf_(uint8_t level, const char *tag, int line, const __Flas
 | 
			
		||||
  if (this->baud_rate_ > 0) {
 | 
			
		||||
    this->write_msg_(this->tx_buffer_ + msg_start);
 | 
			
		||||
  }
 | 
			
		||||
  this->log_callback_.call(level, tag, this->tx_buffer_ + msg_start);
 | 
			
		||||
  size_t msg_length =
 | 
			
		||||
      this->tx_buffer_at_ - msg_start;  // Don't subtract 1 - tx_buffer_at_ is already at the null terminator position
 | 
			
		||||
  this->log_callback_.call(level, tag, this->tx_buffer_ + msg_start, msg_length);
 | 
			
		||||
 | 
			
		||||
  global_recursion_guard_ = false;
 | 
			
		||||
}
 | 
			
		||||
@@ -185,7 +206,8 @@ void Logger::loop() {
 | 
			
		||||
                                  this->tx_buffer_size_);
 | 
			
		||||
      this->write_footer_to_buffer_(this->tx_buffer_, &this->tx_buffer_at_, this->tx_buffer_size_);
 | 
			
		||||
      this->tx_buffer_[this->tx_buffer_at_] = '\0';
 | 
			
		||||
      this->log_callback_.call(message->level, message->tag, this->tx_buffer_);
 | 
			
		||||
      size_t msg_len = this->tx_buffer_at_;  // We already know the length from tx_buffer_at_
 | 
			
		||||
      this->log_callback_.call(message->level, message->tag, this->tx_buffer_, msg_len);
 | 
			
		||||
      // At this point all the data we need from message has been transferred to the tx_buffer
 | 
			
		||||
      // so we can release the message to allow other tasks to use it as soon as possible.
 | 
			
		||||
      this->log_buffer_->release_message_main_loop(received_token);
 | 
			
		||||
@@ -214,7 +236,7 @@ void Logger::set_log_level(const std::string &tag, uint8_t log_level) { this->lo
 | 
			
		||||
UARTSelection Logger::get_uart() const { return this->uart_; }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
void Logger::add_on_log_callback(std::function<void(uint8_t, const char *, const char *)> &&callback) {
 | 
			
		||||
void Logger::add_on_log_callback(std::function<void(uint8_t, const char *, const char *, size_t)> &&callback) {
 | 
			
		||||
  this->log_callback_.add(std::move(callback));
 | 
			
		||||
}
 | 
			
		||||
float Logger::get_setup_priority() const { return setup_priority::BUS + 500.0f; }
 | 
			
		||||
 
 | 
			
		||||
@@ -143,7 +143,7 @@ class Logger : public Component {
 | 
			
		||||
  inline uint8_t level_for(const char *tag);
 | 
			
		||||
 | 
			
		||||
  /// Register a callback that will be called for every log message sent
 | 
			
		||||
  void add_on_log_callback(std::function<void(uint8_t, const char *, const char *)> &&callback);
 | 
			
		||||
  void add_on_log_callback(std::function<void(uint8_t, const char *, const char *, size_t)> &&callback);
 | 
			
		||||
 | 
			
		||||
  // add a listener for log level changes
 | 
			
		||||
  void add_listener(std::function<void(uint8_t)> &&callback) { this->level_callback_.add(std::move(callback)); }
 | 
			
		||||
@@ -192,7 +192,7 @@ class Logger : public Component {
 | 
			
		||||
    if (this->baud_rate_ > 0) {
 | 
			
		||||
      this->write_msg_(this->tx_buffer_);  // If logging is enabled, write to console
 | 
			
		||||
    }
 | 
			
		||||
    this->log_callback_.call(level, tag, this->tx_buffer_);
 | 
			
		||||
    this->log_callback_.call(level, tag, this->tx_buffer_, this->tx_buffer_at_);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Write the body of the log message to the buffer
 | 
			
		||||
@@ -246,7 +246,7 @@ class Logger : public Component {
 | 
			
		||||
 | 
			
		||||
  // Large objects (internally aligned)
 | 
			
		||||
  std::map<std::string, uint8_t> log_levels_{};
 | 
			
		||||
  CallbackManager<void(uint8_t, const char *, const char *)> log_callback_{};
 | 
			
		||||
  CallbackManager<void(uint8_t, const char *, const char *, size_t)> log_callback_{};
 | 
			
		||||
  CallbackManager<void(uint8_t)> level_callback_{};
 | 
			
		||||
#ifdef USE_ESPHOME_TASK_LOG_BUFFER
 | 
			
		||||
  std::unique_ptr<logger::TaskLogBuffer> log_buffer_;  // Will be initialized with init_log_buffer
 | 
			
		||||
@@ -355,7 +355,7 @@ class Logger : public Component {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  inline void HOT write_footer_to_buffer_(char *buffer, uint16_t *buffer_at, uint16_t buffer_size) {
 | 
			
		||||
    static const uint16_t RESET_COLOR_LEN = strlen(ESPHOME_LOG_RESET_COLOR);
 | 
			
		||||
    static constexpr uint16_t RESET_COLOR_LEN = sizeof(ESPHOME_LOG_RESET_COLOR) - 1;
 | 
			
		||||
    this->write_body_to_buffer_(ESPHOME_LOG_RESET_COLOR, RESET_COLOR_LEN, buffer, buffer_at, buffer_size);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@@ -385,7 +385,7 @@ class LoggerMessageTrigger : public Trigger<uint8_t, const char *, const char *>
 | 
			
		||||
 public:
 | 
			
		||||
  explicit LoggerMessageTrigger(Logger *parent, uint8_t level) {
 | 
			
		||||
    this->level_ = level;
 | 
			
		||||
    parent->add_on_log_callback([this](uint8_t level, const char *tag, const char *message) {
 | 
			
		||||
    parent->add_on_log_callback([this](uint8_t level, const char *tag, const char *message, size_t message_len) {
 | 
			
		||||
      if (level <= this->level_) {
 | 
			
		||||
        this->trigger(level, tag, message);
 | 
			
		||||
      }
 | 
			
		||||
 
 | 
			
		||||
@@ -184,7 +184,9 @@ void HOT Logger::write_msg_(const char *msg) {
 | 
			
		||||
  ) {
 | 
			
		||||
    puts(msg);
 | 
			
		||||
  } else {
 | 
			
		||||
    uart_write_bytes(this->uart_num_, msg, strlen(msg));
 | 
			
		||||
    // Use tx_buffer_at_ if msg points to tx_buffer_, otherwise fall back to strlen
 | 
			
		||||
    size_t len = (msg == this->tx_buffer_) ? this->tx_buffer_at_ : strlen(msg);
 | 
			
		||||
    uart_write_bytes(this->uart_num_, msg, len);
 | 
			
		||||
    uart_write_bytes(this->uart_num_, "\n", 1);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										0
									
								
								esphome/components/lps22/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								esphome/components/lps22/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										75
									
								
								esphome/components/lps22/lps22.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								esphome/components/lps22/lps22.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,75 @@
 | 
			
		||||
#include "lps22.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace lps22 {
 | 
			
		||||
 | 
			
		||||
static constexpr const char *const TAG = "lps22";
 | 
			
		||||
 | 
			
		||||
static constexpr uint8_t WHO_AM_I = 0x0F;
 | 
			
		||||
static constexpr uint8_t LPS22HB_ID = 0xB1;
 | 
			
		||||
static constexpr uint8_t LPS22HH_ID = 0xB3;
 | 
			
		||||
static constexpr uint8_t CTRL_REG2 = 0x11;
 | 
			
		||||
static constexpr uint8_t CTRL_REG2_ONE_SHOT_MASK = 0b1;
 | 
			
		||||
static constexpr uint8_t STATUS = 0x27;
 | 
			
		||||
static constexpr uint8_t STATUS_T_DA_MASK = 0b10;
 | 
			
		||||
static constexpr uint8_t STATUS_P_DA_MASK = 0b01;
 | 
			
		||||
static constexpr uint8_t TEMP_L = 0x2b;
 | 
			
		||||
static constexpr uint8_t PRES_OUT_XL = 0x28;
 | 
			
		||||
static constexpr uint8_t REF_P_XL = 0x28;
 | 
			
		||||
static constexpr uint8_t READ_ATTEMPTS = 10;
 | 
			
		||||
static constexpr uint8_t READ_INTERVAL = 5;
 | 
			
		||||
static constexpr float PRESSURE_SCALE = 1.0f / 4096.0f;
 | 
			
		||||
static constexpr float TEMPERATURE_SCALE = 0.01f;
 | 
			
		||||
 | 
			
		||||
void LPS22Component::setup() {
 | 
			
		||||
  uint8_t value = 0x00;
 | 
			
		||||
  this->read_register(WHO_AM_I, &value, 1);
 | 
			
		||||
  if (value != LPS22HB_ID && value != LPS22HH_ID) {
 | 
			
		||||
    ESP_LOGW(TAG, "device IDs as %02x, which isn't a known LPS22HB or LPS22HH ID", value);
 | 
			
		||||
    this->mark_failed();
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LPS22Component::dump_config() {
 | 
			
		||||
  ESP_LOGCONFIG(TAG, "LPS22:");
 | 
			
		||||
  LOG_SENSOR("  ", "Temperature", this->temperature_sensor_);
 | 
			
		||||
  LOG_SENSOR("  ", "Pressure", this->pressure_sensor_);
 | 
			
		||||
  LOG_I2C_DEVICE(this);
 | 
			
		||||
  LOG_UPDATE_INTERVAL(this);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void LPS22Component::update() {
 | 
			
		||||
  uint8_t value = 0x00;
 | 
			
		||||
  this->read_register(CTRL_REG2, &value, 1);
 | 
			
		||||
  value |= CTRL_REG2_ONE_SHOT_MASK;
 | 
			
		||||
  this->write_register(CTRL_REG2, &value, 1);
 | 
			
		||||
  this->set_retry(READ_INTERVAL, READ_ATTEMPTS, [this](uint8_t _) { return this->try_read_(); });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
RetryResult LPS22Component::try_read_() {
 | 
			
		||||
  uint8_t value = 0x00;
 | 
			
		||||
  this->read_register(STATUS, &value, 1);
 | 
			
		||||
  const uint8_t expected_status_mask = STATUS_T_DA_MASK | STATUS_P_DA_MASK;
 | 
			
		||||
  if ((value & expected_status_mask) != expected_status_mask) {
 | 
			
		||||
    ESP_LOGD(TAG, "STATUS not ready: %x", value);
 | 
			
		||||
    return RetryResult::RETRY;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (this->temperature_sensor_ != nullptr) {
 | 
			
		||||
    uint8_t t_buf[2]{0};
 | 
			
		||||
    this->read_register(TEMP_L, t_buf, 2);
 | 
			
		||||
    int16_t encoded = static_cast<int16_t>(encode_uint16(t_buf[1], t_buf[0]));
 | 
			
		||||
    float temp = TEMPERATURE_SCALE * static_cast<float>(encoded);
 | 
			
		||||
    this->temperature_sensor_->publish_state(temp);
 | 
			
		||||
  }
 | 
			
		||||
  if (this->pressure_sensor_ != nullptr) {
 | 
			
		||||
    uint8_t p_buf[3]{0};
 | 
			
		||||
    this->read_register(PRES_OUT_XL, p_buf, 3);
 | 
			
		||||
    uint32_t p_lsb = encode_uint24(p_buf[2], p_buf[1], p_buf[0]);
 | 
			
		||||
    this->pressure_sensor_->publish_state(PRESSURE_SCALE * static_cast<float>(p_lsb));
 | 
			
		||||
  }
 | 
			
		||||
  return RetryResult::DONE;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}  // namespace lps22
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
							
								
								
									
										27
									
								
								esphome/components/lps22/lps22.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								esphome/components/lps22/lps22.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
			
		||||
#pragma once
 | 
			
		||||
 | 
			
		||||
#include "esphome/core/component.h"
 | 
			
		||||
#include "esphome/components/sensor/sensor.h"
 | 
			
		||||
#include "esphome/components/i2c/i2c.h"
 | 
			
		||||
 | 
			
		||||
namespace esphome {
 | 
			
		||||
namespace lps22 {
 | 
			
		||||
 | 
			
		||||
class LPS22Component : public sensor::Sensor, public PollingComponent, public i2c::I2CDevice {
 | 
			
		||||
 public:
 | 
			
		||||
  void set_temperature_sensor(sensor::Sensor *temperature_sensor) { this->temperature_sensor_ = temperature_sensor; }
 | 
			
		||||
  void set_pressure_sensor(sensor::Sensor *pressure_sensor) { this->pressure_sensor_ = pressure_sensor; }
 | 
			
		||||
 | 
			
		||||
  void setup() override;
 | 
			
		||||
  void update() override;
 | 
			
		||||
  void dump_config() override;
 | 
			
		||||
 | 
			
		||||
 protected:
 | 
			
		||||
  sensor::Sensor *temperature_sensor_{nullptr};
 | 
			
		||||
  sensor::Sensor *pressure_sensor_{nullptr};
 | 
			
		||||
 | 
			
		||||
  RetryResult try_read_();
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
}  // namespace lps22
 | 
			
		||||
}  // namespace esphome
 | 
			
		||||
							
								
								
									
										58
									
								
								esphome/components/lps22/sensor.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								esphome/components/lps22/sensor.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,58 @@
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.components import i2c, sensor
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_ID,
 | 
			
		||||
    CONF_TEMPERATURE,
 | 
			
		||||
    CONF_PRESSURE,
 | 
			
		||||
    STATE_CLASS_MEASUREMENT,
 | 
			
		||||
    UNIT_CELSIUS,
 | 
			
		||||
    UNIT_HECTOPASCAL,
 | 
			
		||||
    ICON_THERMOMETER,
 | 
			
		||||
    DEVICE_CLASS_TEMPERATURE,
 | 
			
		||||
    DEVICE_CLASS_PRESSURE,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
CODEOWNERS = ["@nagisa"]
 | 
			
		||||
DEPENDENCIES = ["i2c"]
 | 
			
		||||
 | 
			
		||||
lps22 = cg.esphome_ns.namespace("lps22")
 | 
			
		||||
 | 
			
		||||
LPS22Component = lps22.class_("LPS22Component", cg.PollingComponent, i2c.I2CDevice)
 | 
			
		||||
 | 
			
		||||
CONFIG_SCHEMA = (
 | 
			
		||||
    cv.Schema(
 | 
			
		||||
        {
 | 
			
		||||
            cv.GenerateID(): cv.declare_id(LPS22Component),
 | 
			
		||||
            cv.Optional(CONF_TEMPERATURE): sensor.sensor_schema(
 | 
			
		||||
                unit_of_measurement=UNIT_CELSIUS,
 | 
			
		||||
                icon=ICON_THERMOMETER,
 | 
			
		||||
                accuracy_decimals=2,
 | 
			
		||||
                device_class=DEVICE_CLASS_TEMPERATURE,
 | 
			
		||||
                state_class=STATE_CLASS_MEASUREMENT,
 | 
			
		||||
            ),
 | 
			
		||||
            cv.Optional(CONF_PRESSURE): sensor.sensor_schema(
 | 
			
		||||
                unit_of_measurement=UNIT_HECTOPASCAL,
 | 
			
		||||
                accuracy_decimals=2,
 | 
			
		||||
                device_class=DEVICE_CLASS_PRESSURE,
 | 
			
		||||
                state_class=STATE_CLASS_MEASUREMENT,
 | 
			
		||||
            ),
 | 
			
		||||
        }
 | 
			
		||||
    )
 | 
			
		||||
    .extend(cv.polling_component_schema("60s"))
 | 
			
		||||
    .extend(i2c.i2c_device_schema(0x5D))  # can also be 0x5C
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
async def to_code(config):
 | 
			
		||||
    var = cg.new_Pvariable(config[CONF_ID])
 | 
			
		||||
    await cg.register_component(var, config)
 | 
			
		||||
    await i2c.register_i2c_device(var, config)
 | 
			
		||||
 | 
			
		||||
    if temperature_config := config.get(CONF_TEMPERATURE):
 | 
			
		||||
        sens = await sensor.new_sensor(temperature_config)
 | 
			
		||||
        cg.add(var.set_temperature_sensor(sens))
 | 
			
		||||
 | 
			
		||||
    if pressure_config := config.get(CONF_PRESSURE):
 | 
			
		||||
        sens = await sensor.new_sensor(pressure_config)
 | 
			
		||||
        cg.add(var.set_pressure_sensor(sens))
 | 
			
		||||
@@ -29,9 +29,9 @@ from ..defines import (
 | 
			
		||||
)
 | 
			
		||||
from ..helpers import add_lv_use, lvgl_components_required
 | 
			
		||||
from ..lv_validation import (
 | 
			
		||||
    angle,
 | 
			
		||||
    get_end_value,
 | 
			
		||||
    get_start_value,
 | 
			
		||||
    lv_angle,
 | 
			
		||||
    lv_bool,
 | 
			
		||||
    lv_color,
 | 
			
		||||
    lv_float,
 | 
			
		||||
@@ -162,7 +162,7 @@ SCALE_SCHEMA = cv.Schema(
 | 
			
		||||
        cv.Optional(CONF_RANGE_FROM, default=0.0): cv.float_,
 | 
			
		||||
        cv.Optional(CONF_RANGE_TO, default=100.0): cv.float_,
 | 
			
		||||
        cv.Optional(CONF_ANGLE_RANGE, default=270): cv.int_range(0, 360),
 | 
			
		||||
        cv.Optional(CONF_ROTATION): angle,
 | 
			
		||||
        cv.Optional(CONF_ROTATION): lv_angle,
 | 
			
		||||
        cv.Optional(CONF_INDICATORS): cv.ensure_list(INDICATOR_SCHEMA),
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
@@ -187,7 +187,7 @@ class MeterType(WidgetType):
 | 
			
		||||
        for scale_conf in config.get(CONF_SCALES, ()):
 | 
			
		||||
            rotation = 90 + (360 - scale_conf[CONF_ANGLE_RANGE]) / 2
 | 
			
		||||
            if CONF_ROTATION in scale_conf:
 | 
			
		||||
                rotation = scale_conf[CONF_ROTATION] // 10
 | 
			
		||||
                rotation = await lv_angle.process(scale_conf[CONF_ROTATION])
 | 
			
		||||
            with LocalVariable(
 | 
			
		||||
                "meter_var", "lv_meter_scale_t", lv_expr.meter_add_scale(var)
 | 
			
		||||
            ) as meter_var:
 | 
			
		||||
@@ -205,21 +205,20 @@ class MeterType(WidgetType):
 | 
			
		||||
                        var,
 | 
			
		||||
                        meter_var,
 | 
			
		||||
                        ticks[CONF_COUNT],
 | 
			
		||||
                        ticks[CONF_WIDTH],
 | 
			
		||||
                        ticks[CONF_LENGTH],
 | 
			
		||||
                        await size.process(ticks[CONF_WIDTH]),
 | 
			
		||||
                        await size.process(ticks[CONF_LENGTH]),
 | 
			
		||||
                        color,
 | 
			
		||||
                    )
 | 
			
		||||
                    if CONF_MAJOR in ticks:
 | 
			
		||||
                        major = ticks[CONF_MAJOR]
 | 
			
		||||
                        color = await lv_color.process(major[CONF_COLOR])
 | 
			
		||||
                        lv.meter_set_scale_major_ticks(
 | 
			
		||||
                            var,
 | 
			
		||||
                            meter_var,
 | 
			
		||||
                            major[CONF_STRIDE],
 | 
			
		||||
                            major[CONF_WIDTH],
 | 
			
		||||
                            major[CONF_LENGTH],
 | 
			
		||||
                            color,
 | 
			
		||||
                            major[CONF_LABEL_GAP],
 | 
			
		||||
                            await size.process(major[CONF_WIDTH]),
 | 
			
		||||
                            await size.process(major[CONF_LENGTH]),
 | 
			
		||||
                            await lv_color.process(major[CONF_COLOR]),
 | 
			
		||||
                            await size.process(major[CONF_LABEL_GAP]),
 | 
			
		||||
                        )
 | 
			
		||||
                for indicator in scale_conf.get(CONF_INDICATORS, ()):
 | 
			
		||||
                    (t, v) = next(iter(indicator.items()))
 | 
			
		||||
@@ -233,7 +232,11 @@ class MeterType(WidgetType):
 | 
			
		||||
                        lv_assign(
 | 
			
		||||
                            ivar,
 | 
			
		||||
                            lv_expr.meter_add_needle_line(
 | 
			
		||||
                                var, meter_var, v[CONF_WIDTH], color, v[CONF_R_MOD]
 | 
			
		||||
                                var,
 | 
			
		||||
                                meter_var,
 | 
			
		||||
                                await size.process(v[CONF_WIDTH]),
 | 
			
		||||
                                color,
 | 
			
		||||
                                await size.process(v[CONF_R_MOD]),
 | 
			
		||||
                            ),
 | 
			
		||||
                        )
 | 
			
		||||
                    if t == CONF_ARC:
 | 
			
		||||
@@ -241,7 +244,11 @@ class MeterType(WidgetType):
 | 
			
		||||
                        lv_assign(
 | 
			
		||||
                            ivar,
 | 
			
		||||
                            lv_expr.meter_add_arc(
 | 
			
		||||
                                var, meter_var, v[CONF_WIDTH], color, v[CONF_R_MOD]
 | 
			
		||||
                                var,
 | 
			
		||||
                                meter_var,
 | 
			
		||||
                                await size.process(v[CONF_WIDTH]),
 | 
			
		||||
                                color,
 | 
			
		||||
                                await size.process(v[CONF_R_MOD]),
 | 
			
		||||
                            ),
 | 
			
		||||
                        )
 | 
			
		||||
                    if t == CONF_TICK_STYLE:
 | 
			
		||||
@@ -257,7 +264,7 @@ class MeterType(WidgetType):
 | 
			
		||||
                                color_start,
 | 
			
		||||
                                color_end,
 | 
			
		||||
                                v[CONF_LOCAL],
 | 
			
		||||
                                v[CONF_WIDTH],
 | 
			
		||||
                                size.process(v[CONF_WIDTH]),
 | 
			
		||||
                            ),
 | 
			
		||||
                        )
 | 
			
		||||
                    if t == CONF_IMAGE:
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,6 @@
 | 
			
		||||
import esphome.codegen as cg
 | 
			
		||||
from esphome.components.esp32 import add_idf_component
 | 
			
		||||
from esphome.config_helpers import filter_source_files_from_platform
 | 
			
		||||
import esphome.config_validation as cv
 | 
			
		||||
from esphome.const import (
 | 
			
		||||
    CONF_DISABLED,
 | 
			
		||||
@@ -8,6 +9,7 @@ from esphome.const import (
 | 
			
		||||
    CONF_PROTOCOL,
 | 
			
		||||
    CONF_SERVICE,
 | 
			
		||||
    CONF_SERVICES,
 | 
			
		||||
    PlatformFramework,
 | 
			
		||||
)
 | 
			
		||||
from esphome.core import CORE, coroutine_with_priority
 | 
			
		||||
 | 
			
		||||
@@ -108,3 +110,21 @@ async def to_code(config):
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
        cg.add(var.add_extra_service(exp))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
FILTER_SOURCE_FILES = filter_source_files_from_platform(
 | 
			
		||||
    {
 | 
			
		||||
        "mdns_esp32.cpp": {
 | 
			
		||||
            PlatformFramework.ESP32_ARDUINO,
 | 
			
		||||
            PlatformFramework.ESP32_IDF,
 | 
			
		||||
        },
 | 
			
		||||
        "mdns_esp8266.cpp": {PlatformFramework.ESP8266_ARDUINO},
 | 
			
		||||
        "mdns_host.cpp": {PlatformFramework.HOST_NATIVE},
 | 
			
		||||
        "mdns_rp2040.cpp": {PlatformFramework.RP2040_ARDUINO},
 | 
			
		||||
        "mdns_libretiny.cpp": {
 | 
			
		||||
            PlatformFramework.BK72XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.RTL87XX_ARDUINO,
 | 
			
		||||
            PlatformFramework.LN882X_ARDUINO,
 | 
			
		||||
        },
 | 
			
		||||
    }
 | 
			
		||||
)
 | 
			
		||||
 
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user