mirror of
https://github.com/esphome/esphome.git
synced 2026-02-10 09:42:01 +00:00
Compare commits
16 Commits
dev
...
json_web_s
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
840ad30880 | ||
|
|
cfe121b38b | ||
|
|
5fbd9d5b14 | ||
|
|
2b1783ce61 | ||
|
|
904072ce79 | ||
|
|
0a4b98d74a | ||
|
|
b8017de724 | ||
|
|
ca96604582 | ||
|
|
d18d378f06 | ||
|
|
83e3752544 | ||
|
|
0490b2d450 | ||
|
|
55ff740e4e | ||
|
|
aba8a83cba | ||
|
|
a23809d5db | ||
|
|
32fc3ea6f5 | ||
|
|
deb8ffd348 |
@@ -1 +1 @@
|
|||||||
37ec8d5a343c8d0a485fd2118cbdabcbccd7b9bca197e4a392be75087974dced
|
cf3d341206b4184ec8b7fe85141aef4fe4696aa720c3f8a06d4e57930574bdab
|
||||||
|
|||||||
4
.github/workflows/codeql.yml
vendored
4
.github/workflows/codeql.yml
vendored
@@ -58,7 +58,7 @@ jobs:
|
|||||||
|
|
||||||
# Initializes the CodeQL tools for scanning.
|
# Initializes the CodeQL tools for scanning.
|
||||||
- name: Initialize CodeQL
|
- name: Initialize CodeQL
|
||||||
uses: github/codeql-action/init@45cbd0c69e560cd9e7cd7f8c32362050c9b7ded2 # v4.32.2
|
uses: github/codeql-action/init@b20883b0cd1f46c72ae0ba6d1090936928f9fa30 # v4.32.0
|
||||||
with:
|
with:
|
||||||
languages: ${{ matrix.language }}
|
languages: ${{ matrix.language }}
|
||||||
build-mode: ${{ matrix.build-mode }}
|
build-mode: ${{ matrix.build-mode }}
|
||||||
@@ -86,6 +86,6 @@ jobs:
|
|||||||
exit 1
|
exit 1
|
||||||
|
|
||||||
- name: Perform CodeQL Analysis
|
- name: Perform CodeQL Analysis
|
||||||
uses: github/codeql-action/analyze@45cbd0c69e560cd9e7cd7f8c32362050c9b7ded2 # v4.32.2
|
uses: github/codeql-action/analyze@b20883b0cd1f46c72ae0ba6d1090936928f9fa30 # v4.32.0
|
||||||
with:
|
with:
|
||||||
category: "/language:${{matrix.language}}"
|
category: "/language:${{matrix.language}}"
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ ci:
|
|||||||
repos:
|
repos:
|
||||||
- repo: https://github.com/astral-sh/ruff-pre-commit
|
- repo: https://github.com/astral-sh/ruff-pre-commit
|
||||||
# Ruff version.
|
# Ruff version.
|
||||||
rev: v0.15.0
|
rev: v0.14.14
|
||||||
hooks:
|
hooks:
|
||||||
# Run the linter.
|
# Run the linter.
|
||||||
- id: ruff
|
- id: ruff
|
||||||
|
|||||||
@@ -134,7 +134,6 @@ esphome/components/dfplayer/* @glmnet
|
|||||||
esphome/components/dfrobot_sen0395/* @niklasweber
|
esphome/components/dfrobot_sen0395/* @niklasweber
|
||||||
esphome/components/dht/* @OttoWinter
|
esphome/components/dht/* @OttoWinter
|
||||||
esphome/components/display_menu_base/* @numo68
|
esphome/components/display_menu_base/* @numo68
|
||||||
esphome/components/dlms_meter/* @SimonFischer04
|
|
||||||
esphome/components/dps310/* @kbx81
|
esphome/components/dps310/* @kbx81
|
||||||
esphome/components/ds1307/* @badbadc0ffee
|
esphome/components/ds1307/* @badbadc0ffee
|
||||||
esphome/components/ds2484/* @mrk-its
|
esphome/components/ds2484/* @mrk-its
|
||||||
@@ -532,7 +531,7 @@ esphome/components/uart/packet_transport/* @clydebarrow
|
|||||||
esphome/components/udp/* @clydebarrow
|
esphome/components/udp/* @clydebarrow
|
||||||
esphome/components/ufire_ec/* @pvizeli
|
esphome/components/ufire_ec/* @pvizeli
|
||||||
esphome/components/ufire_ise/* @pvizeli
|
esphome/components/ufire_ise/* @pvizeli
|
||||||
esphome/components/ultrasonic/* @ssieb @swoboda1337
|
esphome/components/ultrasonic/* @OttoWinter
|
||||||
esphome/components/update/* @jesserockz
|
esphome/components/update/* @jesserockz
|
||||||
esphome/components/uponor_smatrix/* @kroimon
|
esphome/components/uponor_smatrix/* @kroimon
|
||||||
esphome/components/usb_cdc_acm/* @kbx81
|
esphome/components/usb_cdc_acm/* @kbx81
|
||||||
|
|||||||
@@ -294,13 +294,8 @@ def has_api() -> bool:
|
|||||||
|
|
||||||
|
|
||||||
def has_ota() -> bool:
|
def has_ota() -> bool:
|
||||||
"""Check if OTA upload is available (requires platform: esphome)."""
|
"""Check if OTA is available."""
|
||||||
if CONF_OTA not in CORE.config:
|
return CONF_OTA in CORE.config
|
||||||
return False
|
|
||||||
return any(
|
|
||||||
ota_item.get(CONF_PLATFORM) == CONF_ESPHOME
|
|
||||||
for ota_item in CORE.config[CONF_OTA]
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def has_mqtt_ip_lookup() -> bool:
|
def has_mqtt_ip_lookup() -> bool:
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ from .const import (
|
|||||||
CORE_SUBCATEGORY_PATTERNS,
|
CORE_SUBCATEGORY_PATTERNS,
|
||||||
DEMANGLED_PATTERNS,
|
DEMANGLED_PATTERNS,
|
||||||
ESPHOME_COMPONENT_PATTERN,
|
ESPHOME_COMPONENT_PATTERN,
|
||||||
|
SECTION_TO_ATTR,
|
||||||
SYMBOL_PATTERNS,
|
SYMBOL_PATTERNS,
|
||||||
)
|
)
|
||||||
from .demangle import batch_demangle
|
from .demangle import batch_demangle
|
||||||
@@ -43,7 +44,6 @@ _READELF_SECTION_PATTERN = re.compile(
|
|||||||
# Component category prefixes
|
# Component category prefixes
|
||||||
_COMPONENT_PREFIX_ESPHOME = "[esphome]"
|
_COMPONENT_PREFIX_ESPHOME = "[esphome]"
|
||||||
_COMPONENT_PREFIX_EXTERNAL = "[external]"
|
_COMPONENT_PREFIX_EXTERNAL = "[external]"
|
||||||
_COMPONENT_PREFIX_LIB = "[lib]"
|
|
||||||
_COMPONENT_CORE = f"{_COMPONENT_PREFIX_ESPHOME}core"
|
_COMPONENT_CORE = f"{_COMPONENT_PREFIX_ESPHOME}core"
|
||||||
_COMPONENT_API = f"{_COMPONENT_PREFIX_ESPHOME}api"
|
_COMPONENT_API = f"{_COMPONENT_PREFIX_ESPHOME}api"
|
||||||
|
|
||||||
@@ -57,16 +57,6 @@ SymbolInfoType = tuple[str, int, str]
|
|||||||
# RAM sections - symbols in these sections consume RAM
|
# RAM sections - symbols in these sections consume RAM
|
||||||
RAM_SECTIONS = frozenset([".data", ".bss"])
|
RAM_SECTIONS = frozenset([".data", ".bss"])
|
||||||
|
|
||||||
# nm symbol types for global/weak defined symbols (used for library symbol mapping)
|
|
||||||
# Only global (uppercase) and weak symbols are safe to use - local symbols (lowercase)
|
|
||||||
# can have name collisions across compilation units
|
|
||||||
_NM_DEFINED_GLOBAL_TYPES = frozenset({"T", "D", "B", "R", "W", "V"})
|
|
||||||
|
|
||||||
# Pattern matching compiler-generated local names that can collide across compilation
|
|
||||||
# units (e.g., packet$19, buf$20, flag$5261). These are unsafe for name-based lookup.
|
|
||||||
# Does NOT match mangled C++ names with optimization suffixes (e.g., func$isra$0).
|
|
||||||
_COMPILER_LOCAL_PATTERN = re.compile(r"^[a-zA-Z_]\w*\$\d+$")
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class MemorySection:
|
class MemorySection:
|
||||||
@@ -101,17 +91,6 @@ class ComponentMemory:
|
|||||||
bss_size: int = 0 # Uninitialized data (ram only)
|
bss_size: int = 0 # Uninitialized data (ram only)
|
||||||
symbol_count: int = 0
|
symbol_count: int = 0
|
||||||
|
|
||||||
def add_section_size(self, section_name: str, size: int) -> None:
|
|
||||||
"""Add size to the appropriate attribute for a section."""
|
|
||||||
if section_name == ".text":
|
|
||||||
self.text_size += size
|
|
||||||
elif section_name == ".rodata":
|
|
||||||
self.rodata_size += size
|
|
||||||
elif section_name == ".data":
|
|
||||||
self.data_size += size
|
|
||||||
elif section_name == ".bss":
|
|
||||||
self.bss_size += size
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def flash_total(self) -> int:
|
def flash_total(self) -> int:
|
||||||
"""Total flash usage (text + rodata + data)."""
|
"""Total flash usage (text + rodata + data)."""
|
||||||
@@ -188,23 +167,12 @@ class MemoryAnalyzer:
|
|||||||
self._elf_symbol_names: set[str] = set()
|
self._elf_symbol_names: set[str] = set()
|
||||||
# SDK symbols not in ELF (static/local symbols from closed-source libs)
|
# SDK symbols not in ELF (static/local symbols from closed-source libs)
|
||||||
self._sdk_symbols: list[SDKSymbol] = []
|
self._sdk_symbols: list[SDKSymbol] = []
|
||||||
# CSWTCH symbols: list of (name, size, source_file, component)
|
|
||||||
self._cswtch_symbols: list[tuple[str, int, str, str]] = []
|
|
||||||
# Library symbol mapping: symbol_name -> library_name
|
|
||||||
self._lib_symbol_map: dict[str, str] = {}
|
|
||||||
# Library dir to name mapping: "lib641" -> "espsoftwareserial",
|
|
||||||
# "espressif__mdns" -> "mdns"
|
|
||||||
self._lib_hash_to_name: dict[str, str] = {}
|
|
||||||
# Heuristic category to library redirect: "mdns_lib" -> "[lib]mdns"
|
|
||||||
self._heuristic_to_lib: dict[str, str] = {}
|
|
||||||
|
|
||||||
def analyze(self) -> dict[str, ComponentMemory]:
|
def analyze(self) -> dict[str, ComponentMemory]:
|
||||||
"""Analyze the ELF file and return component memory usage."""
|
"""Analyze the ELF file and return component memory usage."""
|
||||||
self._parse_sections()
|
self._parse_sections()
|
||||||
self._parse_symbols()
|
self._parse_symbols()
|
||||||
self._scan_libraries()
|
|
||||||
self._categorize_symbols()
|
self._categorize_symbols()
|
||||||
self._analyze_cswtch_symbols()
|
|
||||||
self._analyze_sdk_libraries()
|
self._analyze_sdk_libraries()
|
||||||
return dict(self.components)
|
return dict(self.components)
|
||||||
|
|
||||||
@@ -287,7 +255,8 @@ class MemoryAnalyzer:
|
|||||||
comp_mem.symbol_count += 1
|
comp_mem.symbol_count += 1
|
||||||
|
|
||||||
# Update the appropriate size attribute based on section
|
# Update the appropriate size attribute based on section
|
||||||
comp_mem.add_section_size(section_name, size)
|
if attr_name := SECTION_TO_ATTR.get(section_name):
|
||||||
|
setattr(comp_mem, attr_name, getattr(comp_mem, attr_name) + size)
|
||||||
|
|
||||||
# Track uncategorized symbols
|
# Track uncategorized symbols
|
||||||
if component == "other" and size > 0:
|
if component == "other" and size > 0:
|
||||||
@@ -347,19 +316,15 @@ class MemoryAnalyzer:
|
|||||||
# If no component match found, it's core
|
# If no component match found, it's core
|
||||||
return _COMPONENT_CORE
|
return _COMPONENT_CORE
|
||||||
|
|
||||||
# Check library symbol map (more accurate than heuristic patterns)
|
|
||||||
if lib_name := self._lib_symbol_map.get(symbol_name):
|
|
||||||
return f"{_COMPONENT_PREFIX_LIB}{lib_name}"
|
|
||||||
|
|
||||||
# Check against symbol patterns
|
# Check against symbol patterns
|
||||||
for component, patterns in SYMBOL_PATTERNS.items():
|
for component, patterns in SYMBOL_PATTERNS.items():
|
||||||
if any(pattern in symbol_name for pattern in patterns):
|
if any(pattern in symbol_name for pattern in patterns):
|
||||||
return self._heuristic_to_lib.get(component, component)
|
return component
|
||||||
|
|
||||||
# Check against demangled patterns
|
# Check against demangled patterns
|
||||||
for component, patterns in DEMANGLED_PATTERNS.items():
|
for component, patterns in DEMANGLED_PATTERNS.items():
|
||||||
if any(pattern in demangled for pattern in patterns):
|
if any(pattern in demangled for pattern in patterns):
|
||||||
return self._heuristic_to_lib.get(component, component)
|
return component
|
||||||
|
|
||||||
# Special cases that need more complex logic
|
# Special cases that need more complex logic
|
||||||
|
|
||||||
@@ -407,610 +372,6 @@ class MemoryAnalyzer:
|
|||||||
|
|
||||||
return "Other Core"
|
return "Other Core"
|
||||||
|
|
||||||
def _discover_pio_libraries(
|
|
||||||
self,
|
|
||||||
libraries: dict[str, list[Path]],
|
|
||||||
hash_to_name: dict[str, str],
|
|
||||||
) -> None:
|
|
||||||
"""Discover PlatformIO third-party libraries from the build directory.
|
|
||||||
|
|
||||||
Scans ``lib<hex>/`` directories under ``.pioenvs/<env>/`` to find
|
|
||||||
library names and their ``.a`` archive or ``.o`` file paths.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
libraries: Dict to populate with library name -> file path list mappings.
|
|
||||||
Prefers ``.a`` archives when available, falls back to ``.o`` files
|
|
||||||
(e.g., pioarduino ESP32 Arduino builds only produce ``.o`` files).
|
|
||||||
hash_to_name: Dict to populate with dir name -> library name mappings
|
|
||||||
for CSWTCH attribution (e.g., ``lib641`` -> ``espsoftwareserial``).
|
|
||||||
"""
|
|
||||||
build_dir = self.elf_path.parent
|
|
||||||
|
|
||||||
for entry in build_dir.iterdir():
|
|
||||||
if not entry.is_dir() or not entry.name.startswith("lib"):
|
|
||||||
continue
|
|
||||||
# Validate that the suffix after "lib" is a hex hash
|
|
||||||
hex_part = entry.name[3:]
|
|
||||||
if not hex_part:
|
|
||||||
continue
|
|
||||||
try:
|
|
||||||
int(hex_part, 16)
|
|
||||||
except ValueError:
|
|
||||||
continue
|
|
||||||
|
|
||||||
# Each lib<hex>/ directory contains a subdirectory named after the library
|
|
||||||
for lib_subdir in entry.iterdir():
|
|
||||||
if not lib_subdir.is_dir():
|
|
||||||
continue
|
|
||||||
lib_name = lib_subdir.name.lower()
|
|
||||||
|
|
||||||
# Prefer .a archive (lib<LibraryName>.a), fall back to .o files
|
|
||||||
# e.g., lib72a/ESPAsyncTCP/... has lib72a/libESPAsyncTCP.a
|
|
||||||
archive = entry / f"lib{lib_subdir.name}.a"
|
|
||||||
if archive.exists():
|
|
||||||
file_paths = [archive]
|
|
||||||
elif archives := list(entry.glob("*.a")):
|
|
||||||
# Case-insensitive fallback
|
|
||||||
file_paths = [archives[0]]
|
|
||||||
else:
|
|
||||||
# No .a archive (e.g., pioarduino CMake builds) - use .o files
|
|
||||||
file_paths = sorted(lib_subdir.rglob("*.o"))
|
|
||||||
|
|
||||||
if file_paths:
|
|
||||||
libraries[lib_name] = file_paths
|
|
||||||
hash_to_name[entry.name] = lib_name
|
|
||||||
_LOGGER.debug(
|
|
||||||
"Discovered PlatformIO library: %s -> %s",
|
|
||||||
lib_subdir.name,
|
|
||||||
file_paths[0],
|
|
||||||
)
|
|
||||||
|
|
||||||
def _discover_idf_managed_components(
|
|
||||||
self,
|
|
||||||
libraries: dict[str, list[Path]],
|
|
||||||
hash_to_name: dict[str, str],
|
|
||||||
) -> None:
|
|
||||||
"""Discover ESP-IDF managed component libraries from the build directory.
|
|
||||||
|
|
||||||
ESP-IDF managed components (from the IDF component registry) use a
|
|
||||||
``<vendor>__<name>`` naming convention. Source files live under
|
|
||||||
``managed_components/<vendor>__<name>/`` and the compiled archives are at
|
|
||||||
``esp-idf/<vendor>__<name>/lib<vendor>__<name>.a``.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
libraries: Dict to populate with library name -> file path list mappings.
|
|
||||||
hash_to_name: Dict to populate with dir name -> library name mappings
|
|
||||||
for CSWTCH attribution (e.g., ``espressif__mdns`` -> ``mdns``).
|
|
||||||
"""
|
|
||||||
build_dir = self.elf_path.parent
|
|
||||||
|
|
||||||
managed_dir = build_dir / "managed_components"
|
|
||||||
if not managed_dir.is_dir():
|
|
||||||
return
|
|
||||||
|
|
||||||
espidf_dir = build_dir / "esp-idf"
|
|
||||||
|
|
||||||
for entry in managed_dir.iterdir():
|
|
||||||
if not entry.is_dir() or "__" not in entry.name:
|
|
||||||
continue
|
|
||||||
|
|
||||||
# Extract the short name: espressif__mdns -> mdns
|
|
||||||
full_name = entry.name # e.g., espressif__mdns
|
|
||||||
short_name = full_name.split("__", 1)[1].lower()
|
|
||||||
|
|
||||||
# Find the .a archive under esp-idf/<vendor>__<name>/
|
|
||||||
archive = espidf_dir / full_name / f"lib{full_name}.a"
|
|
||||||
if archive.exists():
|
|
||||||
libraries[short_name] = [archive]
|
|
||||||
hash_to_name[full_name] = short_name
|
|
||||||
_LOGGER.debug(
|
|
||||||
"Discovered IDF managed component: %s -> %s",
|
|
||||||
short_name,
|
|
||||||
archive,
|
|
||||||
)
|
|
||||||
|
|
||||||
def _build_library_symbol_map(
|
|
||||||
self, libraries: dict[str, list[Path]]
|
|
||||||
) -> dict[str, str]:
|
|
||||||
"""Build a symbol-to-library mapping from library archives or object files.
|
|
||||||
|
|
||||||
Runs ``nm --defined-only`` on each ``.a`` or ``.o`` file to collect
|
|
||||||
global and weak defined symbols.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
libraries: Dictionary mapping library name to list of file paths
|
|
||||||
(``.a`` archives or ``.o`` object files).
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Dictionary mapping symbol name to library name.
|
|
||||||
"""
|
|
||||||
symbol_map: dict[str, str] = {}
|
|
||||||
|
|
||||||
if not self.nm_path:
|
|
||||||
return symbol_map
|
|
||||||
|
|
||||||
for lib_name, file_paths in libraries.items():
|
|
||||||
result = run_tool(
|
|
||||||
[self.nm_path, "--defined-only", *(str(p) for p in file_paths)],
|
|
||||||
timeout=10,
|
|
||||||
)
|
|
||||||
if result is None or result.returncode != 0:
|
|
||||||
continue
|
|
||||||
|
|
||||||
for line in result.stdout.splitlines():
|
|
||||||
parts = line.split()
|
|
||||||
if len(parts) < 3:
|
|
||||||
continue
|
|
||||||
|
|
||||||
sym_type = parts[-2]
|
|
||||||
sym_name = parts[-1]
|
|
||||||
|
|
||||||
# Include global defined symbols (uppercase) and weak symbols (W/V)
|
|
||||||
if sym_type in _NM_DEFINED_GLOBAL_TYPES:
|
|
||||||
symbol_map[sym_name] = lib_name
|
|
||||||
|
|
||||||
return symbol_map
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _build_heuristic_to_lib_mapping(
|
|
||||||
library_names: set[str],
|
|
||||||
) -> dict[str, str]:
|
|
||||||
"""Build mapping from heuristic pattern categories to discovered libraries.
|
|
||||||
|
|
||||||
Heuristic categories like ``mdns_lib``, ``web_server_lib``, ``async_tcp``
|
|
||||||
exist as approximations for library attribution. When we discover the
|
|
||||||
actual library, symbols matching those heuristics should be redirected
|
|
||||||
to the ``[lib]`` category instead.
|
|
||||||
|
|
||||||
The mapping is built by checking if the normalized category name
|
|
||||||
(stripped of ``_lib`` suffix and underscores) appears as a substring
|
|
||||||
of any discovered library name.
|
|
||||||
|
|
||||||
Examples::
|
|
||||||
|
|
||||||
mdns_lib -> mdns -> in "mdns" or "esp8266mdns" -> [lib]mdns
|
|
||||||
web_server_lib -> webserver -> in "espasyncwebserver" -> [lib]espasyncwebserver
|
|
||||||
async_tcp -> asynctcp -> in "espasynctcp" -> [lib]espasynctcp
|
|
||||||
|
|
||||||
Args:
|
|
||||||
library_names: Set of discovered library names (lowercase).
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Dictionary mapping heuristic category to ``[lib]<name>`` string.
|
|
||||||
"""
|
|
||||||
mapping: dict[str, str] = {}
|
|
||||||
all_categories = set(SYMBOL_PATTERNS) | set(DEMANGLED_PATTERNS)
|
|
||||||
|
|
||||||
for category in all_categories:
|
|
||||||
base = category.removesuffix("_lib").replace("_", "")
|
|
||||||
# Collect all libraries whose name contains the base string
|
|
||||||
candidates = [lib_name for lib_name in library_names if base in lib_name]
|
|
||||||
if not candidates:
|
|
||||||
continue
|
|
||||||
|
|
||||||
# Choose a deterministic "best" match:
|
|
||||||
# 1. Prefer exact name matches over substring matches.
|
|
||||||
# 2. Among non-exact matches, prefer the shortest library name.
|
|
||||||
# 3. Break remaining ties lexicographically.
|
|
||||||
best_lib = min(
|
|
||||||
candidates,
|
|
||||||
key=lambda lib_name, _base=base: (
|
|
||||||
lib_name != _base,
|
|
||||||
len(lib_name),
|
|
||||||
lib_name,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
mapping[category] = f"{_COMPONENT_PREFIX_LIB}{best_lib}"
|
|
||||||
|
|
||||||
if mapping:
|
|
||||||
_LOGGER.debug(
|
|
||||||
"Heuristic-to-library redirects: %s",
|
|
||||||
", ".join(f"{k} -> {v}" for k, v in sorted(mapping.items())),
|
|
||||||
)
|
|
||||||
|
|
||||||
return mapping
|
|
||||||
|
|
||||||
def _parse_map_file(self) -> dict[str, str] | None:
|
|
||||||
"""Parse linker map file to build authoritative symbol-to-library mapping.
|
|
||||||
|
|
||||||
The linker map file contains the definitive source attribution for every
|
|
||||||
symbol, including local/static ones that ``nm`` cannot safely export.
|
|
||||||
|
|
||||||
Map file format (GNU ld)::
|
|
||||||
|
|
||||||
.text._mdns_service_task
|
|
||||||
0x400e9fdc 0x65c .pioenvs/env/esp-idf/espressif__mdns/libespressif__mdns.a(mdns.c.o)
|
|
||||||
|
|
||||||
Each section entry has a ``.section.symbol_name`` line followed by an
|
|
||||||
indented line with address, size, and source path.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Symbol-to-library dict, or ``None`` if no usable map file exists.
|
|
||||||
"""
|
|
||||||
map_path = self.elf_path.with_suffix(".map")
|
|
||||||
if not map_path.exists() or map_path.stat().st_size < 10000:
|
|
||||||
return None
|
|
||||||
|
|
||||||
_LOGGER.info("Parsing linker map file: %s", map_path.name)
|
|
||||||
|
|
||||||
try:
|
|
||||||
map_text = map_path.read_text(encoding="utf-8", errors="replace")
|
|
||||||
except OSError as err:
|
|
||||||
_LOGGER.warning("Failed to read map file: %s", err)
|
|
||||||
return None
|
|
||||||
|
|
||||||
symbol_map: dict[str, str] = {}
|
|
||||||
current_symbol: str | None = None
|
|
||||||
section_prefixes = (".text.", ".rodata.", ".data.", ".bss.", ".literal.")
|
|
||||||
|
|
||||||
for line in map_text.splitlines():
|
|
||||||
# Match section.symbol line: " .text.symbol_name"
|
|
||||||
# Single space indent, starts with dot
|
|
||||||
if len(line) > 2 and line[0] == " " and line[1] == ".":
|
|
||||||
stripped = line.strip()
|
|
||||||
for prefix in section_prefixes:
|
|
||||||
if stripped.startswith(prefix):
|
|
||||||
current_symbol = stripped[len(prefix) :]
|
|
||||||
break
|
|
||||||
else:
|
|
||||||
current_symbol = None
|
|
||||||
continue
|
|
||||||
|
|
||||||
# Match source attribution line: " 0xADDR 0xSIZE source_path"
|
|
||||||
if current_symbol is None:
|
|
||||||
continue
|
|
||||||
|
|
||||||
fields = line.split()
|
|
||||||
# Skip compiler-generated local names (e.g., packet$19, buf$20)
|
|
||||||
# that can collide across compilation units
|
|
||||||
if (
|
|
||||||
len(fields) >= 3
|
|
||||||
and fields[0].startswith("0x")
|
|
||||||
and fields[1].startswith("0x")
|
|
||||||
and not _COMPILER_LOCAL_PATTERN.match(current_symbol)
|
|
||||||
):
|
|
||||||
source_path = fields[2]
|
|
||||||
# Check if source path contains a known library directory
|
|
||||||
for dir_key, lib_name in self._lib_hash_to_name.items():
|
|
||||||
if dir_key in source_path:
|
|
||||||
symbol_map[current_symbol] = lib_name
|
|
||||||
break
|
|
||||||
|
|
||||||
current_symbol = None
|
|
||||||
|
|
||||||
return symbol_map or None
|
|
||||||
|
|
||||||
def _scan_libraries(self) -> None:
|
|
||||||
"""Discover third-party libraries and build symbol mapping.
|
|
||||||
|
|
||||||
Scans both PlatformIO ``lib<hex>/`` directories (Arduino builds) and
|
|
||||||
ESP-IDF ``managed_components/`` (IDF builds) to find library archives.
|
|
||||||
|
|
||||||
Uses the linker map file for authoritative symbol attribution when
|
|
||||||
available, falling back to ``nm`` scanning with heuristic redirects.
|
|
||||||
"""
|
|
||||||
libraries: dict[str, list[Path]] = {}
|
|
||||||
self._discover_pio_libraries(libraries, self._lib_hash_to_name)
|
|
||||||
self._discover_idf_managed_components(libraries, self._lib_hash_to_name)
|
|
||||||
|
|
||||||
if not libraries:
|
|
||||||
_LOGGER.debug("No third-party libraries found")
|
|
||||||
return
|
|
||||||
|
|
||||||
_LOGGER.info(
|
|
||||||
"Scanning %d libraries: %s",
|
|
||||||
len(libraries),
|
|
||||||
", ".join(sorted(libraries)),
|
|
||||||
)
|
|
||||||
|
|
||||||
# Heuristic redirect catches local symbols (e.g., mdns_task_buffer$14)
|
|
||||||
# that can't be safely added to the symbol map due to name collisions
|
|
||||||
self._heuristic_to_lib = self._build_heuristic_to_lib_mapping(
|
|
||||||
set(libraries.keys())
|
|
||||||
)
|
|
||||||
|
|
||||||
# Try linker map file first (authoritative, includes local symbols)
|
|
||||||
map_symbols = self._parse_map_file()
|
|
||||||
if map_symbols is not None:
|
|
||||||
self._lib_symbol_map = map_symbols
|
|
||||||
_LOGGER.info(
|
|
||||||
"Built library symbol map from linker map: %d symbols",
|
|
||||||
len(self._lib_symbol_map),
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
# Fall back to nm scanning (global symbols only)
|
|
||||||
self._lib_symbol_map = self._build_library_symbol_map(libraries)
|
|
||||||
|
|
||||||
_LOGGER.info(
|
|
||||||
"Built library symbol map from nm: %d symbols from %d libraries",
|
|
||||||
len(self._lib_symbol_map),
|
|
||||||
len(libraries),
|
|
||||||
)
|
|
||||||
|
|
||||||
def _find_object_files_dir(self) -> Path | None:
|
|
||||||
"""Find the directory containing object files for this build.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Path to the directory containing .o files, or None if not found.
|
|
||||||
"""
|
|
||||||
# The ELF is typically at .pioenvs/<env>/firmware.elf
|
|
||||||
# Object files are in .pioenvs/<env>/src/ and .pioenvs/<env>/lib*/
|
|
||||||
pioenvs_dir = self.elf_path.parent
|
|
||||||
if pioenvs_dir.exists() and any(pioenvs_dir.glob("src/*.o")):
|
|
||||||
return pioenvs_dir
|
|
||||||
return None
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _parse_nm_cswtch_output(
|
|
||||||
output: str,
|
|
||||||
base_dir: Path | None,
|
|
||||||
cswtch_map: dict[str, list[tuple[str, int]]],
|
|
||||||
) -> None:
|
|
||||||
"""Parse nm output for CSWTCH symbols and add to cswtch_map.
|
|
||||||
|
|
||||||
Handles both ``.o`` files and ``.a`` archives.
|
|
||||||
|
|
||||||
nm output formats::
|
|
||||||
|
|
||||||
.o files: /path/file.o:hex_addr hex_size type name
|
|
||||||
.a files: /path/lib.a:member.o:hex_addr hex_size type name
|
|
||||||
|
|
||||||
For ``.o`` files, paths are made relative to *base_dir* when possible.
|
|
||||||
For ``.a`` archives (detected by ``:`` in the file portion), paths are
|
|
||||||
formatted as ``archive_stem/member.o`` (e.g. ``liblwip2-536-feat/lwip-esp.o``).
|
|
||||||
|
|
||||||
Args:
|
|
||||||
output: Raw stdout from ``nm --print-file-name -S``.
|
|
||||||
base_dir: Base directory for computing relative paths of ``.o`` files.
|
|
||||||
Pass ``None`` when scanning archives outside the build tree.
|
|
||||||
cswtch_map: Dict to populate, mapping ``"CSWTCH$N:size"`` to source list.
|
|
||||||
"""
|
|
||||||
for line in output.splitlines():
|
|
||||||
if "CSWTCH$" not in line:
|
|
||||||
continue
|
|
||||||
|
|
||||||
# Split on last ":" that precedes a hex address.
|
|
||||||
# For .o: "filepath.o" : "hex_addr hex_size type name"
|
|
||||||
# For .a: "filepath.a:member.o" : "hex_addr hex_size type name"
|
|
||||||
parts_after_colon = line.rsplit(":", 1)
|
|
||||||
if len(parts_after_colon) != 2:
|
|
||||||
continue
|
|
||||||
|
|
||||||
file_path = parts_after_colon[0]
|
|
||||||
fields = parts_after_colon[1].split()
|
|
||||||
# fields: [address, size, type, name]
|
|
||||||
if len(fields) < 4:
|
|
||||||
continue
|
|
||||||
|
|
||||||
sym_name = fields[3]
|
|
||||||
if not sym_name.startswith("CSWTCH$"):
|
|
||||||
continue
|
|
||||||
|
|
||||||
try:
|
|
||||||
size = int(fields[1], 16)
|
|
||||||
except ValueError:
|
|
||||||
continue
|
|
||||||
|
|
||||||
# Determine readable source path
|
|
||||||
# Use ".a:" to detect archive format (not bare ":" which matches
|
|
||||||
# Windows drive letters like "C:\...\file.o").
|
|
||||||
if ".a:" in file_path:
|
|
||||||
# Archive format: "archive.a:member.o" → "archive_stem/member.o"
|
|
||||||
archive_part, member = file_path.rsplit(":", 1)
|
|
||||||
archive_name = Path(archive_part).stem
|
|
||||||
rel_path = f"{archive_name}/{member}"
|
|
||||||
elif base_dir is not None:
|
|
||||||
try:
|
|
||||||
rel_path = str(Path(file_path).relative_to(base_dir))
|
|
||||||
except ValueError:
|
|
||||||
rel_path = file_path
|
|
||||||
else:
|
|
||||||
rel_path = file_path
|
|
||||||
|
|
||||||
key = f"{sym_name}:{size}"
|
|
||||||
cswtch_map[key].append((rel_path, size))
|
|
||||||
|
|
||||||
def _run_nm_cswtch_scan(
|
|
||||||
self,
|
|
||||||
files: list[Path],
|
|
||||||
base_dir: Path | None,
|
|
||||||
cswtch_map: dict[str, list[tuple[str, int]]],
|
|
||||||
) -> None:
|
|
||||||
"""Run nm on *files* and add any CSWTCH symbols to *cswtch_map*.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
files: Object (``.o``) or archive (``.a``) files to scan.
|
|
||||||
base_dir: Base directory for relative path computation (see
|
|
||||||
:meth:`_parse_nm_cswtch_output`).
|
|
||||||
cswtch_map: Dict to populate with results.
|
|
||||||
"""
|
|
||||||
if not self.nm_path or not files:
|
|
||||||
return
|
|
||||||
|
|
||||||
_LOGGER.debug("Scanning %d files for CSWTCH symbols", len(files))
|
|
||||||
|
|
||||||
result = run_tool(
|
|
||||||
[self.nm_path, "--print-file-name", "-S"] + [str(f) for f in files],
|
|
||||||
timeout=30,
|
|
||||||
)
|
|
||||||
if result is None or result.returncode != 0:
|
|
||||||
_LOGGER.debug(
|
|
||||||
"nm failed or timed out scanning %d files for CSWTCH symbols",
|
|
||||||
len(files),
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
self._parse_nm_cswtch_output(result.stdout, base_dir, cswtch_map)
|
|
||||||
|
|
||||||
def _scan_cswtch_in_sdk_archives(
|
|
||||||
self, cswtch_map: dict[str, list[tuple[str, int]]]
|
|
||||||
) -> None:
|
|
||||||
"""Scan SDK library archives (.a) for CSWTCH symbols.
|
|
||||||
|
|
||||||
Prebuilt SDK libraries (e.g. lwip, bearssl) are not compiled from source,
|
|
||||||
so their CSWTCH symbols only exist inside ``.a`` archives. Results are
|
|
||||||
merged into *cswtch_map* for keys not already found in ``.o`` files.
|
|
||||||
|
|
||||||
The same source file (e.g. ``lwip-esp.o``) often appears in multiple
|
|
||||||
library variants (``liblwip2-536.a``, ``liblwip2-1460-feat.a``, etc.),
|
|
||||||
so results are deduplicated by member name.
|
|
||||||
"""
|
|
||||||
sdk_dirs = self._find_sdk_library_dirs()
|
|
||||||
if not sdk_dirs:
|
|
||||||
return
|
|
||||||
|
|
||||||
sdk_archives = sorted(a for sdk_dir in sdk_dirs for a in sdk_dir.glob("*.a"))
|
|
||||||
|
|
||||||
sdk_map: dict[str, list[tuple[str, int]]] = defaultdict(list)
|
|
||||||
self._run_nm_cswtch_scan(sdk_archives, None, sdk_map)
|
|
||||||
|
|
||||||
# Merge SDK results, deduplicating by member name.
|
|
||||||
for key, sources in sdk_map.items():
|
|
||||||
if key in cswtch_map:
|
|
||||||
continue
|
|
||||||
seen: dict[str, tuple[str, int]] = {}
|
|
||||||
for path, sz in sources:
|
|
||||||
member = Path(path).name
|
|
||||||
if member not in seen:
|
|
||||||
seen[member] = (path, sz)
|
|
||||||
cswtch_map[key] = list(seen.values())
|
|
||||||
|
|
||||||
def _source_file_to_component(self, source_file: str) -> str:
|
|
||||||
"""Map a source object file path to its component name.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
source_file: Relative path like 'src/esphome/components/wifi/wifi_component.cpp.o'
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Component name like '[esphome]wifi' or the source file if unknown.
|
|
||||||
"""
|
|
||||||
parts = Path(source_file).parts
|
|
||||||
|
|
||||||
# ESPHome component: src/esphome/components/<name>/...
|
|
||||||
if "components" in parts:
|
|
||||||
idx = parts.index("components")
|
|
||||||
if idx + 1 < len(parts):
|
|
||||||
component_name = parts[idx + 1]
|
|
||||||
if component_name in get_esphome_components():
|
|
||||||
return f"{_COMPONENT_PREFIX_ESPHOME}{component_name}"
|
|
||||||
if component_name in self.external_components:
|
|
||||||
return f"{_COMPONENT_PREFIX_EXTERNAL}{component_name}"
|
|
||||||
|
|
||||||
# ESPHome core: src/esphome/core/... or src/esphome/...
|
|
||||||
if "core" in parts and "esphome" in parts:
|
|
||||||
return _COMPONENT_CORE
|
|
||||||
if "esphome" in parts and "components" not in parts:
|
|
||||||
return _COMPONENT_CORE
|
|
||||||
|
|
||||||
# Framework/library files - check for PlatformIO library hash dirs
|
|
||||||
# e.g., lib65b/ESPAsyncTCP/... -> [lib]espasynctcp
|
|
||||||
if parts and parts[0] in self._lib_hash_to_name:
|
|
||||||
return f"{_COMPONENT_PREFIX_LIB}{self._lib_hash_to_name[parts[0]]}"
|
|
||||||
|
|
||||||
# ESP-IDF managed components: managed_components/espressif__mdns/... -> [lib]mdns
|
|
||||||
if (
|
|
||||||
len(parts) >= 2
|
|
||||||
and parts[0] == "managed_components"
|
|
||||||
and parts[1] in self._lib_hash_to_name
|
|
||||||
):
|
|
||||||
return f"{_COMPONENT_PREFIX_LIB}{self._lib_hash_to_name[parts[1]]}"
|
|
||||||
|
|
||||||
# Other framework/library files - return the first path component
|
|
||||||
# e.g., FrameworkArduino/... -> FrameworkArduino
|
|
||||||
return parts[0] if parts else source_file
|
|
||||||
|
|
||||||
def _analyze_cswtch_symbols(self) -> None:
|
|
||||||
"""Analyze CSWTCH (GCC switch table) symbols by tracing to source objects.
|
|
||||||
|
|
||||||
CSWTCH symbols are compiler-generated lookup tables for switch statements.
|
|
||||||
They are local symbols, so the same name can appear in different object files.
|
|
||||||
This method scans .o files and SDK archives to attribute them to their
|
|
||||||
source components.
|
|
||||||
"""
|
|
||||||
obj_dir = self._find_object_files_dir()
|
|
||||||
if obj_dir is None:
|
|
||||||
_LOGGER.debug("No object files directory found, skipping CSWTCH analysis")
|
|
||||||
return
|
|
||||||
|
|
||||||
# Scan build-dir object files for CSWTCH symbols
|
|
||||||
cswtch_map: dict[str, list[tuple[str, int]]] = defaultdict(list)
|
|
||||||
self._run_nm_cswtch_scan(sorted(obj_dir.rglob("*.o")), obj_dir, cswtch_map)
|
|
||||||
|
|
||||||
# Also scan SDK library archives (.a) for CSWTCH symbols.
|
|
||||||
# Prebuilt SDK libraries (e.g. lwip, bearssl) are not compiled from source
|
|
||||||
# so their symbols only exist inside .a archives, not as loose .o files.
|
|
||||||
self._scan_cswtch_in_sdk_archives(cswtch_map)
|
|
||||||
|
|
||||||
if not cswtch_map:
|
|
||||||
_LOGGER.debug("No CSWTCH symbols found in object files or SDK archives")
|
|
||||||
return
|
|
||||||
|
|
||||||
# Collect CSWTCH symbols from the ELF (already parsed in sections)
|
|
||||||
# Include section_name for re-attribution of component totals
|
|
||||||
elf_cswtch = [
|
|
||||||
(symbol_name, size, section_name)
|
|
||||||
for section_name, section in self.sections.items()
|
|
||||||
for symbol_name, size, _ in section.symbols
|
|
||||||
if symbol_name.startswith("CSWTCH$")
|
|
||||||
]
|
|
||||||
|
|
||||||
_LOGGER.debug(
|
|
||||||
"Found %d CSWTCH symbols in ELF, %d unique in object files",
|
|
||||||
len(elf_cswtch),
|
|
||||||
len(cswtch_map),
|
|
||||||
)
|
|
||||||
|
|
||||||
# Match ELF CSWTCH symbols to source files and re-attribute component totals.
|
|
||||||
# _categorize_symbols() already ran and put these into "other" since CSWTCH$
|
|
||||||
# names don't match any component pattern. We move the bytes to the correct
|
|
||||||
# component based on the object file mapping.
|
|
||||||
other_mem = self.components.get("other")
|
|
||||||
|
|
||||||
for sym_name, size, section_name in elf_cswtch:
|
|
||||||
key = f"{sym_name}:{size}"
|
|
||||||
sources = cswtch_map.get(key, [])
|
|
||||||
|
|
||||||
if len(sources) == 1:
|
|
||||||
source_file = sources[0][0]
|
|
||||||
component = self._source_file_to_component(source_file)
|
|
||||||
elif len(sources) > 1:
|
|
||||||
# Ambiguous - multiple object files have same CSWTCH name+size
|
|
||||||
source_file = "ambiguous"
|
|
||||||
component = "ambiguous"
|
|
||||||
_LOGGER.debug(
|
|
||||||
"Ambiguous CSWTCH %s (%d B) found in %d files: %s",
|
|
||||||
sym_name,
|
|
||||||
size,
|
|
||||||
len(sources),
|
|
||||||
", ".join(src for src, _ in sources),
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
source_file = "unknown"
|
|
||||||
component = "unknown"
|
|
||||||
|
|
||||||
self._cswtch_symbols.append((sym_name, size, source_file, component))
|
|
||||||
|
|
||||||
# Re-attribute from "other" to the correct component
|
|
||||||
if (
|
|
||||||
component not in ("other", "unknown", "ambiguous")
|
|
||||||
and other_mem is not None
|
|
||||||
):
|
|
||||||
other_mem.add_section_size(section_name, -size)
|
|
||||||
if component not in self.components:
|
|
||||||
self.components[component] = ComponentMemory(component)
|
|
||||||
self.components[component].add_section_size(section_name, size)
|
|
||||||
|
|
||||||
# Sort by size descending
|
|
||||||
self._cswtch_symbols.sort(key=lambda x: x[1], reverse=True)
|
|
||||||
|
|
||||||
total_size = sum(size for _, size, _, _ in self._cswtch_symbols)
|
|
||||||
_LOGGER.debug(
|
|
||||||
"CSWTCH analysis: %d symbols, %d bytes total",
|
|
||||||
len(self._cswtch_symbols),
|
|
||||||
total_size,
|
|
||||||
)
|
|
||||||
|
|
||||||
def get_unattributed_ram(self) -> tuple[int, int, int]:
|
def get_unattributed_ram(self) -> tuple[int, int, int]:
|
||||||
"""Get unattributed RAM sizes (SDK/framework overhead).
|
"""Get unattributed RAM sizes (SDK/framework overhead).
|
||||||
|
|
||||||
|
|||||||
@@ -4,8 +4,6 @@ from __future__ import annotations
|
|||||||
|
|
||||||
from collections import defaultdict
|
from collections import defaultdict
|
||||||
from collections.abc import Callable
|
from collections.abc import Callable
|
||||||
import heapq
|
|
||||||
from operator import itemgetter
|
|
||||||
import sys
|
import sys
|
||||||
from typing import TYPE_CHECKING
|
from typing import TYPE_CHECKING
|
||||||
|
|
||||||
@@ -14,7 +12,6 @@ from . import (
|
|||||||
_COMPONENT_CORE,
|
_COMPONENT_CORE,
|
||||||
_COMPONENT_PREFIX_ESPHOME,
|
_COMPONENT_PREFIX_ESPHOME,
|
||||||
_COMPONENT_PREFIX_EXTERNAL,
|
_COMPONENT_PREFIX_EXTERNAL,
|
||||||
_COMPONENT_PREFIX_LIB,
|
|
||||||
RAM_SECTIONS,
|
RAM_SECTIONS,
|
||||||
MemoryAnalyzer,
|
MemoryAnalyzer,
|
||||||
)
|
)
|
||||||
@@ -32,10 +29,6 @@ class MemoryAnalyzerCLI(MemoryAnalyzer):
|
|||||||
)
|
)
|
||||||
# Lower threshold for RAM symbols (RAM is more constrained)
|
# Lower threshold for RAM symbols (RAM is more constrained)
|
||||||
RAM_SYMBOL_SIZE_THRESHOLD: int = 24
|
RAM_SYMBOL_SIZE_THRESHOLD: int = 24
|
||||||
# Number of top symbols to show in the largest symbols report
|
|
||||||
TOP_SYMBOLS_LIMIT: int = 30
|
|
||||||
# Width for symbol name display in top symbols report
|
|
||||||
COL_TOP_SYMBOL_NAME: int = 55
|
|
||||||
|
|
||||||
# Column width constants
|
# Column width constants
|
||||||
COL_COMPONENT: int = 29
|
COL_COMPONENT: int = 29
|
||||||
@@ -154,83 +147,6 @@ class MemoryAnalyzerCLI(MemoryAnalyzer):
|
|||||||
section_label = f" [{section[1:]}]" # .data -> [data], .bss -> [bss]
|
section_label = f" [{section[1:]}]" # .data -> [data], .bss -> [bss]
|
||||||
return f"{demangled} ({size:,} B){section_label}"
|
return f"{demangled} ({size:,} B){section_label}"
|
||||||
|
|
||||||
def _add_top_symbols(self, lines: list[str]) -> None:
|
|
||||||
"""Add a section showing the top largest symbols in the binary."""
|
|
||||||
# Collect all symbols from all components: (symbol, demangled, size, section, component)
|
|
||||||
all_symbols = [
|
|
||||||
(symbol, demangled, size, section, component)
|
|
||||||
for component, symbols in self._component_symbols.items()
|
|
||||||
for symbol, demangled, size, section in symbols
|
|
||||||
]
|
|
||||||
|
|
||||||
# Get top N symbols by size using heapq for efficiency
|
|
||||||
top_symbols = heapq.nlargest(
|
|
||||||
self.TOP_SYMBOLS_LIMIT, all_symbols, key=itemgetter(2)
|
|
||||||
)
|
|
||||||
|
|
||||||
lines.append("")
|
|
||||||
lines.append(f"Top {self.TOP_SYMBOLS_LIMIT} Largest Symbols:")
|
|
||||||
# Calculate truncation limit from column width (leaving room for "...")
|
|
||||||
truncate_limit = self.COL_TOP_SYMBOL_NAME - 3
|
|
||||||
for i, (_, demangled, size, section, component) in enumerate(top_symbols):
|
|
||||||
# Format section label
|
|
||||||
section_label = f"[{section[1:]}]" if section else ""
|
|
||||||
# Truncate demangled name if too long
|
|
||||||
demangled_display = (
|
|
||||||
f"{demangled[:truncate_limit]}..."
|
|
||||||
if len(demangled) > self.COL_TOP_SYMBOL_NAME
|
|
||||||
else demangled
|
|
||||||
)
|
|
||||||
lines.append(
|
|
||||||
f"{i + 1:>2}. {size:>7,} B {section_label:<8} {demangled_display:<{self.COL_TOP_SYMBOL_NAME}} {component}"
|
|
||||||
)
|
|
||||||
|
|
||||||
def _add_cswtch_analysis(self, lines: list[str]) -> None:
|
|
||||||
"""Add CSWTCH (GCC switch table lookup) analysis section."""
|
|
||||||
self._add_section_header(lines, "CSWTCH Analysis (GCC Switch Table Lookups)")
|
|
||||||
|
|
||||||
total_size = sum(size for _, size, _, _ in self._cswtch_symbols)
|
|
||||||
lines.append(
|
|
||||||
f"Total: {len(self._cswtch_symbols)} switch table(s), {total_size:,} B"
|
|
||||||
)
|
|
||||||
lines.append("")
|
|
||||||
|
|
||||||
# Group by component
|
|
||||||
by_component: dict[str, list[tuple[str, int, str]]] = defaultdict(list)
|
|
||||||
for sym_name, size, source_file, component in self._cswtch_symbols:
|
|
||||||
by_component[component].append((sym_name, size, source_file))
|
|
||||||
|
|
||||||
# Sort components by total size descending
|
|
||||||
sorted_components = sorted(
|
|
||||||
by_component.items(),
|
|
||||||
key=lambda x: sum(s[1] for s in x[1]),
|
|
||||||
reverse=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
for component, symbols in sorted_components:
|
|
||||||
comp_total = sum(s[1] for s in symbols)
|
|
||||||
lines.append(f"{component} ({comp_total:,} B, {len(symbols)} tables):")
|
|
||||||
|
|
||||||
# Group by source file within component
|
|
||||||
by_file: dict[str, list[tuple[str, int]]] = defaultdict(list)
|
|
||||||
for sym_name, size, source_file in symbols:
|
|
||||||
by_file[source_file].append((sym_name, size))
|
|
||||||
|
|
||||||
for source_file, file_symbols in sorted(
|
|
||||||
by_file.items(),
|
|
||||||
key=lambda x: sum(s[1] for s in x[1]),
|
|
||||||
reverse=True,
|
|
||||||
):
|
|
||||||
file_total = sum(s[1] for s in file_symbols)
|
|
||||||
lines.append(
|
|
||||||
f" {source_file} ({file_total:,} B, {len(file_symbols)} tables)"
|
|
||||||
)
|
|
||||||
for sym_name, size in sorted(
|
|
||||||
file_symbols, key=lambda x: x[1], reverse=True
|
|
||||||
):
|
|
||||||
lines.append(f" {size:>6,} B {sym_name}")
|
|
||||||
lines.append("")
|
|
||||||
|
|
||||||
def generate_report(self, detailed: bool = False) -> str:
|
def generate_report(self, detailed: bool = False) -> str:
|
||||||
"""Generate a formatted memory report."""
|
"""Generate a formatted memory report."""
|
||||||
components = sorted(
|
components = sorted(
|
||||||
@@ -332,9 +248,6 @@ class MemoryAnalyzerCLI(MemoryAnalyzer):
|
|||||||
"RAM",
|
"RAM",
|
||||||
)
|
)
|
||||||
|
|
||||||
# Top largest symbols in the binary
|
|
||||||
self._add_top_symbols(lines)
|
|
||||||
|
|
||||||
# Add ESPHome core detailed analysis if there are core symbols
|
# Add ESPHome core detailed analysis if there are core symbols
|
||||||
if self._esphome_core_symbols:
|
if self._esphome_core_symbols:
|
||||||
self._add_section_header(lines, f"{_COMPONENT_CORE} Detailed Analysis")
|
self._add_section_header(lines, f"{_COMPONENT_CORE} Detailed Analysis")
|
||||||
@@ -408,11 +321,6 @@ class MemoryAnalyzerCLI(MemoryAnalyzer):
|
|||||||
for name, mem in components
|
for name, mem in components
|
||||||
if name.startswith(_COMPONENT_PREFIX_EXTERNAL)
|
if name.startswith(_COMPONENT_PREFIX_EXTERNAL)
|
||||||
]
|
]
|
||||||
library_components = [
|
|
||||||
(name, mem)
|
|
||||||
for name, mem in components
|
|
||||||
if name.startswith(_COMPONENT_PREFIX_LIB)
|
|
||||||
]
|
|
||||||
|
|
||||||
top_esphome_components = sorted(
|
top_esphome_components = sorted(
|
||||||
esphome_components, key=lambda x: x[1].flash_total, reverse=True
|
esphome_components, key=lambda x: x[1].flash_total, reverse=True
|
||||||
@@ -423,11 +331,6 @@ class MemoryAnalyzerCLI(MemoryAnalyzer):
|
|||||||
external_components, key=lambda x: x[1].flash_total, reverse=True
|
external_components, key=lambda x: x[1].flash_total, reverse=True
|
||||||
)
|
)
|
||||||
|
|
||||||
# Include all library components
|
|
||||||
top_library_components = sorted(
|
|
||||||
library_components, key=lambda x: x[1].flash_total, reverse=True
|
|
||||||
)
|
|
||||||
|
|
||||||
# Check if API component exists and ensure it's included
|
# Check if API component exists and ensure it's included
|
||||||
api_component = None
|
api_component = None
|
||||||
for name, mem in components:
|
for name, mem in components:
|
||||||
@@ -446,11 +349,10 @@ class MemoryAnalyzerCLI(MemoryAnalyzer):
|
|||||||
if name in system_components_to_include
|
if name in system_components_to_include
|
||||||
]
|
]
|
||||||
|
|
||||||
# Combine all components to analyze: top ESPHome + all external + libraries + API if not already included + system components
|
# Combine all components to analyze: top ESPHome + all external + API if not already included + system components
|
||||||
components_to_analyze = (
|
components_to_analyze = (
|
||||||
list(top_esphome_components)
|
list(top_esphome_components)
|
||||||
+ list(top_external_components)
|
+ list(top_external_components)
|
||||||
+ list(top_library_components)
|
|
||||||
+ system_components
|
+ system_components
|
||||||
)
|
)
|
||||||
if api_component and api_component not in components_to_analyze:
|
if api_component and api_component not in components_to_analyze:
|
||||||
@@ -529,10 +431,6 @@ class MemoryAnalyzerCLI(MemoryAnalyzer):
|
|||||||
lines.append(f" ... and {len(large_ram_syms) - 10} more")
|
lines.append(f" ... and {len(large_ram_syms) - 10} more")
|
||||||
lines.append("")
|
lines.append("")
|
||||||
|
|
||||||
# CSWTCH (GCC switch table) analysis
|
|
||||||
if self._cswtch_symbols:
|
|
||||||
self._add_cswtch_analysis(lines)
|
|
||||||
|
|
||||||
lines.append(
|
lines.append(
|
||||||
"Note: This analysis covers symbols in the ELF file. Some runtime allocations may not be included."
|
"Note: This analysis covers symbols in the ELF file. Some runtime allocations may not be included."
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -66,6 +66,15 @@ SECTION_MAPPING = {
|
|||||||
),
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Section to ComponentMemory attribute mapping
|
||||||
|
# Maps section names to the attribute name in ComponentMemory dataclass
|
||||||
|
SECTION_TO_ATTR = {
|
||||||
|
".text": "text_size",
|
||||||
|
".rodata": "rodata_size",
|
||||||
|
".data": "data_size",
|
||||||
|
".bss": "bss_size",
|
||||||
|
}
|
||||||
|
|
||||||
# Component identification rules
|
# Component identification rules
|
||||||
# Symbol patterns: patterns found in raw symbol names
|
# Symbol patterns: patterns found in raw symbol names
|
||||||
SYMBOL_PATTERNS = {
|
SYMBOL_PATTERNS = {
|
||||||
@@ -504,9 +513,7 @@ SYMBOL_PATTERNS = {
|
|||||||
"__FUNCTION__$",
|
"__FUNCTION__$",
|
||||||
"DAYS_IN_MONTH",
|
"DAYS_IN_MONTH",
|
||||||
"_DAYS_BEFORE_MONTH",
|
"_DAYS_BEFORE_MONTH",
|
||||||
# Note: CSWTCH$ symbols are GCC switch table lookup tables.
|
"CSWTCH$",
|
||||||
# They are attributed to their source object files via _analyze_cswtch_symbols()
|
|
||||||
# rather than being lumped into libc.
|
|
||||||
"dst$",
|
"dst$",
|
||||||
"sulp",
|
"sulp",
|
||||||
"_strtol_l", # String to long with locale
|
"_strtol_l", # String to long with locale
|
||||||
|
|||||||
@@ -87,7 +87,6 @@ from esphome.cpp_types import ( # noqa: F401
|
|||||||
size_t,
|
size_t,
|
||||||
std_ns,
|
std_ns,
|
||||||
std_shared_ptr,
|
std_shared_ptr,
|
||||||
std_span,
|
|
||||||
std_string,
|
std_string,
|
||||||
std_string_ref,
|
std_string_ref,
|
||||||
std_vector,
|
std_vector,
|
||||||
|
|||||||
@@ -45,6 +45,8 @@ void AbsoluteHumidityComponent::dump_config() {
|
|||||||
this->temperature_sensor_->get_name().c_str(), this->humidity_sensor_->get_name().c_str());
|
this->temperature_sensor_->get_name().c_str(), this->humidity_sensor_->get_name().c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float AbsoluteHumidityComponent::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void AbsoluteHumidityComponent::loop() {
|
void AbsoluteHumidityComponent::loop() {
|
||||||
if (!this->next_update_) {
|
if (!this->next_update_) {
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ class AbsoluteHumidityComponent : public sensor::Sensor, public Component {
|
|||||||
|
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void loop() override;
|
void loop() override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|||||||
@@ -68,6 +68,11 @@ class ADCSensor : public sensor::Sensor, public PollingComponent, public voltage
|
|||||||
/// This method is called during the ESPHome setup process to log the configuration.
|
/// This method is called during the ESPHome setup process to log the configuration.
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
|
||||||
|
/// Return the setup priority for this component.
|
||||||
|
/// Components with higher priority are initialized earlier during setup.
|
||||||
|
/// @return A float representing the setup priority.
|
||||||
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
#ifdef USE_ZEPHYR
|
#ifdef USE_ZEPHYR
|
||||||
/// Set the ADC channel to be used by the ADC sensor.
|
/// Set the ADC channel to be used by the ADC sensor.
|
||||||
/// @param channel Pointer to an adc_dt_spec structure representing the ADC channel.
|
/// @param channel Pointer to an adc_dt_spec structure representing the ADC channel.
|
||||||
|
|||||||
@@ -79,5 +79,7 @@ void ADCSensor::set_sample_count(uint8_t sample_count) {
|
|||||||
|
|
||||||
void ADCSensor::set_sampling_mode(SamplingMode sampling_mode) { this->sampling_mode_ = sampling_mode; }
|
void ADCSensor::set_sampling_mode(SamplingMode sampling_mode) { this->sampling_mode_ = sampling_mode; }
|
||||||
|
|
||||||
|
float ADCSensor::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
} // namespace adc
|
} // namespace adc
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -42,11 +42,11 @@ void ADCSensor::setup() {
|
|||||||
adc_oneshot_unit_init_cfg_t init_config = {}; // Zero initialize
|
adc_oneshot_unit_init_cfg_t init_config = {}; // Zero initialize
|
||||||
init_config.unit_id = this->adc_unit_;
|
init_config.unit_id = this->adc_unit_;
|
||||||
init_config.ulp_mode = ADC_ULP_MODE_DISABLE;
|
init_config.ulp_mode = ADC_ULP_MODE_DISABLE;
|
||||||
#if USE_ESP32_VARIANT_ESP32C2 || USE_ESP32_VARIANT_ESP32C3 || USE_ESP32_VARIANT_ESP32C5 || \
|
#if USE_ESP32_VARIANT_ESP32C3 || USE_ESP32_VARIANT_ESP32C5 || USE_ESP32_VARIANT_ESP32C6 || \
|
||||||
USE_ESP32_VARIANT_ESP32C6 || USE_ESP32_VARIANT_ESP32C61 || USE_ESP32_VARIANT_ESP32H2
|
USE_ESP32_VARIANT_ESP32C61 || USE_ESP32_VARIANT_ESP32H2
|
||||||
init_config.clk_src = ADC_DIGI_CLK_SRC_DEFAULT;
|
init_config.clk_src = ADC_DIGI_CLK_SRC_DEFAULT;
|
||||||
#endif // USE_ESP32_VARIANT_ESP32C2 || USE_ESP32_VARIANT_ESP32C3 || USE_ESP32_VARIANT_ESP32C5 ||
|
#endif // USE_ESP32_VARIANT_ESP32C3 || USE_ESP32_VARIANT_ESP32C5 || USE_ESP32_VARIANT_ESP32C6 ||
|
||||||
// USE_ESP32_VARIANT_ESP32C6 || USE_ESP32_VARIANT_ESP32C61 || USE_ESP32_VARIANT_ESP32H2
|
// USE_ESP32_VARIANT_ESP32C61 || USE_ESP32_VARIANT_ESP32H2
|
||||||
esp_err_t err = adc_oneshot_new_unit(&init_config, &ADCSensor::shared_adc_handles[this->adc_unit_]);
|
esp_err_t err = adc_oneshot_new_unit(&init_config, &ADCSensor::shared_adc_handles[this->adc_unit_]);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
ESP_LOGE(TAG, "Error initializing %s: %d", LOG_STR_ARG(adc_unit_to_str(this->adc_unit_)), err);
|
ESP_LOGE(TAG, "Error initializing %s: %d", LOG_STR_ARG(adc_unit_to_str(this->adc_unit_)), err);
|
||||||
@@ -76,7 +76,7 @@ void ADCSensor::setup() {
|
|||||||
|
|
||||||
#if USE_ESP32_VARIANT_ESP32C3 || USE_ESP32_VARIANT_ESP32C5 || USE_ESP32_VARIANT_ESP32C6 || \
|
#if USE_ESP32_VARIANT_ESP32C3 || USE_ESP32_VARIANT_ESP32C5 || USE_ESP32_VARIANT_ESP32C6 || \
|
||||||
USE_ESP32_VARIANT_ESP32C61 || USE_ESP32_VARIANT_ESP32H2 || USE_ESP32_VARIANT_ESP32P4 || USE_ESP32_VARIANT_ESP32S3
|
USE_ESP32_VARIANT_ESP32C61 || USE_ESP32_VARIANT_ESP32H2 || USE_ESP32_VARIANT_ESP32P4 || USE_ESP32_VARIANT_ESP32S3
|
||||||
// RISC-V variants (except C2) and S3 use curve fitting calibration
|
// RISC-V variants and S3 use curve fitting calibration
|
||||||
adc_cali_curve_fitting_config_t cali_config = {}; // Zero initialize first
|
adc_cali_curve_fitting_config_t cali_config = {}; // Zero initialize first
|
||||||
#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 3, 0)
|
#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 3, 0)
|
||||||
cali_config.chan = this->channel_;
|
cali_config.chan = this->channel_;
|
||||||
@@ -94,14 +94,14 @@ void ADCSensor::setup() {
|
|||||||
ESP_LOGW(TAG, "Curve fitting calibration failed with error %d, will use uncalibrated readings", err);
|
ESP_LOGW(TAG, "Curve fitting calibration failed with error %d, will use uncalibrated readings", err);
|
||||||
this->setup_flags_.calibration_complete = false;
|
this->setup_flags_.calibration_complete = false;
|
||||||
}
|
}
|
||||||
#else // ESP32, ESP32-S2, and ESP32-C2 use line fitting calibration
|
#else // Other ESP32 variants use line fitting calibration
|
||||||
adc_cali_line_fitting_config_t cali_config = {
|
adc_cali_line_fitting_config_t cali_config = {
|
||||||
.unit_id = this->adc_unit_,
|
.unit_id = this->adc_unit_,
|
||||||
.atten = this->attenuation_,
|
.atten = this->attenuation_,
|
||||||
.bitwidth = ADC_BITWIDTH_DEFAULT,
|
.bitwidth = ADC_BITWIDTH_DEFAULT,
|
||||||
#if !defined(USE_ESP32_VARIANT_ESP32S2) && !defined(USE_ESP32_VARIANT_ESP32C2)
|
#if !defined(USE_ESP32_VARIANT_ESP32S2)
|
||||||
.default_vref = 1100, // Default reference voltage in mV
|
.default_vref = 1100, // Default reference voltage in mV
|
||||||
#endif // !defined(USE_ESP32_VARIANT_ESP32S2) && !defined(USE_ESP32_VARIANT_ESP32C2)
|
#endif // !defined(USE_ESP32_VARIANT_ESP32S2)
|
||||||
};
|
};
|
||||||
err = adc_cali_create_scheme_line_fitting(&cali_config, &handle);
|
err = adc_cali_create_scheme_line_fitting(&cali_config, &handle);
|
||||||
if (err == ESP_OK) {
|
if (err == ESP_OK) {
|
||||||
@@ -112,7 +112,7 @@ void ADCSensor::setup() {
|
|||||||
ESP_LOGW(TAG, "Line fitting calibration failed with error %d, will use uncalibrated readings", err);
|
ESP_LOGW(TAG, "Line fitting calibration failed with error %d, will use uncalibrated readings", err);
|
||||||
this->setup_flags_.calibration_complete = false;
|
this->setup_flags_.calibration_complete = false;
|
||||||
}
|
}
|
||||||
#endif // ESP32C3 || ESP32C5 || ESP32C6 || ESP32C61 || ESP32H2 || ESP32P4 || ESP32S3
|
#endif // USE_ESP32_VARIANT_ESP32C3 || ESP32C5 || ESP32C6 || ESP32C61 || ESP32H2 || ESP32P4 || ESP32S3
|
||||||
}
|
}
|
||||||
|
|
||||||
this->setup_flags_.init_complete = true;
|
this->setup_flags_.init_complete = true;
|
||||||
@@ -189,7 +189,7 @@ float ADCSensor::sample_fixed_attenuation_() {
|
|||||||
adc_cali_delete_scheme_curve_fitting(this->calibration_handle_);
|
adc_cali_delete_scheme_curve_fitting(this->calibration_handle_);
|
||||||
#else // Other ESP32 variants use line fitting calibration
|
#else // Other ESP32 variants use line fitting calibration
|
||||||
adc_cali_delete_scheme_line_fitting(this->calibration_handle_);
|
adc_cali_delete_scheme_line_fitting(this->calibration_handle_);
|
||||||
#endif // ESP32C3 || ESP32C5 || ESP32C6 || ESP32C61 || ESP32H2 || ESP32P4 || ESP32S3
|
#endif // USE_ESP32_VARIANT_ESP32C3 || ESP32C5 || ESP32C6 || ESP32C61 || ESP32H2 || ESP32P4 || ESP32S3
|
||||||
this->calibration_handle_ = nullptr;
|
this->calibration_handle_ = nullptr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -247,7 +247,7 @@ float ADCSensor::sample_autorange_() {
|
|||||||
.unit_id = this->adc_unit_,
|
.unit_id = this->adc_unit_,
|
||||||
.atten = atten,
|
.atten = atten,
|
||||||
.bitwidth = ADC_BITWIDTH_DEFAULT,
|
.bitwidth = ADC_BITWIDTH_DEFAULT,
|
||||||
#if !defined(USE_ESP32_VARIANT_ESP32S2) && !defined(USE_ESP32_VARIANT_ESP32C2)
|
#if !defined(USE_ESP32_VARIANT_ESP32S2)
|
||||||
.default_vref = 1100,
|
.default_vref = 1100,
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -9,6 +9,8 @@ static const char *const TAG = "adc128s102.sensor";
|
|||||||
|
|
||||||
ADC128S102Sensor::ADC128S102Sensor(uint8_t channel) : channel_(channel) {}
|
ADC128S102Sensor::ADC128S102Sensor(uint8_t channel) : channel_(channel) {}
|
||||||
|
|
||||||
|
float ADC128S102Sensor::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void ADC128S102Sensor::dump_config() {
|
void ADC128S102Sensor::dump_config() {
|
||||||
LOG_SENSOR("", "ADC128S102 Sensor", this);
|
LOG_SENSOR("", "ADC128S102 Sensor", this);
|
||||||
ESP_LOGCONFIG(TAG, " Pin: %u", this->channel_);
|
ESP_LOGCONFIG(TAG, " Pin: %u", this->channel_);
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ class ADC128S102Sensor : public PollingComponent,
|
|||||||
|
|
||||||
void update() override;
|
void update() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
float sample() override;
|
float sample() override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|||||||
@@ -150,6 +150,8 @@ void AHT10Component::update() {
|
|||||||
this->restart_read_();
|
this->restart_read_();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float AHT10Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void AHT10Component::dump_config() {
|
void AHT10Component::dump_config() {
|
||||||
ESP_LOGCONFIG(TAG, "AHT10:");
|
ESP_LOGCONFIG(TAG, "AHT10:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ class AHT10Component : public PollingComponent, public i2c::I2CDevice {
|
|||||||
void setup() override;
|
void setup() override;
|
||||||
void update() override;
|
void update() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void set_variant(AHT10Variant variant) { this->variant_ = variant; }
|
void set_variant(AHT10Variant variant) { this->variant_ = variant; }
|
||||||
|
|
||||||
void set_temperature_sensor(sensor::Sensor *temperature_sensor) { temperature_sensor_ = temperature_sensor; }
|
void set_temperature_sensor(sensor::Sensor *temperature_sensor) { temperature_sensor_ = temperature_sensor; }
|
||||||
|
|||||||
@@ -1,15 +1,32 @@
|
|||||||
#include "alarm_control_panel_state.h"
|
#include "alarm_control_panel_state.h"
|
||||||
#include "esphome/core/progmem.h"
|
|
||||||
|
|
||||||
namespace esphome::alarm_control_panel {
|
namespace esphome::alarm_control_panel {
|
||||||
|
|
||||||
// Alarm control panel state strings indexed by AlarmControlPanelState enum (0-9)
|
|
||||||
PROGMEM_STRING_TABLE(AlarmControlPanelStateStrings, "DISARMED", "ARMED_HOME", "ARMED_AWAY", "ARMED_NIGHT",
|
|
||||||
"ARMED_VACATION", "ARMED_CUSTOM_BYPASS", "PENDING", "ARMING", "DISARMING", "TRIGGERED", "UNKNOWN");
|
|
||||||
|
|
||||||
const LogString *alarm_control_panel_state_to_string(AlarmControlPanelState state) {
|
const LogString *alarm_control_panel_state_to_string(AlarmControlPanelState state) {
|
||||||
return AlarmControlPanelStateStrings::get_log_str(static_cast<uint8_t>(state),
|
switch (state) {
|
||||||
AlarmControlPanelStateStrings::LAST_INDEX);
|
case ACP_STATE_DISARMED:
|
||||||
|
return LOG_STR("DISARMED");
|
||||||
|
case ACP_STATE_ARMED_HOME:
|
||||||
|
return LOG_STR("ARMED_HOME");
|
||||||
|
case ACP_STATE_ARMED_AWAY:
|
||||||
|
return LOG_STR("ARMED_AWAY");
|
||||||
|
case ACP_STATE_ARMED_NIGHT:
|
||||||
|
return LOG_STR("ARMED_NIGHT");
|
||||||
|
case ACP_STATE_ARMED_VACATION:
|
||||||
|
return LOG_STR("ARMED_VACATION");
|
||||||
|
case ACP_STATE_ARMED_CUSTOM_BYPASS:
|
||||||
|
return LOG_STR("ARMED_CUSTOM_BYPASS");
|
||||||
|
case ACP_STATE_PENDING:
|
||||||
|
return LOG_STR("PENDING");
|
||||||
|
case ACP_STATE_ARMING:
|
||||||
|
return LOG_STR("ARMING");
|
||||||
|
case ACP_STATE_DISARMING:
|
||||||
|
return LOG_STR("DISARMING");
|
||||||
|
case ACP_STATE_TRIGGERED:
|
||||||
|
return LOG_STR("TRIGGERED");
|
||||||
|
default:
|
||||||
|
return LOG_STR("UNKNOWN");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace esphome::alarm_control_panel
|
} // namespace esphome::alarm_control_panel
|
||||||
|
|||||||
@@ -176,5 +176,7 @@ void AM2315C::dump_config() {
|
|||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float AM2315C::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
} // namespace am2315c
|
} // namespace am2315c
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -33,6 +33,7 @@ class AM2315C : public PollingComponent, public i2c::I2CDevice {
|
|||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
void update() override;
|
void update() override;
|
||||||
void setup() override;
|
void setup() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
void set_temperature_sensor(sensor::Sensor *temperature_sensor) { this->temperature_sensor_ = temperature_sensor; }
|
void set_temperature_sensor(sensor::Sensor *temperature_sensor) { this->temperature_sensor_ = temperature_sensor; }
|
||||||
void set_humidity_sensor(sensor::Sensor *humidity_sensor) { this->humidity_sensor_ = humidity_sensor; }
|
void set_humidity_sensor(sensor::Sensor *humidity_sensor) { this->humidity_sensor_ = humidity_sensor; }
|
||||||
|
|||||||
@@ -51,6 +51,7 @@ void AM2320Component::dump_config() {
|
|||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
}
|
}
|
||||||
|
float AM2320Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
bool AM2320Component::read_bytes_(uint8_t a_register, uint8_t *data, uint8_t len, uint32_t conversion) {
|
bool AM2320Component::read_bytes_(uint8_t a_register, uint8_t *data, uint8_t len, uint32_t conversion) {
|
||||||
if (!this->write_bytes(a_register, data, 2)) {
|
if (!this->write_bytes(a_register, data, 2)) {
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ class AM2320Component : public PollingComponent, public i2c::I2CDevice {
|
|||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void update() override;
|
void update() override;
|
||||||
|
|
||||||
void set_temperature_sensor(sensor::Sensor *temperature_sensor) { temperature_sensor_ = temperature_sensor; }
|
void set_temperature_sensor(sensor::Sensor *temperature_sensor) { temperature_sensor_ = temperature_sensor; }
|
||||||
|
|||||||
@@ -384,6 +384,7 @@ void APDS9960::process_dataset_(int up, int down, int left, int right) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
float APDS9960::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
bool APDS9960::is_proximity_enabled_() const {
|
bool APDS9960::is_proximity_enabled_() const {
|
||||||
return
|
return
|
||||||
#ifdef USE_SENSOR
|
#ifdef USE_SENSOR
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ class APDS9960 : public PollingComponent, public i2c::I2CDevice {
|
|||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void update() override;
|
void update() override;
|
||||||
void loop() override;
|
void loop() override;
|
||||||
|
|
||||||
|
|||||||
@@ -45,7 +45,6 @@ service APIConnection {
|
|||||||
rpc time_command (TimeCommandRequest) returns (void) {}
|
rpc time_command (TimeCommandRequest) returns (void) {}
|
||||||
rpc update_command (UpdateCommandRequest) returns (void) {}
|
rpc update_command (UpdateCommandRequest) returns (void) {}
|
||||||
rpc valve_command (ValveCommandRequest) returns (void) {}
|
rpc valve_command (ValveCommandRequest) returns (void) {}
|
||||||
rpc water_heater_command (WaterHeaterCommandRequest) returns (void) {}
|
|
||||||
|
|
||||||
rpc subscribe_bluetooth_le_advertisements(SubscribeBluetoothLEAdvertisementsRequest) returns (void) {}
|
rpc subscribe_bluetooth_le_advertisements(SubscribeBluetoothLEAdvertisementsRequest) returns (void) {}
|
||||||
rpc bluetooth_device_request(BluetoothDeviceRequest) returns (void) {}
|
rpc bluetooth_device_request(BluetoothDeviceRequest) returns (void) {}
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -28,7 +28,7 @@ static constexpr size_t MAX_INITIAL_PER_BATCH = 34; // For clients >= AP
|
|||||||
static_assert(MAX_MESSAGES_PER_BATCH >= MAX_INITIAL_PER_BATCH,
|
static_assert(MAX_MESSAGES_PER_BATCH >= MAX_INITIAL_PER_BATCH,
|
||||||
"MAX_MESSAGES_PER_BATCH must be >= MAX_INITIAL_PER_BATCH");
|
"MAX_MESSAGES_PER_BATCH must be >= MAX_INITIAL_PER_BATCH");
|
||||||
|
|
||||||
class APIConnection final : public APIServerConnectionBase {
|
class APIConnection final : public APIServerConnection {
|
||||||
public:
|
public:
|
||||||
friend class APIServer;
|
friend class APIServer;
|
||||||
friend class ListEntitiesIterator;
|
friend class ListEntitiesIterator;
|
||||||
@@ -47,72 +47,72 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
#endif
|
#endif
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
bool send_cover_state(cover::Cover *cover);
|
bool send_cover_state(cover::Cover *cover);
|
||||||
void on_cover_command_request(const CoverCommandRequest &msg) override;
|
void cover_command(const CoverCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_FAN
|
#ifdef USE_FAN
|
||||||
bool send_fan_state(fan::Fan *fan);
|
bool send_fan_state(fan::Fan *fan);
|
||||||
void on_fan_command_request(const FanCommandRequest &msg) override;
|
void fan_command(const FanCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LIGHT
|
#ifdef USE_LIGHT
|
||||||
bool send_light_state(light::LightState *light);
|
bool send_light_state(light::LightState *light);
|
||||||
void on_light_command_request(const LightCommandRequest &msg) override;
|
void light_command(const LightCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SENSOR
|
#ifdef USE_SENSOR
|
||||||
bool send_sensor_state(sensor::Sensor *sensor);
|
bool send_sensor_state(sensor::Sensor *sensor);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SWITCH
|
#ifdef USE_SWITCH
|
||||||
bool send_switch_state(switch_::Switch *a_switch);
|
bool send_switch_state(switch_::Switch *a_switch);
|
||||||
void on_switch_command_request(const SwitchCommandRequest &msg) override;
|
void switch_command(const SwitchCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT_SENSOR
|
#ifdef USE_TEXT_SENSOR
|
||||||
bool send_text_sensor_state(text_sensor::TextSensor *text_sensor);
|
bool send_text_sensor_state(text_sensor::TextSensor *text_sensor);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_CAMERA
|
#ifdef USE_CAMERA
|
||||||
void set_camera_state(std::shared_ptr<camera::CameraImage> image);
|
void set_camera_state(std::shared_ptr<camera::CameraImage> image);
|
||||||
void on_camera_image_request(const CameraImageRequest &msg) override;
|
void camera_image(const CameraImageRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
bool send_climate_state(climate::Climate *climate);
|
bool send_climate_state(climate::Climate *climate);
|
||||||
void on_climate_command_request(const ClimateCommandRequest &msg) override;
|
void climate_command(const ClimateCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_NUMBER
|
#ifdef USE_NUMBER
|
||||||
bool send_number_state(number::Number *number);
|
bool send_number_state(number::Number *number);
|
||||||
void on_number_command_request(const NumberCommandRequest &msg) override;
|
void number_command(const NumberCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATE
|
#ifdef USE_DATETIME_DATE
|
||||||
bool send_date_state(datetime::DateEntity *date);
|
bool send_date_state(datetime::DateEntity *date);
|
||||||
void on_date_command_request(const DateCommandRequest &msg) override;
|
void date_command(const DateCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_TIME
|
#ifdef USE_DATETIME_TIME
|
||||||
bool send_time_state(datetime::TimeEntity *time);
|
bool send_time_state(datetime::TimeEntity *time);
|
||||||
void on_time_command_request(const TimeCommandRequest &msg) override;
|
void time_command(const TimeCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATETIME
|
#ifdef USE_DATETIME_DATETIME
|
||||||
bool send_datetime_state(datetime::DateTimeEntity *datetime);
|
bool send_datetime_state(datetime::DateTimeEntity *datetime);
|
||||||
void on_date_time_command_request(const DateTimeCommandRequest &msg) override;
|
void datetime_command(const DateTimeCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT
|
#ifdef USE_TEXT
|
||||||
bool send_text_state(text::Text *text);
|
bool send_text_state(text::Text *text);
|
||||||
void on_text_command_request(const TextCommandRequest &msg) override;
|
void text_command(const TextCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
bool send_select_state(select::Select *select);
|
bool send_select_state(select::Select *select);
|
||||||
void on_select_command_request(const SelectCommandRequest &msg) override;
|
void select_command(const SelectCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BUTTON
|
#ifdef USE_BUTTON
|
||||||
void on_button_command_request(const ButtonCommandRequest &msg) override;
|
void button_command(const ButtonCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LOCK
|
#ifdef USE_LOCK
|
||||||
bool send_lock_state(lock::Lock *a_lock);
|
bool send_lock_state(lock::Lock *a_lock);
|
||||||
void on_lock_command_request(const LockCommandRequest &msg) override;
|
void lock_command(const LockCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VALVE
|
#ifdef USE_VALVE
|
||||||
bool send_valve_state(valve::Valve *valve);
|
bool send_valve_state(valve::Valve *valve);
|
||||||
void on_valve_command_request(const ValveCommandRequest &msg) override;
|
void valve_command(const ValveCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
bool send_media_player_state(media_player::MediaPlayer *media_player);
|
bool send_media_player_state(media_player::MediaPlayer *media_player);
|
||||||
void on_media_player_command_request(const MediaPlayerCommandRequest &msg) override;
|
void media_player_command(const MediaPlayerCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
bool try_send_log_message(int level, const char *tag, const char *line, size_t message_len);
|
bool try_send_log_message(int level, const char *tag, const char *line, size_t message_len);
|
||||||
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
||||||
@@ -126,18 +126,18 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
#endif // USE_API_HOMEASSISTANT_ACTION_RESPONSES
|
#endif // USE_API_HOMEASSISTANT_ACTION_RESPONSES
|
||||||
#endif // USE_API_HOMEASSISTANT_SERVICES
|
#endif // USE_API_HOMEASSISTANT_SERVICES
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
void on_subscribe_bluetooth_le_advertisements_request(const SubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
void subscribe_bluetooth_le_advertisements(const SubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
||||||
void on_unsubscribe_bluetooth_le_advertisements_request() override;
|
void unsubscribe_bluetooth_le_advertisements(const UnsubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
||||||
|
|
||||||
void on_bluetooth_device_request(const BluetoothDeviceRequest &msg) override;
|
void bluetooth_device_request(const BluetoothDeviceRequest &msg) override;
|
||||||
void on_bluetooth_gatt_read_request(const BluetoothGATTReadRequest &msg) override;
|
void bluetooth_gatt_read(const BluetoothGATTReadRequest &msg) override;
|
||||||
void on_bluetooth_gatt_write_request(const BluetoothGATTWriteRequest &msg) override;
|
void bluetooth_gatt_write(const BluetoothGATTWriteRequest &msg) override;
|
||||||
void on_bluetooth_gatt_read_descriptor_request(const BluetoothGATTReadDescriptorRequest &msg) override;
|
void bluetooth_gatt_read_descriptor(const BluetoothGATTReadDescriptorRequest &msg) override;
|
||||||
void on_bluetooth_gatt_write_descriptor_request(const BluetoothGATTWriteDescriptorRequest &msg) override;
|
void bluetooth_gatt_write_descriptor(const BluetoothGATTWriteDescriptorRequest &msg) override;
|
||||||
void on_bluetooth_gatt_get_services_request(const BluetoothGATTGetServicesRequest &msg) override;
|
void bluetooth_gatt_get_services(const BluetoothGATTGetServicesRequest &msg) override;
|
||||||
void on_bluetooth_gatt_notify_request(const BluetoothGATTNotifyRequest &msg) override;
|
void bluetooth_gatt_notify(const BluetoothGATTNotifyRequest &msg) override;
|
||||||
void on_subscribe_bluetooth_connections_free_request() override;
|
bool send_subscribe_bluetooth_connections_free_response(const SubscribeBluetoothConnectionsFreeRequest &msg) override;
|
||||||
void on_bluetooth_scanner_set_mode_request(const BluetoothScannerSetModeRequest &msg) override;
|
void bluetooth_scanner_set_mode(const BluetoothScannerSetModeRequest &msg) override;
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_HOMEASSISTANT_TIME
|
#ifdef USE_HOMEASSISTANT_TIME
|
||||||
@@ -148,24 +148,24 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
void on_subscribe_voice_assistant_request(const SubscribeVoiceAssistantRequest &msg) override;
|
void subscribe_voice_assistant(const SubscribeVoiceAssistantRequest &msg) override;
|
||||||
void on_voice_assistant_response(const VoiceAssistantResponse &msg) override;
|
void on_voice_assistant_response(const VoiceAssistantResponse &msg) override;
|
||||||
void on_voice_assistant_event_response(const VoiceAssistantEventResponse &msg) override;
|
void on_voice_assistant_event_response(const VoiceAssistantEventResponse &msg) override;
|
||||||
void on_voice_assistant_audio(const VoiceAssistantAudio &msg) override;
|
void on_voice_assistant_audio(const VoiceAssistantAudio &msg) override;
|
||||||
void on_voice_assistant_timer_event_response(const VoiceAssistantTimerEventResponse &msg) override;
|
void on_voice_assistant_timer_event_response(const VoiceAssistantTimerEventResponse &msg) override;
|
||||||
void on_voice_assistant_announce_request(const VoiceAssistantAnnounceRequest &msg) override;
|
void on_voice_assistant_announce_request(const VoiceAssistantAnnounceRequest &msg) override;
|
||||||
void on_voice_assistant_configuration_request(const VoiceAssistantConfigurationRequest &msg) override;
|
bool send_voice_assistant_get_configuration_response(const VoiceAssistantConfigurationRequest &msg) override;
|
||||||
void on_voice_assistant_set_configuration(const VoiceAssistantSetConfiguration &msg) override;
|
void voice_assistant_set_configuration(const VoiceAssistantSetConfiguration &msg) override;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_ZWAVE_PROXY
|
#ifdef USE_ZWAVE_PROXY
|
||||||
void on_z_wave_proxy_frame(const ZWaveProxyFrame &msg) override;
|
void zwave_proxy_frame(const ZWaveProxyFrame &msg) override;
|
||||||
void on_z_wave_proxy_request(const ZWaveProxyRequest &msg) override;
|
void zwave_proxy_request(const ZWaveProxyRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
bool send_alarm_control_panel_state(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel);
|
bool send_alarm_control_panel_state(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel);
|
||||||
void on_alarm_control_panel_command_request(const AlarmControlPanelCommandRequest &msg) override;
|
void alarm_control_panel_command(const AlarmControlPanelCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_WATER_HEATER
|
#ifdef USE_WATER_HEATER
|
||||||
@@ -174,7 +174,7 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_IR_RF
|
#ifdef USE_IR_RF
|
||||||
void on_infrared_rf_transmit_raw_timings_request(const InfraredRFTransmitRawTimingsRequest &msg) override;
|
void infrared_rf_transmit_raw_timings(const InfraredRFTransmitRawTimingsRequest &msg) override;
|
||||||
void send_infrared_rf_receive_event(const InfraredRFReceiveEvent &msg);
|
void send_infrared_rf_receive_event(const InfraredRFReceiveEvent &msg);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -184,11 +184,11 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
|
|
||||||
#ifdef USE_UPDATE
|
#ifdef USE_UPDATE
|
||||||
bool send_update_state(update::UpdateEntity *update);
|
bool send_update_state(update::UpdateEntity *update);
|
||||||
void on_update_command_request(const UpdateCommandRequest &msg) override;
|
void update_command(const UpdateCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
void on_disconnect_response() override;
|
void on_disconnect_response(const DisconnectResponse &value) override;
|
||||||
void on_ping_response() override {
|
void on_ping_response(const PingResponse &value) override {
|
||||||
// we initiated ping
|
// we initiated ping
|
||||||
this->flags_.sent_ping = false;
|
this->flags_.sent_ping = false;
|
||||||
}
|
}
|
||||||
@@ -198,12 +198,12 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
#ifdef USE_HOMEASSISTANT_TIME
|
#ifdef USE_HOMEASSISTANT_TIME
|
||||||
void on_get_time_response(const GetTimeResponse &value) override;
|
void on_get_time_response(const GetTimeResponse &value) override;
|
||||||
#endif
|
#endif
|
||||||
void on_hello_request(const HelloRequest &msg) override;
|
bool send_hello_response(const HelloRequest &msg) override;
|
||||||
void on_disconnect_request() override;
|
bool send_disconnect_response(const DisconnectRequest &msg) override;
|
||||||
void on_ping_request() override;
|
bool send_ping_response(const PingRequest &msg) override;
|
||||||
void on_device_info_request() override;
|
bool send_device_info_response(const DeviceInfoRequest &msg) override;
|
||||||
void on_list_entities_request() override { this->begin_iterator_(ActiveIterator::LIST_ENTITIES); }
|
void list_entities(const ListEntitiesRequest &msg) override { this->begin_iterator_(ActiveIterator::LIST_ENTITIES); }
|
||||||
void on_subscribe_states_request() override {
|
void subscribe_states(const SubscribeStatesRequest &msg) override {
|
||||||
this->flags_.state_subscription = true;
|
this->flags_.state_subscription = true;
|
||||||
// Start initial state iterator only if no iterator is active
|
// Start initial state iterator only if no iterator is active
|
||||||
// If list_entities is running, we'll start initial_state when it completes
|
// If list_entities is running, we'll start initial_state when it completes
|
||||||
@@ -211,19 +211,21 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
this->begin_iterator_(ActiveIterator::INITIAL_STATE);
|
this->begin_iterator_(ActiveIterator::INITIAL_STATE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
void on_subscribe_logs_request(const SubscribeLogsRequest &msg) override {
|
void subscribe_logs(const SubscribeLogsRequest &msg) override {
|
||||||
this->flags_.log_subscription = msg.level;
|
this->flags_.log_subscription = msg.level;
|
||||||
if (msg.dump_config)
|
if (msg.dump_config)
|
||||||
App.schedule_dump_config();
|
App.schedule_dump_config();
|
||||||
}
|
}
|
||||||
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
||||||
void on_subscribe_homeassistant_services_request() override { this->flags_.service_call_subscription = true; }
|
void subscribe_homeassistant_services(const SubscribeHomeassistantServicesRequest &msg) override {
|
||||||
|
this->flags_.service_call_subscription = true;
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_HOMEASSISTANT_STATES
|
#ifdef USE_API_HOMEASSISTANT_STATES
|
||||||
void on_subscribe_home_assistant_states_request() override;
|
void subscribe_home_assistant_states(const SubscribeHomeAssistantStatesRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_USER_DEFINED_ACTIONS
|
#ifdef USE_API_USER_DEFINED_ACTIONS
|
||||||
void on_execute_service_request(const ExecuteServiceRequest &msg) override;
|
void execute_service(const ExecuteServiceRequest &msg) override;
|
||||||
#ifdef USE_API_USER_DEFINED_ACTION_RESPONSES
|
#ifdef USE_API_USER_DEFINED_ACTION_RESPONSES
|
||||||
void send_execute_service_response(uint32_t call_id, bool success, StringRef error_message);
|
void send_execute_service_response(uint32_t call_id, bool success, StringRef error_message);
|
||||||
#ifdef USE_API_USER_DEFINED_ACTION_RESPONSES_JSON
|
#ifdef USE_API_USER_DEFINED_ACTION_RESPONSES_JSON
|
||||||
@@ -233,7 +235,7 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
#endif // USE_API_USER_DEFINED_ACTION_RESPONSES
|
#endif // USE_API_USER_DEFINED_ACTION_RESPONSES
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_NOISE
|
#ifdef USE_API_NOISE
|
||||||
void on_noise_encryption_set_key_request(const NoiseEncryptionSetKeyRequest &msg) override;
|
bool send_noise_encryption_set_key_response(const NoiseEncryptionSetKeyRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
bool is_authenticated() override {
|
bool is_authenticated() override {
|
||||||
@@ -253,7 +255,17 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
|
|
||||||
void on_fatal_error() override;
|
void on_fatal_error() override;
|
||||||
void on_no_setup_connection() override;
|
void on_no_setup_connection() override;
|
||||||
bool send_message_impl(const ProtoMessage &msg, uint8_t message_type) override;
|
ProtoWriteBuffer create_buffer(uint32_t reserve_size) override {
|
||||||
|
// FIXME: ensure no recursive writes can happen
|
||||||
|
|
||||||
|
// Get header padding size - used for both reserve and insert
|
||||||
|
uint8_t header_padding = this->helper_->frame_header_padding();
|
||||||
|
// Get shared buffer from parent server
|
||||||
|
std::vector<uint8_t> &shared_buf = this->parent_->get_shared_buffer_ref();
|
||||||
|
this->prepare_first_message_buffer(shared_buf, header_padding,
|
||||||
|
reserve_size + header_padding + this->helper_->frame_footer_size());
|
||||||
|
return {&shared_buf};
|
||||||
|
}
|
||||||
|
|
||||||
void prepare_first_message_buffer(std::vector<uint8_t> &shared_buf, size_t header_padding, size_t total_size) {
|
void prepare_first_message_buffer(std::vector<uint8_t> &shared_buf, size_t header_padding, size_t total_size) {
|
||||||
shared_buf.clear();
|
shared_buf.clear();
|
||||||
@@ -265,13 +277,6 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
shared_buf.resize(header_padding);
|
shared_buf.resize(header_padding);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convenience overload - computes frame overhead internally
|
|
||||||
void prepare_first_message_buffer(std::vector<uint8_t> &shared_buf, size_t payload_size) {
|
|
||||||
const uint8_t header_padding = this->helper_->frame_header_padding();
|
|
||||||
const uint8_t footer_size = this->helper_->frame_footer_size();
|
|
||||||
this->prepare_first_message_buffer(shared_buf, header_padding, payload_size + header_padding + footer_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool try_to_clear_buffer(bool log_out_of_space);
|
bool try_to_clear_buffer(bool log_out_of_space);
|
||||||
bool send_buffer(ProtoWriteBuffer buffer, uint8_t message_type) override;
|
bool send_buffer(ProtoWriteBuffer buffer, uint8_t message_type) override;
|
||||||
|
|
||||||
@@ -283,21 +288,6 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
// Helper function to handle authentication completion
|
// Helper function to handle authentication completion
|
||||||
void complete_authentication_();
|
void complete_authentication_();
|
||||||
|
|
||||||
// Pattern B helpers: send response and return success/failure
|
|
||||||
bool send_hello_response_(const HelloRequest &msg);
|
|
||||||
bool send_disconnect_response_();
|
|
||||||
bool send_ping_response_();
|
|
||||||
bool send_device_info_response_();
|
|
||||||
#ifdef USE_API_NOISE
|
|
||||||
bool send_noise_encryption_set_key_response_(const NoiseEncryptionSetKeyRequest &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_subscribe_bluetooth_connections_free_response_();
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
bool send_voice_assistant_get_configuration_response_(const VoiceAssistantConfigurationRequest &msg);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef USE_CAMERA
|
#ifdef USE_CAMERA
|
||||||
void try_send_camera_image_();
|
void try_send_camera_image_();
|
||||||
#endif
|
#endif
|
||||||
@@ -308,21 +298,21 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
|
|
||||||
// Non-template helper to encode any ProtoMessage
|
// Non-template helper to encode any ProtoMessage
|
||||||
static uint16_t encode_message_to_buffer(ProtoMessage &msg, uint8_t message_type, APIConnection *conn,
|
static uint16_t encode_message_to_buffer(ProtoMessage &msg, uint8_t message_type, APIConnection *conn,
|
||||||
uint32_t remaining_size);
|
uint32_t remaining_size, bool is_single);
|
||||||
|
|
||||||
// Helper to fill entity state base and encode message
|
// Helper to fill entity state base and encode message
|
||||||
static uint16_t fill_and_encode_entity_state(EntityBase *entity, StateResponseProtoMessage &msg, uint8_t message_type,
|
static uint16_t fill_and_encode_entity_state(EntityBase *entity, StateResponseProtoMessage &msg, uint8_t message_type,
|
||||||
APIConnection *conn, uint32_t remaining_size) {
|
APIConnection *conn, uint32_t remaining_size, bool is_single) {
|
||||||
msg.key = entity->get_object_id_hash();
|
msg.key = entity->get_object_id_hash();
|
||||||
#ifdef USE_DEVICES
|
#ifdef USE_DEVICES
|
||||||
msg.device_id = entity->get_device_id();
|
msg.device_id = entity->get_device_id();
|
||||||
#endif
|
#endif
|
||||||
return encode_message_to_buffer(msg, message_type, conn, remaining_size);
|
return encode_message_to_buffer(msg, message_type, conn, remaining_size, is_single);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Helper to fill entity info base and encode message
|
// Helper to fill entity info base and encode message
|
||||||
static uint16_t fill_and_encode_entity_info(EntityBase *entity, InfoResponseProtoMessage &msg, uint8_t message_type,
|
static uint16_t fill_and_encode_entity_info(EntityBase *entity, InfoResponseProtoMessage &msg, uint8_t message_type,
|
||||||
APIConnection *conn, uint32_t remaining_size) {
|
APIConnection *conn, uint32_t remaining_size, bool is_single) {
|
||||||
// Set common fields that are shared by all entity types
|
// Set common fields that are shared by all entity types
|
||||||
msg.key = entity->get_object_id_hash();
|
msg.key = entity->get_object_id_hash();
|
||||||
|
|
||||||
@@ -349,7 +339,7 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
#ifdef USE_DEVICES
|
#ifdef USE_DEVICES
|
||||||
msg.device_id = entity->get_device_id();
|
msg.device_id = entity->get_device_id();
|
||||||
#endif
|
#endif
|
||||||
return encode_message_to_buffer(msg, message_type, conn, remaining_size);
|
return encode_message_to_buffer(msg, message_type, conn, remaining_size, is_single);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
@@ -380,108 +370,141 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
static uint16_t try_send_binary_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_binary_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_binary_sensor_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_binary_sensor_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
static uint16_t try_send_cover_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_cover_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_cover_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_cover_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_FAN
|
#ifdef USE_FAN
|
||||||
static uint16_t try_send_fan_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_fan_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
static uint16_t try_send_fan_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_fan_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LIGHT
|
#ifdef USE_LIGHT
|
||||||
static uint16_t try_send_light_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_light_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_light_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_light_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SENSOR
|
#ifdef USE_SENSOR
|
||||||
static uint16_t try_send_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_sensor_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_sensor_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SWITCH
|
#ifdef USE_SWITCH
|
||||||
static uint16_t try_send_switch_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_switch_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_switch_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_switch_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT_SENSOR
|
#ifdef USE_TEXT_SENSOR
|
||||||
static uint16_t try_send_text_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_text_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_text_sensor_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_text_sensor_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
static uint16_t try_send_climate_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_climate_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_climate_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_climate_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_NUMBER
|
#ifdef USE_NUMBER
|
||||||
static uint16_t try_send_number_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_number_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_number_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_number_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATE
|
#ifdef USE_DATETIME_DATE
|
||||||
static uint16_t try_send_date_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_date_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
static uint16_t try_send_date_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_date_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_TIME
|
#ifdef USE_DATETIME_TIME
|
||||||
static uint16_t try_send_time_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_time_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
static uint16_t try_send_time_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_time_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATETIME
|
#ifdef USE_DATETIME_DATETIME
|
||||||
static uint16_t try_send_datetime_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_datetime_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_datetime_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_datetime_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT
|
#ifdef USE_TEXT
|
||||||
static uint16_t try_send_text_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_text_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
static uint16_t try_send_text_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_text_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
static uint16_t try_send_select_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_select_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_select_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_select_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BUTTON
|
#ifdef USE_BUTTON
|
||||||
static uint16_t try_send_button_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_button_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LOCK
|
#ifdef USE_LOCK
|
||||||
static uint16_t try_send_lock_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_lock_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
static uint16_t try_send_lock_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_lock_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VALVE
|
#ifdef USE_VALVE
|
||||||
static uint16_t try_send_valve_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_valve_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_valve_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_valve_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
static uint16_t try_send_media_player_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_media_player_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_media_player_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_media_player_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
static uint16_t try_send_alarm_control_panel_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_alarm_control_panel_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_alarm_control_panel_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_alarm_control_panel_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_WATER_HEATER
|
#ifdef USE_WATER_HEATER
|
||||||
static uint16_t try_send_water_heater_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_water_heater_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_water_heater_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_water_heater_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_INFRARED
|
#ifdef USE_INFRARED
|
||||||
static uint16_t try_send_infrared_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_infrared_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_EVENT
|
#ifdef USE_EVENT
|
||||||
static uint16_t try_send_event_response(event::Event *event, StringRef event_type, APIConnection *conn,
|
static uint16_t try_send_event_response(event::Event *event, StringRef event_type, APIConnection *conn,
|
||||||
uint32_t remaining_size);
|
uint32_t remaining_size, bool is_single);
|
||||||
static uint16_t try_send_event_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_event_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_UPDATE
|
#ifdef USE_UPDATE
|
||||||
static uint16_t try_send_update_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_update_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
static uint16_t try_send_update_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
bool is_single);
|
||||||
|
static uint16_t try_send_update_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_CAMERA
|
#ifdef USE_CAMERA
|
||||||
static uint16_t try_send_camera_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_camera_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Method for ListEntitiesDone batching
|
// Method for ListEntitiesDone batching
|
||||||
static uint16_t try_send_list_info_done(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_list_info_done(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
|
||||||
// Method for DisconnectRequest batching
|
// Method for DisconnectRequest batching
|
||||||
static uint16_t try_send_disconnect_request(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_disconnect_request(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
|
||||||
// Batch message method for ping requests
|
// Batch message method for ping requests
|
||||||
static uint16_t try_send_ping_request(EntityBase *entity, APIConnection *conn, uint32_t remaining_size);
|
static uint16_t try_send_ping_request(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
|
||||||
// === Optimal member ordering for 32-bit systems ===
|
// === Optimal member ordering for 32-bit systems ===
|
||||||
|
|
||||||
@@ -516,7 +539,7 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Function pointer type for message encoding
|
// Function pointer type for message encoding
|
||||||
using MessageCreatorPtr = uint16_t (*)(EntityBase *, APIConnection *, uint32_t remaining_size);
|
using MessageCreatorPtr = uint16_t (*)(EntityBase *, APIConnection *, uint32_t remaining_size, bool is_single);
|
||||||
|
|
||||||
// Generic batching mechanism for both state updates and entity info
|
// Generic batching mechanism for both state updates and entity info
|
||||||
struct DeferredBatch {
|
struct DeferredBatch {
|
||||||
@@ -629,7 +652,7 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
|
|
||||||
// Dispatch message encoding based on message_type - replaces function pointer storage
|
// Dispatch message encoding based on message_type - replaces function pointer storage
|
||||||
// Switch assigns pointer, single call site for smaller code size
|
// Switch assigns pointer, single call site for smaller code size
|
||||||
uint16_t dispatch_message_(const DeferredBatch::BatchItem &item, uint32_t remaining_size, bool batch_first);
|
uint16_t dispatch_message_(const DeferredBatch::BatchItem &item, uint32_t remaining_size, bool is_single);
|
||||||
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
void log_batch_item_(const DeferredBatch::BatchItem &item) {
|
void log_batch_item_(const DeferredBatch::BatchItem &item) {
|
||||||
@@ -661,7 +684,19 @@ class APIConnection final : public APIServerConnectionBase {
|
|||||||
// Tries immediate send if should_send_immediately_() returns true and buffer has space
|
// Tries immediate send if should_send_immediately_() returns true and buffer has space
|
||||||
// Falls back to batching if immediate send fails or isn't applicable
|
// Falls back to batching if immediate send fails or isn't applicable
|
||||||
bool send_message_smart_(EntityBase *entity, uint8_t message_type, uint8_t estimated_size,
|
bool send_message_smart_(EntityBase *entity, uint8_t message_type, uint8_t estimated_size,
|
||||||
uint8_t aux_data_index = DeferredBatch::AUX_DATA_UNUSED);
|
uint8_t aux_data_index = DeferredBatch::AUX_DATA_UNUSED) {
|
||||||
|
if (this->should_send_immediately_(message_type) && this->helper_->can_write_without_blocking()) {
|
||||||
|
DeferredBatch::BatchItem item{entity, message_type, estimated_size, aux_data_index};
|
||||||
|
if (this->dispatch_message_(item, MAX_BATCH_PACKET_SIZE, true) &&
|
||||||
|
this->send_buffer(ProtoWriteBuffer{&this->parent_->get_shared_buffer_ref()}, message_type)) {
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
this->log_batch_item_(item);
|
||||||
|
#endif
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return this->schedule_message_(entity, message_type, estimated_size, aux_data_index);
|
||||||
|
}
|
||||||
|
|
||||||
// Helper function to schedule a deferred message with known message type
|
// Helper function to schedule a deferred message with known message type
|
||||||
bool schedule_message_(EntityBase *entity, uint8_t message_type, uint8_t estimated_size,
|
bool schedule_message_(EntityBase *entity, uint8_t message_type, uint8_t estimated_size,
|
||||||
|
|||||||
@@ -440,6 +440,19 @@ class PingResponse final : public ProtoMessage {
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
};
|
};
|
||||||
|
class DeviceInfoRequest final : public ProtoMessage {
|
||||||
|
public:
|
||||||
|
static constexpr uint8_t MESSAGE_TYPE = 9;
|
||||||
|
static constexpr uint8_t ESTIMATED_SIZE = 0;
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *message_name() const override { return "device_info_request"; }
|
||||||
|
#endif
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *dump_to(DumpBuffer &out) const override;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
protected:
|
||||||
|
};
|
||||||
#ifdef USE_AREAS
|
#ifdef USE_AREAS
|
||||||
class AreaInfo final : public ProtoMessage {
|
class AreaInfo final : public ProtoMessage {
|
||||||
public:
|
public:
|
||||||
@@ -533,6 +546,19 @@ class DeviceInfoResponse final : public ProtoMessage {
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
};
|
};
|
||||||
|
class ListEntitiesRequest final : public ProtoMessage {
|
||||||
|
public:
|
||||||
|
static constexpr uint8_t MESSAGE_TYPE = 11;
|
||||||
|
static constexpr uint8_t ESTIMATED_SIZE = 0;
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *message_name() const override { return "list_entities_request"; }
|
||||||
|
#endif
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *dump_to(DumpBuffer &out) const override;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
protected:
|
||||||
|
};
|
||||||
class ListEntitiesDoneResponse final : public ProtoMessage {
|
class ListEntitiesDoneResponse final : public ProtoMessage {
|
||||||
public:
|
public:
|
||||||
static constexpr uint8_t MESSAGE_TYPE = 19;
|
static constexpr uint8_t MESSAGE_TYPE = 19;
|
||||||
@@ -546,6 +572,19 @@ class ListEntitiesDoneResponse final : public ProtoMessage {
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
};
|
};
|
||||||
|
class SubscribeStatesRequest final : public ProtoMessage {
|
||||||
|
public:
|
||||||
|
static constexpr uint8_t MESSAGE_TYPE = 20;
|
||||||
|
static constexpr uint8_t ESTIMATED_SIZE = 0;
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *message_name() const override { return "subscribe_states_request"; }
|
||||||
|
#endif
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *dump_to(DumpBuffer &out) const override;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
protected:
|
||||||
|
};
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
class ListEntitiesBinarySensorResponse final : public InfoResponseProtoMessage {
|
class ListEntitiesBinarySensorResponse final : public InfoResponseProtoMessage {
|
||||||
public:
|
public:
|
||||||
@@ -998,6 +1037,19 @@ class NoiseEncryptionSetKeyResponse final : public ProtoMessage {
|
|||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
||||||
|
class SubscribeHomeassistantServicesRequest final : public ProtoMessage {
|
||||||
|
public:
|
||||||
|
static constexpr uint8_t MESSAGE_TYPE = 34;
|
||||||
|
static constexpr uint8_t ESTIMATED_SIZE = 0;
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *message_name() const override { return "subscribe_homeassistant_services_request"; }
|
||||||
|
#endif
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *dump_to(DumpBuffer &out) const override;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
protected:
|
||||||
|
};
|
||||||
class HomeassistantServiceMap final : public ProtoMessage {
|
class HomeassistantServiceMap final : public ProtoMessage {
|
||||||
public:
|
public:
|
||||||
StringRef key{};
|
StringRef key{};
|
||||||
@@ -1065,6 +1117,19 @@ class HomeassistantActionResponse final : public ProtoDecodableMessage {
|
|||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_HOMEASSISTANT_STATES
|
#ifdef USE_API_HOMEASSISTANT_STATES
|
||||||
|
class SubscribeHomeAssistantStatesRequest final : public ProtoMessage {
|
||||||
|
public:
|
||||||
|
static constexpr uint8_t MESSAGE_TYPE = 38;
|
||||||
|
static constexpr uint8_t ESTIMATED_SIZE = 0;
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *message_name() const override { return "subscribe_home_assistant_states_request"; }
|
||||||
|
#endif
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *dump_to(DumpBuffer &out) const override;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
protected:
|
||||||
|
};
|
||||||
class SubscribeHomeAssistantStateResponse final : public ProtoMessage {
|
class SubscribeHomeAssistantStateResponse final : public ProtoMessage {
|
||||||
public:
|
public:
|
||||||
static constexpr uint8_t MESSAGE_TYPE = 39;
|
static constexpr uint8_t MESSAGE_TYPE = 39;
|
||||||
@@ -2095,6 +2160,19 @@ class BluetoothGATTNotifyDataResponse final : public ProtoMessage {
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
};
|
};
|
||||||
|
class SubscribeBluetoothConnectionsFreeRequest final : public ProtoMessage {
|
||||||
|
public:
|
||||||
|
static constexpr uint8_t MESSAGE_TYPE = 80;
|
||||||
|
static constexpr uint8_t ESTIMATED_SIZE = 0;
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *message_name() const override { return "subscribe_bluetooth_connections_free_request"; }
|
||||||
|
#endif
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *dump_to(DumpBuffer &out) const override;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
protected:
|
||||||
|
};
|
||||||
class BluetoothConnectionsFreeResponse final : public ProtoMessage {
|
class BluetoothConnectionsFreeResponse final : public ProtoMessage {
|
||||||
public:
|
public:
|
||||||
static constexpr uint8_t MESSAGE_TYPE = 81;
|
static constexpr uint8_t MESSAGE_TYPE = 81;
|
||||||
@@ -2201,6 +2279,19 @@ class BluetoothDeviceUnpairingResponse final : public ProtoMessage {
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
};
|
};
|
||||||
|
class UnsubscribeBluetoothLEAdvertisementsRequest final : public ProtoMessage {
|
||||||
|
public:
|
||||||
|
static constexpr uint8_t MESSAGE_TYPE = 87;
|
||||||
|
static constexpr uint8_t ESTIMATED_SIZE = 0;
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *message_name() const override { return "unsubscribe_bluetooth_le_advertisements_request"; }
|
||||||
|
#endif
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
const char *dump_to(DumpBuffer &out) const override;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
protected:
|
||||||
|
};
|
||||||
class BluetoothDeviceClearCacheResponse final : public ProtoMessage {
|
class BluetoothDeviceClearCacheResponse final : public ProtoMessage {
|
||||||
public:
|
public:
|
||||||
static constexpr uint8_t MESSAGE_TYPE = 88;
|
static constexpr uint8_t MESSAGE_TYPE = 88;
|
||||||
|
|||||||
@@ -23,8 +23,15 @@ static inline void append_field_prefix(DumpBuffer &out, const char *field_name,
|
|||||||
out.append(indent, ' ').append(field_name).append(": ");
|
out.append(indent, ' ').append(field_name).append(": ");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void append_with_newline(DumpBuffer &out, const char *str) {
|
||||||
|
out.append(str);
|
||||||
|
out.append("\n");
|
||||||
|
}
|
||||||
|
|
||||||
static inline void append_uint(DumpBuffer &out, uint32_t value) {
|
static inline void append_uint(DumpBuffer &out, uint32_t value) {
|
||||||
out.set_pos(buf_append_printf(out.data(), DumpBuffer::CAPACITY, out.pos(), "%" PRIu32, value));
|
char buf[16];
|
||||||
|
snprintf(buf, sizeof(buf), "%" PRIu32, value);
|
||||||
|
out.append(buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
// RAII helper for message dump formatting
|
// RAII helper for message dump formatting
|
||||||
@@ -42,23 +49,31 @@ class MessageDumpHelper {
|
|||||||
|
|
||||||
// Helper functions to reduce code duplication in dump methods
|
// Helper functions to reduce code duplication in dump methods
|
||||||
static void dump_field(DumpBuffer &out, const char *field_name, int32_t value, int indent = 2) {
|
static void dump_field(DumpBuffer &out, const char *field_name, int32_t value, int indent = 2) {
|
||||||
|
char buffer[64];
|
||||||
append_field_prefix(out, field_name, indent);
|
append_field_prefix(out, field_name, indent);
|
||||||
out.set_pos(buf_append_printf(out.data(), DumpBuffer::CAPACITY, out.pos(), "%" PRId32 "\n", value));
|
snprintf(buffer, 64, "%" PRId32, value);
|
||||||
|
append_with_newline(out, buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void dump_field(DumpBuffer &out, const char *field_name, uint32_t value, int indent = 2) {
|
static void dump_field(DumpBuffer &out, const char *field_name, uint32_t value, int indent = 2) {
|
||||||
|
char buffer[64];
|
||||||
append_field_prefix(out, field_name, indent);
|
append_field_prefix(out, field_name, indent);
|
||||||
out.set_pos(buf_append_printf(out.data(), DumpBuffer::CAPACITY, out.pos(), "%" PRIu32 "\n", value));
|
snprintf(buffer, 64, "%" PRIu32, value);
|
||||||
|
append_with_newline(out, buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void dump_field(DumpBuffer &out, const char *field_name, float value, int indent = 2) {
|
static void dump_field(DumpBuffer &out, const char *field_name, float value, int indent = 2) {
|
||||||
|
char buffer[64];
|
||||||
append_field_prefix(out, field_name, indent);
|
append_field_prefix(out, field_name, indent);
|
||||||
out.set_pos(buf_append_printf(out.data(), DumpBuffer::CAPACITY, out.pos(), "%g\n", value));
|
snprintf(buffer, 64, "%g", value);
|
||||||
|
append_with_newline(out, buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void dump_field(DumpBuffer &out, const char *field_name, uint64_t value, int indent = 2) {
|
static void dump_field(DumpBuffer &out, const char *field_name, uint64_t value, int indent = 2) {
|
||||||
|
char buffer[64];
|
||||||
append_field_prefix(out, field_name, indent);
|
append_field_prefix(out, field_name, indent);
|
||||||
out.set_pos(buf_append_printf(out.data(), DumpBuffer::CAPACITY, out.pos(), "%" PRIu64 "\n", value));
|
snprintf(buffer, 64, "%" PRIu64, value);
|
||||||
|
append_with_newline(out, buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void dump_field(DumpBuffer &out, const char *field_name, bool value, int indent = 2) {
|
static void dump_field(DumpBuffer &out, const char *field_name, bool value, int indent = 2) {
|
||||||
@@ -97,7 +112,7 @@ static void dump_bytes_field(DumpBuffer &out, const char *field_name, const uint
|
|||||||
char hex_buf[format_hex_pretty_size(160)];
|
char hex_buf[format_hex_pretty_size(160)];
|
||||||
append_field_prefix(out, field_name, indent);
|
append_field_prefix(out, field_name, indent);
|
||||||
format_hex_pretty_to(hex_buf, data, len);
|
format_hex_pretty_to(hex_buf, data, len);
|
||||||
out.append(hex_buf).append("\n");
|
append_with_newline(out, hex_buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<> const char *proto_enum_to_string<enums::EntityCategory>(enums::EntityCategory value) {
|
template<> const char *proto_enum_to_string<enums::EntityCategory>(enums::EntityCategory value) {
|
||||||
@@ -764,6 +779,10 @@ const char *PingResponse::dump_to(DumpBuffer &out) const {
|
|||||||
out.append("PingResponse {}");
|
out.append("PingResponse {}");
|
||||||
return out.c_str();
|
return out.c_str();
|
||||||
}
|
}
|
||||||
|
const char *DeviceInfoRequest::dump_to(DumpBuffer &out) const {
|
||||||
|
out.append("DeviceInfoRequest {}");
|
||||||
|
return out.c_str();
|
||||||
|
}
|
||||||
#ifdef USE_AREAS
|
#ifdef USE_AREAS
|
||||||
const char *AreaInfo::dump_to(DumpBuffer &out) const {
|
const char *AreaInfo::dump_to(DumpBuffer &out) const {
|
||||||
MessageDumpHelper helper(out, "AreaInfo");
|
MessageDumpHelper helper(out, "AreaInfo");
|
||||||
@@ -844,10 +863,18 @@ const char *DeviceInfoResponse::dump_to(DumpBuffer &out) const {
|
|||||||
#endif
|
#endif
|
||||||
return out.c_str();
|
return out.c_str();
|
||||||
}
|
}
|
||||||
|
const char *ListEntitiesRequest::dump_to(DumpBuffer &out) const {
|
||||||
|
out.append("ListEntitiesRequest {}");
|
||||||
|
return out.c_str();
|
||||||
|
}
|
||||||
const char *ListEntitiesDoneResponse::dump_to(DumpBuffer &out) const {
|
const char *ListEntitiesDoneResponse::dump_to(DumpBuffer &out) const {
|
||||||
out.append("ListEntitiesDoneResponse {}");
|
out.append("ListEntitiesDoneResponse {}");
|
||||||
return out.c_str();
|
return out.c_str();
|
||||||
}
|
}
|
||||||
|
const char *SubscribeStatesRequest::dump_to(DumpBuffer &out) const {
|
||||||
|
out.append("SubscribeStatesRequest {}");
|
||||||
|
return out.c_str();
|
||||||
|
}
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
const char *ListEntitiesBinarySensorResponse::dump_to(DumpBuffer &out) const {
|
const char *ListEntitiesBinarySensorResponse::dump_to(DumpBuffer &out) const {
|
||||||
MessageDumpHelper helper(out, "ListEntitiesBinarySensorResponse");
|
MessageDumpHelper helper(out, "ListEntitiesBinarySensorResponse");
|
||||||
@@ -1179,6 +1206,10 @@ const char *NoiseEncryptionSetKeyResponse::dump_to(DumpBuffer &out) const {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
||||||
|
const char *SubscribeHomeassistantServicesRequest::dump_to(DumpBuffer &out) const {
|
||||||
|
out.append("SubscribeHomeassistantServicesRequest {}");
|
||||||
|
return out.c_str();
|
||||||
|
}
|
||||||
const char *HomeassistantServiceMap::dump_to(DumpBuffer &out) const {
|
const char *HomeassistantServiceMap::dump_to(DumpBuffer &out) const {
|
||||||
MessageDumpHelper helper(out, "HomeassistantServiceMap");
|
MessageDumpHelper helper(out, "HomeassistantServiceMap");
|
||||||
dump_field(out, "key", this->key);
|
dump_field(out, "key", this->key);
|
||||||
@@ -1229,6 +1260,10 @@ const char *HomeassistantActionResponse::dump_to(DumpBuffer &out) const {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_HOMEASSISTANT_STATES
|
#ifdef USE_API_HOMEASSISTANT_STATES
|
||||||
|
const char *SubscribeHomeAssistantStatesRequest::dump_to(DumpBuffer &out) const {
|
||||||
|
out.append("SubscribeHomeAssistantStatesRequest {}");
|
||||||
|
return out.c_str();
|
||||||
|
}
|
||||||
const char *SubscribeHomeAssistantStateResponse::dump_to(DumpBuffer &out) const {
|
const char *SubscribeHomeAssistantStateResponse::dump_to(DumpBuffer &out) const {
|
||||||
MessageDumpHelper helper(out, "SubscribeHomeAssistantStateResponse");
|
MessageDumpHelper helper(out, "SubscribeHomeAssistantStateResponse");
|
||||||
dump_field(out, "entity_id", this->entity_id);
|
dump_field(out, "entity_id", this->entity_id);
|
||||||
@@ -1904,6 +1939,10 @@ const char *BluetoothGATTNotifyDataResponse::dump_to(DumpBuffer &out) const {
|
|||||||
dump_bytes_field(out, "data", this->data_ptr_, this->data_len_);
|
dump_bytes_field(out, "data", this->data_ptr_, this->data_len_);
|
||||||
return out.c_str();
|
return out.c_str();
|
||||||
}
|
}
|
||||||
|
const char *SubscribeBluetoothConnectionsFreeRequest::dump_to(DumpBuffer &out) const {
|
||||||
|
out.append("SubscribeBluetoothConnectionsFreeRequest {}");
|
||||||
|
return out.c_str();
|
||||||
|
}
|
||||||
const char *BluetoothConnectionsFreeResponse::dump_to(DumpBuffer &out) const {
|
const char *BluetoothConnectionsFreeResponse::dump_to(DumpBuffer &out) const {
|
||||||
MessageDumpHelper helper(out, "BluetoothConnectionsFreeResponse");
|
MessageDumpHelper helper(out, "BluetoothConnectionsFreeResponse");
|
||||||
dump_field(out, "free", this->free);
|
dump_field(out, "free", this->free);
|
||||||
@@ -1946,6 +1985,10 @@ const char *BluetoothDeviceUnpairingResponse::dump_to(DumpBuffer &out) const {
|
|||||||
dump_field(out, "error", this->error);
|
dump_field(out, "error", this->error);
|
||||||
return out.c_str();
|
return out.c_str();
|
||||||
}
|
}
|
||||||
|
const char *UnsubscribeBluetoothLEAdvertisementsRequest::dump_to(DumpBuffer &out) const {
|
||||||
|
out.append("UnsubscribeBluetoothLEAdvertisementsRequest {}");
|
||||||
|
return out.c_str();
|
||||||
|
}
|
||||||
const char *BluetoothDeviceClearCacheResponse::dump_to(DumpBuffer &out) const {
|
const char *BluetoothDeviceClearCacheResponse::dump_to(DumpBuffer &out) const {
|
||||||
MessageDumpHelper helper(out, "BluetoothDeviceClearCacheResponse");
|
MessageDumpHelper helper(out, "BluetoothDeviceClearCacheResponse");
|
||||||
dump_field(out, "address", this->address);
|
dump_field(out, "address", this->address);
|
||||||
|
|||||||
@@ -15,29 +15,9 @@ void APIServerConnectionBase::log_receive_message_(const LogString *name, const
|
|||||||
DumpBuffer dump_buf;
|
DumpBuffer dump_buf;
|
||||||
ESP_LOGVV(TAG, "%s: %s", LOG_STR_ARG(name), msg.dump_to(dump_buf));
|
ESP_LOGVV(TAG, "%s: %s", LOG_STR_ARG(name), msg.dump_to(dump_buf));
|
||||||
}
|
}
|
||||||
void APIServerConnectionBase::log_receive_message_(const LogString *name) {
|
|
||||||
ESP_LOGVV(TAG, "%s: {}", LOG_STR_ARG(name));
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
void APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type, const uint8_t *msg_data) {
|
void APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type, const uint8_t *msg_data) {
|
||||||
// Check authentication/connection requirements
|
|
||||||
switch (msg_type) {
|
|
||||||
case HelloRequest::MESSAGE_TYPE: // No setup required
|
|
||||||
case DisconnectRequest::MESSAGE_TYPE: // No setup required
|
|
||||||
case PingRequest::MESSAGE_TYPE: // No setup required
|
|
||||||
break;
|
|
||||||
case 9 /* DeviceInfoRequest is empty */: // Connection setup only
|
|
||||||
if (!this->check_connection_setup_()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
if (!this->check_authenticated_()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
switch (msg_type) {
|
switch (msg_type) {
|
||||||
case HelloRequest::MESSAGE_TYPE: {
|
case HelloRequest::MESSAGE_TYPE: {
|
||||||
HelloRequest msg;
|
HelloRequest msg;
|
||||||
@@ -49,52 +29,66 @@ void APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type,
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case DisconnectRequest::MESSAGE_TYPE: {
|
case DisconnectRequest::MESSAGE_TYPE: {
|
||||||
|
DisconnectRequest msg;
|
||||||
|
// Empty message: no decode needed
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
this->log_receive_message_(LOG_STR("on_disconnect_request"));
|
this->log_receive_message_(LOG_STR("on_disconnect_request"), msg);
|
||||||
#endif
|
#endif
|
||||||
this->on_disconnect_request();
|
this->on_disconnect_request(msg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case DisconnectResponse::MESSAGE_TYPE: {
|
case DisconnectResponse::MESSAGE_TYPE: {
|
||||||
|
DisconnectResponse msg;
|
||||||
|
// Empty message: no decode needed
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
this->log_receive_message_(LOG_STR("on_disconnect_response"));
|
this->log_receive_message_(LOG_STR("on_disconnect_response"), msg);
|
||||||
#endif
|
#endif
|
||||||
this->on_disconnect_response();
|
this->on_disconnect_response(msg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case PingRequest::MESSAGE_TYPE: {
|
case PingRequest::MESSAGE_TYPE: {
|
||||||
|
PingRequest msg;
|
||||||
|
// Empty message: no decode needed
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
this->log_receive_message_(LOG_STR("on_ping_request"));
|
this->log_receive_message_(LOG_STR("on_ping_request"), msg);
|
||||||
#endif
|
#endif
|
||||||
this->on_ping_request();
|
this->on_ping_request(msg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case PingResponse::MESSAGE_TYPE: {
|
case PingResponse::MESSAGE_TYPE: {
|
||||||
|
PingResponse msg;
|
||||||
|
// Empty message: no decode needed
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
this->log_receive_message_(LOG_STR("on_ping_response"));
|
this->log_receive_message_(LOG_STR("on_ping_response"), msg);
|
||||||
#endif
|
#endif
|
||||||
this->on_ping_response();
|
this->on_ping_response(msg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case 9 /* DeviceInfoRequest is empty */: {
|
case DeviceInfoRequest::MESSAGE_TYPE: {
|
||||||
|
DeviceInfoRequest msg;
|
||||||
|
// Empty message: no decode needed
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
this->log_receive_message_(LOG_STR("on_device_info_request"));
|
this->log_receive_message_(LOG_STR("on_device_info_request"), msg);
|
||||||
#endif
|
#endif
|
||||||
this->on_device_info_request();
|
this->on_device_info_request(msg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case 11 /* ListEntitiesRequest is empty */: {
|
case ListEntitiesRequest::MESSAGE_TYPE: {
|
||||||
|
ListEntitiesRequest msg;
|
||||||
|
// Empty message: no decode needed
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
this->log_receive_message_(LOG_STR("on_list_entities_request"));
|
this->log_receive_message_(LOG_STR("on_list_entities_request"), msg);
|
||||||
#endif
|
#endif
|
||||||
this->on_list_entities_request();
|
this->on_list_entities_request(msg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case 20 /* SubscribeStatesRequest is empty */: {
|
case SubscribeStatesRequest::MESSAGE_TYPE: {
|
||||||
|
SubscribeStatesRequest msg;
|
||||||
|
// Empty message: no decode needed
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
this->log_receive_message_(LOG_STR("on_subscribe_states_request"));
|
this->log_receive_message_(LOG_STR("on_subscribe_states_request"), msg);
|
||||||
#endif
|
#endif
|
||||||
this->on_subscribe_states_request();
|
this->on_subscribe_states_request(msg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case SubscribeLogsRequest::MESSAGE_TYPE: {
|
case SubscribeLogsRequest::MESSAGE_TYPE: {
|
||||||
@@ -151,11 +145,13 @@ void APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type,
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
||||||
case 34 /* SubscribeHomeassistantServicesRequest is empty */: {
|
case SubscribeHomeassistantServicesRequest::MESSAGE_TYPE: {
|
||||||
|
SubscribeHomeassistantServicesRequest msg;
|
||||||
|
// Empty message: no decode needed
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
this->log_receive_message_(LOG_STR("on_subscribe_homeassistant_services_request"));
|
this->log_receive_message_(LOG_STR("on_subscribe_homeassistant_services_request"), msg);
|
||||||
#endif
|
#endif
|
||||||
this->on_subscribe_homeassistant_services_request();
|
this->on_subscribe_homeassistant_services_request(msg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@@ -169,11 +165,13 @@ void APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type,
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
#ifdef USE_API_HOMEASSISTANT_STATES
|
#ifdef USE_API_HOMEASSISTANT_STATES
|
||||||
case 38 /* SubscribeHomeAssistantStatesRequest is empty */: {
|
case SubscribeHomeAssistantStatesRequest::MESSAGE_TYPE: {
|
||||||
|
SubscribeHomeAssistantStatesRequest msg;
|
||||||
|
// Empty message: no decode needed
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
this->log_receive_message_(LOG_STR("on_subscribe_home_assistant_states_request"));
|
this->log_receive_message_(LOG_STR("on_subscribe_home_assistant_states_request"), msg);
|
||||||
#endif
|
#endif
|
||||||
this->on_subscribe_home_assistant_states_request();
|
this->on_subscribe_home_assistant_states_request(msg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@@ -376,20 +374,24 @@ void APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type,
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
case 80 /* SubscribeBluetoothConnectionsFreeRequest is empty */: {
|
case SubscribeBluetoothConnectionsFreeRequest::MESSAGE_TYPE: {
|
||||||
|
SubscribeBluetoothConnectionsFreeRequest msg;
|
||||||
|
// Empty message: no decode needed
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
this->log_receive_message_(LOG_STR("on_subscribe_bluetooth_connections_free_request"));
|
this->log_receive_message_(LOG_STR("on_subscribe_bluetooth_connections_free_request"), msg);
|
||||||
#endif
|
#endif
|
||||||
this->on_subscribe_bluetooth_connections_free_request();
|
this->on_subscribe_bluetooth_connections_free_request(msg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
case 87 /* UnsubscribeBluetoothLEAdvertisementsRequest is empty */: {
|
case UnsubscribeBluetoothLEAdvertisementsRequest::MESSAGE_TYPE: {
|
||||||
|
UnsubscribeBluetoothLEAdvertisementsRequest msg;
|
||||||
|
// Empty message: no decode needed
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
this->log_receive_message_(LOG_STR("on_unsubscribe_bluetooth_le_advertisements_request"));
|
this->log_receive_message_(LOG_STR("on_unsubscribe_bluetooth_le_advertisements_request"), msg);
|
||||||
#endif
|
#endif
|
||||||
this->on_unsubscribe_bluetooth_le_advertisements_request();
|
this->on_unsubscribe_bluetooth_le_advertisements_request(msg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@@ -640,4 +642,226 @@ void APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void APIServerConnection::on_hello_request(const HelloRequest &msg) {
|
||||||
|
if (!this->send_hello_response(msg)) {
|
||||||
|
this->on_fatal_error();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void APIServerConnection::on_disconnect_request(const DisconnectRequest &msg) {
|
||||||
|
if (!this->send_disconnect_response(msg)) {
|
||||||
|
this->on_fatal_error();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void APIServerConnection::on_ping_request(const PingRequest &msg) {
|
||||||
|
if (!this->send_ping_response(msg)) {
|
||||||
|
this->on_fatal_error();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void APIServerConnection::on_device_info_request(const DeviceInfoRequest &msg) {
|
||||||
|
if (!this->send_device_info_response(msg)) {
|
||||||
|
this->on_fatal_error();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void APIServerConnection::on_list_entities_request(const ListEntitiesRequest &msg) { this->list_entities(msg); }
|
||||||
|
void APIServerConnection::on_subscribe_states_request(const SubscribeStatesRequest &msg) {
|
||||||
|
this->subscribe_states(msg);
|
||||||
|
}
|
||||||
|
void APIServerConnection::on_subscribe_logs_request(const SubscribeLogsRequest &msg) { this->subscribe_logs(msg); }
|
||||||
|
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
||||||
|
void APIServerConnection::on_subscribe_homeassistant_services_request(
|
||||||
|
const SubscribeHomeassistantServicesRequest &msg) {
|
||||||
|
this->subscribe_homeassistant_services(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_API_HOMEASSISTANT_STATES
|
||||||
|
void APIServerConnection::on_subscribe_home_assistant_states_request(const SubscribeHomeAssistantStatesRequest &msg) {
|
||||||
|
this->subscribe_home_assistant_states(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_API_USER_DEFINED_ACTIONS
|
||||||
|
void APIServerConnection::on_execute_service_request(const ExecuteServiceRequest &msg) { this->execute_service(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_API_NOISE
|
||||||
|
void APIServerConnection::on_noise_encryption_set_key_request(const NoiseEncryptionSetKeyRequest &msg) {
|
||||||
|
if (!this->send_noise_encryption_set_key_response(msg)) {
|
||||||
|
this->on_fatal_error();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BUTTON
|
||||||
|
void APIServerConnection::on_button_command_request(const ButtonCommandRequest &msg) { this->button_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_CAMERA
|
||||||
|
void APIServerConnection::on_camera_image_request(const CameraImageRequest &msg) { this->camera_image(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_CLIMATE
|
||||||
|
void APIServerConnection::on_climate_command_request(const ClimateCommandRequest &msg) { this->climate_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_COVER
|
||||||
|
void APIServerConnection::on_cover_command_request(const CoverCommandRequest &msg) { this->cover_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_DATE
|
||||||
|
void APIServerConnection::on_date_command_request(const DateCommandRequest &msg) { this->date_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_DATETIME
|
||||||
|
void APIServerConnection::on_date_time_command_request(const DateTimeCommandRequest &msg) {
|
||||||
|
this->datetime_command(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_FAN
|
||||||
|
void APIServerConnection::on_fan_command_request(const FanCommandRequest &msg) { this->fan_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_LIGHT
|
||||||
|
void APIServerConnection::on_light_command_request(const LightCommandRequest &msg) { this->light_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_LOCK
|
||||||
|
void APIServerConnection::on_lock_command_request(const LockCommandRequest &msg) { this->lock_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_MEDIA_PLAYER
|
||||||
|
void APIServerConnection::on_media_player_command_request(const MediaPlayerCommandRequest &msg) {
|
||||||
|
this->media_player_command(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_NUMBER
|
||||||
|
void APIServerConnection::on_number_command_request(const NumberCommandRequest &msg) { this->number_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SELECT
|
||||||
|
void APIServerConnection::on_select_command_request(const SelectCommandRequest &msg) { this->select_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SIREN
|
||||||
|
void APIServerConnection::on_siren_command_request(const SirenCommandRequest &msg) { this->siren_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SWITCH
|
||||||
|
void APIServerConnection::on_switch_command_request(const SwitchCommandRequest &msg) { this->switch_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_TEXT
|
||||||
|
void APIServerConnection::on_text_command_request(const TextCommandRequest &msg) { this->text_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_TIME
|
||||||
|
void APIServerConnection::on_time_command_request(const TimeCommandRequest &msg) { this->time_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_UPDATE
|
||||||
|
void APIServerConnection::on_update_command_request(const UpdateCommandRequest &msg) { this->update_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VALVE
|
||||||
|
void APIServerConnection::on_valve_command_request(const ValveCommandRequest &msg) { this->valve_command(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServerConnection::on_subscribe_bluetooth_le_advertisements_request(
|
||||||
|
const SubscribeBluetoothLEAdvertisementsRequest &msg) {
|
||||||
|
this->subscribe_bluetooth_le_advertisements(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServerConnection::on_bluetooth_device_request(const BluetoothDeviceRequest &msg) {
|
||||||
|
this->bluetooth_device_request(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServerConnection::on_bluetooth_gatt_get_services_request(const BluetoothGATTGetServicesRequest &msg) {
|
||||||
|
this->bluetooth_gatt_get_services(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServerConnection::on_bluetooth_gatt_read_request(const BluetoothGATTReadRequest &msg) {
|
||||||
|
this->bluetooth_gatt_read(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServerConnection::on_bluetooth_gatt_write_request(const BluetoothGATTWriteRequest &msg) {
|
||||||
|
this->bluetooth_gatt_write(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServerConnection::on_bluetooth_gatt_read_descriptor_request(const BluetoothGATTReadDescriptorRequest &msg) {
|
||||||
|
this->bluetooth_gatt_read_descriptor(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServerConnection::on_bluetooth_gatt_write_descriptor_request(const BluetoothGATTWriteDescriptorRequest &msg) {
|
||||||
|
this->bluetooth_gatt_write_descriptor(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServerConnection::on_bluetooth_gatt_notify_request(const BluetoothGATTNotifyRequest &msg) {
|
||||||
|
this->bluetooth_gatt_notify(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServerConnection::on_subscribe_bluetooth_connections_free_request(
|
||||||
|
const SubscribeBluetoothConnectionsFreeRequest &msg) {
|
||||||
|
if (!this->send_subscribe_bluetooth_connections_free_response(msg)) {
|
||||||
|
this->on_fatal_error();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServerConnection::on_unsubscribe_bluetooth_le_advertisements_request(
|
||||||
|
const UnsubscribeBluetoothLEAdvertisementsRequest &msg) {
|
||||||
|
this->unsubscribe_bluetooth_le_advertisements(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServerConnection::on_bluetooth_scanner_set_mode_request(const BluetoothScannerSetModeRequest &msg) {
|
||||||
|
this->bluetooth_scanner_set_mode(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
|
void APIServerConnection::on_subscribe_voice_assistant_request(const SubscribeVoiceAssistantRequest &msg) {
|
||||||
|
this->subscribe_voice_assistant(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
|
void APIServerConnection::on_voice_assistant_configuration_request(const VoiceAssistantConfigurationRequest &msg) {
|
||||||
|
if (!this->send_voice_assistant_get_configuration_response(msg)) {
|
||||||
|
this->on_fatal_error();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
|
void APIServerConnection::on_voice_assistant_set_configuration(const VoiceAssistantSetConfiguration &msg) {
|
||||||
|
this->voice_assistant_set_configuration(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
|
void APIServerConnection::on_alarm_control_panel_command_request(const AlarmControlPanelCommandRequest &msg) {
|
||||||
|
this->alarm_control_panel_command(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef USE_ZWAVE_PROXY
|
||||||
|
void APIServerConnection::on_z_wave_proxy_frame(const ZWaveProxyFrame &msg) { this->zwave_proxy_frame(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_ZWAVE_PROXY
|
||||||
|
void APIServerConnection::on_z_wave_proxy_request(const ZWaveProxyRequest &msg) { this->zwave_proxy_request(msg); }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_IR_RF
|
||||||
|
void APIServerConnection::on_infrared_rf_transmit_raw_timings_request(const InfraredRFTransmitRawTimingsRequest &msg) {
|
||||||
|
this->infrared_rf_transmit_raw_timings(msg);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void APIServerConnection::read_message(uint32_t msg_size, uint32_t msg_type, const uint8_t *msg_data) {
|
||||||
|
// Check authentication/connection requirements for messages
|
||||||
|
switch (msg_type) {
|
||||||
|
case HelloRequest::MESSAGE_TYPE: // No setup required
|
||||||
|
case DisconnectRequest::MESSAGE_TYPE: // No setup required
|
||||||
|
case PingRequest::MESSAGE_TYPE: // No setup required
|
||||||
|
break; // Skip all checks for these messages
|
||||||
|
case DeviceInfoRequest::MESSAGE_TYPE: // Connection setup only
|
||||||
|
if (!this->check_connection_setup_()) {
|
||||||
|
return; // Connection not setup
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
// All other messages require authentication (which includes connection check)
|
||||||
|
if (!this->check_authenticated_()) {
|
||||||
|
return; // Authentication failed
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call base implementation to process the message
|
||||||
|
APIServerConnectionBase::read_message(msg_size, msg_type, msg_data);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace esphome::api
|
} // namespace esphome::api
|
||||||
|
|||||||
@@ -14,7 +14,6 @@ class APIServerConnectionBase : public ProtoService {
|
|||||||
protected:
|
protected:
|
||||||
void log_send_message_(const char *name, const char *dump);
|
void log_send_message_(const char *name, const char *dump);
|
||||||
void log_receive_message_(const LogString *name, const ProtoMessage &msg);
|
void log_receive_message_(const LogString *name, const ProtoMessage &msg);
|
||||||
void log_receive_message_(const LogString *name);
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
#endif
|
#endif
|
||||||
@@ -24,20 +23,20 @@ class APIServerConnectionBase : public ProtoService {
|
|||||||
DumpBuffer dump_buf;
|
DumpBuffer dump_buf;
|
||||||
this->log_send_message_(msg.message_name(), msg.dump_to(dump_buf));
|
this->log_send_message_(msg.message_name(), msg.dump_to(dump_buf));
|
||||||
#endif
|
#endif
|
||||||
return this->send_message_impl(msg, message_type);
|
return this->send_message_(msg, message_type);
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void on_hello_request(const HelloRequest &value){};
|
virtual void on_hello_request(const HelloRequest &value){};
|
||||||
|
|
||||||
virtual void on_disconnect_request(){};
|
virtual void on_disconnect_request(const DisconnectRequest &value){};
|
||||||
virtual void on_disconnect_response(){};
|
virtual void on_disconnect_response(const DisconnectResponse &value){};
|
||||||
virtual void on_ping_request(){};
|
virtual void on_ping_request(const PingRequest &value){};
|
||||||
virtual void on_ping_response(){};
|
virtual void on_ping_response(const PingResponse &value){};
|
||||||
virtual void on_device_info_request(){};
|
virtual void on_device_info_request(const DeviceInfoRequest &value){};
|
||||||
|
|
||||||
virtual void on_list_entities_request(){};
|
virtual void on_list_entities_request(const ListEntitiesRequest &value){};
|
||||||
|
|
||||||
virtual void on_subscribe_states_request(){};
|
virtual void on_subscribe_states_request(const SubscribeStatesRequest &value){};
|
||||||
|
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
virtual void on_cover_command_request(const CoverCommandRequest &value){};
|
virtual void on_cover_command_request(const CoverCommandRequest &value){};
|
||||||
@@ -62,14 +61,14 @@ class APIServerConnectionBase : public ProtoService {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
||||||
virtual void on_subscribe_homeassistant_services_request(){};
|
virtual void on_subscribe_homeassistant_services_request(const SubscribeHomeassistantServicesRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES
|
#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES
|
||||||
virtual void on_homeassistant_action_response(const HomeassistantActionResponse &value){};
|
virtual void on_homeassistant_action_response(const HomeassistantActionResponse &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_HOMEASSISTANT_STATES
|
#ifdef USE_API_HOMEASSISTANT_STATES
|
||||||
virtual void on_subscribe_home_assistant_states_request(){};
|
virtual void on_subscribe_home_assistant_states_request(const SubscribeHomeAssistantStatesRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_API_HOMEASSISTANT_STATES
|
#ifdef USE_API_HOMEASSISTANT_STATES
|
||||||
@@ -148,11 +147,12 @@ class APIServerConnectionBase : public ProtoService {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void on_subscribe_bluetooth_connections_free_request(){};
|
virtual void on_subscribe_bluetooth_connections_free_request(const SubscribeBluetoothConnectionsFreeRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void on_unsubscribe_bluetooth_le_advertisements_request(){};
|
virtual void on_unsubscribe_bluetooth_le_advertisements_request(
|
||||||
|
const UnsubscribeBluetoothLEAdvertisementsRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
@@ -228,4 +228,266 @@ class APIServerConnectionBase : public ProtoService {
|
|||||||
void read_message(uint32_t msg_size, uint32_t msg_type, const uint8_t *msg_data) override;
|
void read_message(uint32_t msg_size, uint32_t msg_type, const uint8_t *msg_data) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class APIServerConnection : public APIServerConnectionBase {
|
||||||
|
public:
|
||||||
|
virtual bool send_hello_response(const HelloRequest &msg) = 0;
|
||||||
|
virtual bool send_disconnect_response(const DisconnectRequest &msg) = 0;
|
||||||
|
virtual bool send_ping_response(const PingRequest &msg) = 0;
|
||||||
|
virtual bool send_device_info_response(const DeviceInfoRequest &msg) = 0;
|
||||||
|
virtual void list_entities(const ListEntitiesRequest &msg) = 0;
|
||||||
|
virtual void subscribe_states(const SubscribeStatesRequest &msg) = 0;
|
||||||
|
virtual void subscribe_logs(const SubscribeLogsRequest &msg) = 0;
|
||||||
|
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
||||||
|
virtual void subscribe_homeassistant_services(const SubscribeHomeassistantServicesRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_API_HOMEASSISTANT_STATES
|
||||||
|
virtual void subscribe_home_assistant_states(const SubscribeHomeAssistantStatesRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_API_USER_DEFINED_ACTIONS
|
||||||
|
virtual void execute_service(const ExecuteServiceRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_API_NOISE
|
||||||
|
virtual bool send_noise_encryption_set_key_response(const NoiseEncryptionSetKeyRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BUTTON
|
||||||
|
virtual void button_command(const ButtonCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_CAMERA
|
||||||
|
virtual void camera_image(const CameraImageRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_CLIMATE
|
||||||
|
virtual void climate_command(const ClimateCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_COVER
|
||||||
|
virtual void cover_command(const CoverCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_DATE
|
||||||
|
virtual void date_command(const DateCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_DATETIME
|
||||||
|
virtual void datetime_command(const DateTimeCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_FAN
|
||||||
|
virtual void fan_command(const FanCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_LIGHT
|
||||||
|
virtual void light_command(const LightCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_LOCK
|
||||||
|
virtual void lock_command(const LockCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_MEDIA_PLAYER
|
||||||
|
virtual void media_player_command(const MediaPlayerCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_NUMBER
|
||||||
|
virtual void number_command(const NumberCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SELECT
|
||||||
|
virtual void select_command(const SelectCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SIREN
|
||||||
|
virtual void siren_command(const SirenCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SWITCH
|
||||||
|
virtual void switch_command(const SwitchCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_TEXT
|
||||||
|
virtual void text_command(const TextCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_TIME
|
||||||
|
virtual void time_command(const TimeCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_UPDATE
|
||||||
|
virtual void update_command(const UpdateCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VALVE
|
||||||
|
virtual void valve_command(const ValveCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
virtual void subscribe_bluetooth_le_advertisements(const SubscribeBluetoothLEAdvertisementsRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
virtual void bluetooth_device_request(const BluetoothDeviceRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
virtual void bluetooth_gatt_get_services(const BluetoothGATTGetServicesRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
virtual void bluetooth_gatt_read(const BluetoothGATTReadRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
virtual void bluetooth_gatt_write(const BluetoothGATTWriteRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
virtual void bluetooth_gatt_read_descriptor(const BluetoothGATTReadDescriptorRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
virtual void bluetooth_gatt_write_descriptor(const BluetoothGATTWriteDescriptorRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
virtual void bluetooth_gatt_notify(const BluetoothGATTNotifyRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
virtual bool send_subscribe_bluetooth_connections_free_response(
|
||||||
|
const SubscribeBluetoothConnectionsFreeRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
virtual void unsubscribe_bluetooth_le_advertisements(const UnsubscribeBluetoothLEAdvertisementsRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
virtual void bluetooth_scanner_set_mode(const BluetoothScannerSetModeRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
|
virtual void subscribe_voice_assistant(const SubscribeVoiceAssistantRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
|
virtual bool send_voice_assistant_get_configuration_response(const VoiceAssistantConfigurationRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
|
virtual void voice_assistant_set_configuration(const VoiceAssistantSetConfiguration &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
|
virtual void alarm_control_panel_command(const AlarmControlPanelCommandRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_ZWAVE_PROXY
|
||||||
|
virtual void zwave_proxy_frame(const ZWaveProxyFrame &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_ZWAVE_PROXY
|
||||||
|
virtual void zwave_proxy_request(const ZWaveProxyRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_IR_RF
|
||||||
|
virtual void infrared_rf_transmit_raw_timings(const InfraredRFTransmitRawTimingsRequest &msg) = 0;
|
||||||
|
#endif
|
||||||
|
protected:
|
||||||
|
void on_hello_request(const HelloRequest &msg) override;
|
||||||
|
void on_disconnect_request(const DisconnectRequest &msg) override;
|
||||||
|
void on_ping_request(const PingRequest &msg) override;
|
||||||
|
void on_device_info_request(const DeviceInfoRequest &msg) override;
|
||||||
|
void on_list_entities_request(const ListEntitiesRequest &msg) override;
|
||||||
|
void on_subscribe_states_request(const SubscribeStatesRequest &msg) override;
|
||||||
|
void on_subscribe_logs_request(const SubscribeLogsRequest &msg) override;
|
||||||
|
#ifdef USE_API_HOMEASSISTANT_SERVICES
|
||||||
|
void on_subscribe_homeassistant_services_request(const SubscribeHomeassistantServicesRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_API_HOMEASSISTANT_STATES
|
||||||
|
void on_subscribe_home_assistant_states_request(const SubscribeHomeAssistantStatesRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_API_USER_DEFINED_ACTIONS
|
||||||
|
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_CAMERA
|
||||||
|
void on_camera_image_request(const CameraImageRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_CLIMATE
|
||||||
|
void on_climate_command_request(const ClimateCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_COVER
|
||||||
|
void on_cover_command_request(const CoverCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_DATE
|
||||||
|
void on_date_command_request(const DateCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_DATETIME
|
||||||
|
void on_date_time_command_request(const DateTimeCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_FAN
|
||||||
|
void on_fan_command_request(const FanCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_LIGHT
|
||||||
|
void on_light_command_request(const LightCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_LOCK
|
||||||
|
void on_lock_command_request(const LockCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_MEDIA_PLAYER
|
||||||
|
void on_media_player_command_request(const MediaPlayerCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_NUMBER
|
||||||
|
void on_number_command_request(const NumberCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SELECT
|
||||||
|
void on_select_command_request(const SelectCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SIREN
|
||||||
|
void on_siren_command_request(const SirenCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SWITCH
|
||||||
|
void on_switch_command_request(const SwitchCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_TEXT
|
||||||
|
void on_text_command_request(const TextCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_TIME
|
||||||
|
void on_time_command_request(const TimeCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_UPDATE
|
||||||
|
void on_update_command_request(const UpdateCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VALVE
|
||||||
|
void on_valve_command_request(const ValveCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void on_subscribe_bluetooth_le_advertisements_request(const SubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void on_bluetooth_device_request(const BluetoothDeviceRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void on_bluetooth_gatt_get_services_request(const BluetoothGATTGetServicesRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void on_bluetooth_gatt_read_request(const BluetoothGATTReadRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void on_bluetooth_gatt_write_request(const BluetoothGATTWriteRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void on_bluetooth_gatt_read_descriptor_request(const BluetoothGATTReadDescriptorRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void on_bluetooth_gatt_write_descriptor_request(const BluetoothGATTWriteDescriptorRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void on_bluetooth_gatt_notify_request(const BluetoothGATTNotifyRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void on_subscribe_bluetooth_connections_free_request(const SubscribeBluetoothConnectionsFreeRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void on_unsubscribe_bluetooth_le_advertisements_request(
|
||||||
|
const UnsubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void on_bluetooth_scanner_set_mode_request(const BluetoothScannerSetModeRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
|
void on_subscribe_voice_assistant_request(const SubscribeVoiceAssistantRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
|
void on_voice_assistant_configuration_request(const VoiceAssistantConfigurationRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
|
void on_voice_assistant_set_configuration(const VoiceAssistantSetConfiguration &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
|
void on_alarm_control_panel_command_request(const AlarmControlPanelCommandRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_ZWAVE_PROXY
|
||||||
|
void on_z_wave_proxy_frame(const ZWaveProxyFrame &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_ZWAVE_PROXY
|
||||||
|
void on_z_wave_proxy_request(const ZWaveProxyRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
#ifdef USE_IR_RF
|
||||||
|
void on_infrared_rf_transmit_raw_timings_request(const InfraredRFTransmitRawTimingsRequest &msg) override;
|
||||||
|
#endif
|
||||||
|
void read_message(uint32_t msg_size, uint32_t msg_type, const uint8_t *msg_data) override;
|
||||||
|
};
|
||||||
|
|
||||||
} // namespace esphome::api
|
} // namespace esphome::api
|
||||||
|
|||||||
@@ -211,7 +211,7 @@ void APIServer::loop() {
|
|||||||
|
|
||||||
#ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER
|
#ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER
|
||||||
// Fire trigger after client is removed so api.connected reflects the true state
|
// Fire trigger after client is removed so api.connected reflects the true state
|
||||||
this->client_disconnected_trigger_.trigger(client_name, client_peername);
|
this->client_disconnected_trigger_->trigger(client_name, client_peername);
|
||||||
#endif
|
#endif
|
||||||
// Don't increment client_index since we need to process the swapped element
|
// Don't increment client_index since we need to process the swapped element
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -227,10 +227,12 @@ class APIServer : public Component,
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_API_CLIENT_CONNECTED_TRIGGER
|
#ifdef USE_API_CLIENT_CONNECTED_TRIGGER
|
||||||
Trigger<std::string, std::string> *get_client_connected_trigger() { return &this->client_connected_trigger_; }
|
Trigger<std::string, std::string> *get_client_connected_trigger() const { return this->client_connected_trigger_; }
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER
|
#ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER
|
||||||
Trigger<std::string, std::string> *get_client_disconnected_trigger() { return &this->client_disconnected_trigger_; }
|
Trigger<std::string, std::string> *get_client_disconnected_trigger() const {
|
||||||
|
return this->client_disconnected_trigger_;
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@@ -251,10 +253,10 @@ class APIServer : public Component,
|
|||||||
// Pointers and pointer-like types first (4 bytes each)
|
// Pointers and pointer-like types first (4 bytes each)
|
||||||
std::unique_ptr<socket::Socket> socket_ = nullptr;
|
std::unique_ptr<socket::Socket> socket_ = nullptr;
|
||||||
#ifdef USE_API_CLIENT_CONNECTED_TRIGGER
|
#ifdef USE_API_CLIENT_CONNECTED_TRIGGER
|
||||||
Trigger<std::string, std::string> client_connected_trigger_;
|
Trigger<std::string, std::string> *client_connected_trigger_ = new Trigger<std::string, std::string>();
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER
|
#ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER
|
||||||
Trigger<std::string, std::string> client_disconnected_trigger_;
|
Trigger<std::string, std::string> *client_disconnected_trigger_ = new Trigger<std::string, std::string>();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// 4-byte aligned types
|
// 4-byte aligned types
|
||||||
|
|||||||
@@ -25,9 +25,7 @@ template<typename... X> class TemplatableStringValue : public TemplatableValue<s
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
// Helper to convert value to string - handles the case where value is already a string
|
// Helper to convert value to string - handles the case where value is already a string
|
||||||
template<typename T> static std::string value_to_string(T &&val) {
|
template<typename T> static std::string value_to_string(T &&val) { return to_string(std::forward<T>(val)); }
|
||||||
return to_string(std::forward<T>(val)); // NOLINT
|
|
||||||
}
|
|
||||||
|
|
||||||
// Overloads for string types - needed because std::to_string doesn't support them
|
// Overloads for string types - needed because std::to_string doesn't support them
|
||||||
static std::string value_to_string(char *val) {
|
static std::string value_to_string(char *val) {
|
||||||
@@ -138,10 +136,12 @@ template<typename... Ts> class HomeAssistantServiceCallAction : public Action<Ts
|
|||||||
void set_wants_response() { this->flags_.wants_response = true; }
|
void set_wants_response() { this->flags_.wants_response = true; }
|
||||||
|
|
||||||
#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES_JSON
|
#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES_JSON
|
||||||
Trigger<JsonObjectConst, Ts...> *get_success_trigger_with_response() { return &this->success_trigger_with_response_; }
|
Trigger<JsonObjectConst, Ts...> *get_success_trigger_with_response() const {
|
||||||
|
return this->success_trigger_with_response_;
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
Trigger<Ts...> *get_success_trigger() { return &this->success_trigger_; }
|
Trigger<Ts...> *get_success_trigger() const { return this->success_trigger_; }
|
||||||
Trigger<std::string, Ts...> *get_error_trigger() { return &this->error_trigger_; }
|
Trigger<std::string, Ts...> *get_error_trigger() const { return this->error_trigger_; }
|
||||||
#endif // USE_API_HOMEASSISTANT_ACTION_RESPONSES
|
#endif // USE_API_HOMEASSISTANT_ACTION_RESPONSES
|
||||||
|
|
||||||
void play(const Ts &...x) override {
|
void play(const Ts &...x) override {
|
||||||
@@ -187,14 +187,14 @@ template<typename... Ts> class HomeAssistantServiceCallAction : public Action<Ts
|
|||||||
if (response.is_success()) {
|
if (response.is_success()) {
|
||||||
#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES_JSON
|
#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES_JSON
|
||||||
if (this->flags_.wants_response) {
|
if (this->flags_.wants_response) {
|
||||||
this->success_trigger_with_response_.trigger(response.get_json(), args...);
|
this->success_trigger_with_response_->trigger(response.get_json(), args...);
|
||||||
} else
|
} else
|
||||||
#endif
|
#endif
|
||||||
{
|
{
|
||||||
this->success_trigger_.trigger(args...);
|
this->success_trigger_->trigger(args...);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
this->error_trigger_.trigger(response.get_error_message(), args...);
|
this->error_trigger_->trigger(response.get_error_message(), args...);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
captured_args);
|
captured_args);
|
||||||
@@ -251,10 +251,10 @@ template<typename... Ts> class HomeAssistantServiceCallAction : public Action<Ts
|
|||||||
#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES
|
#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES
|
||||||
#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES_JSON
|
#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES_JSON
|
||||||
TemplatableStringValue<Ts...> response_template_{""};
|
TemplatableStringValue<Ts...> response_template_{""};
|
||||||
Trigger<JsonObjectConst, Ts...> success_trigger_with_response_;
|
Trigger<JsonObjectConst, Ts...> *success_trigger_with_response_ = new Trigger<JsonObjectConst, Ts...>();
|
||||||
#endif // USE_API_HOMEASSISTANT_ACTION_RESPONSES_JSON
|
#endif // USE_API_HOMEASSISTANT_ACTION_RESPONSES_JSON
|
||||||
Trigger<Ts...> success_trigger_;
|
Trigger<Ts...> *success_trigger_ = new Trigger<Ts...>();
|
||||||
Trigger<std::string, Ts...> error_trigger_;
|
Trigger<std::string, Ts...> *error_trigger_ = new Trigger<std::string, Ts...>();
|
||||||
#endif // USE_API_HOMEASSISTANT_ACTION_RESPONSES
|
#endif // USE_API_HOMEASSISTANT_ACTION_RESPONSES
|
||||||
|
|
||||||
struct Flags {
|
struct Flags {
|
||||||
|
|||||||
@@ -112,12 +112,8 @@ class ProtoVarInt {
|
|||||||
uint64_t result = buffer[0] & 0x7F;
|
uint64_t result = buffer[0] & 0x7F;
|
||||||
uint8_t bitpos = 7;
|
uint8_t bitpos = 7;
|
||||||
|
|
||||||
// A 64-bit varint is at most 10 bytes (ceil(64/7)). Reject overlong encodings
|
|
||||||
// to avoid undefined behavior from shifting uint64_t by >= 64 bits.
|
|
||||||
uint32_t max_len = std::min(len, uint32_t(10));
|
|
||||||
|
|
||||||
// Start from the second byte since we've already processed the first
|
// Start from the second byte since we've already processed the first
|
||||||
for (uint32_t i = 1; i < max_len; i++) {
|
for (uint32_t i = 1; i < len; i++) {
|
||||||
uint8_t val = buffer[i];
|
uint8_t val = buffer[i];
|
||||||
result |= uint64_t(val & 0x7F) << uint64_t(bitpos);
|
result |= uint64_t(val & 0x7F) << uint64_t(bitpos);
|
||||||
bitpos += 7;
|
bitpos += 7;
|
||||||
@@ -406,20 +402,6 @@ class DumpBuffer {
|
|||||||
const char *c_str() const { return buf_; }
|
const char *c_str() const { return buf_; }
|
||||||
size_t size() const { return pos_; }
|
size_t size() const { return pos_; }
|
||||||
|
|
||||||
/// Get writable buffer pointer for use with buf_append_printf
|
|
||||||
char *data() { return buf_; }
|
|
||||||
/// Get current position for use with buf_append_printf
|
|
||||||
size_t pos() const { return pos_; }
|
|
||||||
/// Update position after buf_append_printf call
|
|
||||||
void set_pos(size_t pos) {
|
|
||||||
if (pos >= CAPACITY) {
|
|
||||||
pos_ = CAPACITY - 1;
|
|
||||||
} else {
|
|
||||||
pos_ = pos;
|
|
||||||
}
|
|
||||||
buf_[pos_] = '\0';
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void append_impl_(const char *str, size_t len) {
|
void append_impl_(const char *str, size_t len) {
|
||||||
size_t space = CAPACITY - 1 - pos_;
|
size_t space = CAPACITY - 1 - pos_;
|
||||||
@@ -961,16 +943,32 @@ class ProtoService {
|
|||||||
virtual bool is_connection_setup() = 0;
|
virtual bool is_connection_setup() = 0;
|
||||||
virtual void on_fatal_error() = 0;
|
virtual void on_fatal_error() = 0;
|
||||||
virtual void on_no_setup_connection() = 0;
|
virtual void on_no_setup_connection() = 0;
|
||||||
|
/**
|
||||||
|
* Create a buffer with a reserved size.
|
||||||
|
* @param reserve_size The number of bytes to pre-allocate in the buffer. This is a hint
|
||||||
|
* to optimize memory usage and avoid reallocations during encoding.
|
||||||
|
* Implementations should aim to allocate at least this size.
|
||||||
|
* @return A ProtoWriteBuffer object with the reserved size.
|
||||||
|
*/
|
||||||
|
virtual ProtoWriteBuffer create_buffer(uint32_t reserve_size) = 0;
|
||||||
virtual bool send_buffer(ProtoWriteBuffer buffer, uint8_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, const uint8_t *msg_data) = 0;
|
virtual void read_message(uint32_t msg_size, uint32_t msg_type, const uint8_t *msg_data) = 0;
|
||||||
/**
|
|
||||||
* Send a protobuf message by calculating its size, allocating a buffer, encoding, and sending.
|
// Optimized method that pre-allocates buffer based on message size
|
||||||
* This is the implementation method - callers should use send_message() which adds logging.
|
bool send_message_(const ProtoMessage &msg, uint8_t message_type) {
|
||||||
* @param msg The protobuf message to send.
|
ProtoSize size;
|
||||||
* @param message_type The message type identifier.
|
msg.calculate_size(size);
|
||||||
* @return True if the message was sent successfully, false otherwise.
|
uint32_t msg_size = size.get_size();
|
||||||
*/
|
|
||||||
virtual bool send_message_impl(const ProtoMessage &msg, uint8_t message_type) = 0;
|
// Create a pre-sized buffer
|
||||||
|
auto buffer = this->create_buffer(msg_size);
|
||||||
|
|
||||||
|
// Encode message into the buffer
|
||||||
|
msg.encode(buffer);
|
||||||
|
|
||||||
|
// Send the buffer
|
||||||
|
return this->send_buffer(buffer, message_type);
|
||||||
|
}
|
||||||
|
|
||||||
// Authentication helper methods
|
// Authentication helper methods
|
||||||
inline bool check_connection_setup_() {
|
inline bool check_connection_setup_() {
|
||||||
|
|||||||
@@ -264,9 +264,9 @@ template<typename... Ts> class APIRespondAction : public Action<Ts...> {
|
|||||||
// Build and send JSON response
|
// Build and send JSON response
|
||||||
json::JsonBuilder builder;
|
json::JsonBuilder builder;
|
||||||
this->json_builder_(x..., builder.root());
|
this->json_builder_(x..., builder.root());
|
||||||
std::string json_str = builder.serialize();
|
auto json_buf = builder.serialize();
|
||||||
this->parent_->send_action_response(call_id, success, StringRef(error_message),
|
this->parent_->send_action_response(call_id, success, StringRef(error_message),
|
||||||
reinterpret_cast<const uint8_t *>(json_str.data()), json_str.size());
|
reinterpret_cast<const uint8_t *>(json_buf.data()), json_buf.size());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <algorithm>
|
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
#include <limits>
|
#include <limits>
|
||||||
#include "abstract_aqi_calculator.h"
|
#include "abstract_aqi_calculator.h"
|
||||||
@@ -15,11 +14,7 @@ class AQICalculator : public AbstractAQICalculator {
|
|||||||
float pm2_5_index = calculate_index(pm2_5_value, PM2_5_GRID);
|
float pm2_5_index = calculate_index(pm2_5_value, PM2_5_GRID);
|
||||||
float pm10_0_index = calculate_index(pm10_0_value, PM10_0_GRID);
|
float pm10_0_index = calculate_index(pm10_0_value, PM10_0_GRID);
|
||||||
|
|
||||||
float aqi = std::max(pm2_5_index, pm10_0_index);
|
return static_cast<uint16_t>(std::round((pm2_5_index < pm10_0_index) ? pm10_0_index : pm2_5_index));
|
||||||
if (aqi < 0.0f) {
|
|
||||||
aqi = 0.0f;
|
|
||||||
}
|
|
||||||
return static_cast<uint16_t>(std::lround(aqi));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@@ -27,27 +22,13 @@ class AQICalculator : public AbstractAQICalculator {
|
|||||||
|
|
||||||
static constexpr int INDEX_GRID[NUM_LEVELS][2] = {{0, 50}, {51, 100}, {101, 150}, {151, 200}, {201, 300}, {301, 500}};
|
static constexpr int INDEX_GRID[NUM_LEVELS][2] = {{0, 50}, {51, 100}, {101, 150}, {151, 200}, {201, 300}, {301, 500}};
|
||||||
|
|
||||||
static constexpr float PM2_5_GRID[NUM_LEVELS][2] = {
|
static constexpr float PM2_5_GRID[NUM_LEVELS][2] = {{0.0f, 9.0f}, {9.1f, 35.4f},
|
||||||
// clang-format off
|
{35.5f, 55.4f}, {55.5f, 125.4f},
|
||||||
{0.0f, 9.1f},
|
{125.5f, 225.4f}, {225.5f, std::numeric_limits<float>::max()}};
|
||||||
{9.1f, 35.5f},
|
|
||||||
{35.5f, 55.5f},
|
|
||||||
{55.5f, 125.5f},
|
|
||||||
{125.5f, 225.5f},
|
|
||||||
{225.5f, std::numeric_limits<float>::max()}
|
|
||||||
// clang-format on
|
|
||||||
};
|
|
||||||
|
|
||||||
static constexpr float PM10_0_GRID[NUM_LEVELS][2] = {
|
static constexpr float PM10_0_GRID[NUM_LEVELS][2] = {{0.0f, 54.0f}, {55.0f, 154.0f},
|
||||||
// clang-format off
|
{155.0f, 254.0f}, {255.0f, 354.0f},
|
||||||
{0.0f, 55.0f},
|
{355.0f, 424.0f}, {425.0f, std::numeric_limits<float>::max()}};
|
||||||
{55.0f, 155.0f},
|
|
||||||
{155.0f, 255.0f},
|
|
||||||
{255.0f, 355.0f},
|
|
||||||
{355.0f, 425.0f},
|
|
||||||
{425.0f, std::numeric_limits<float>::max()}
|
|
||||||
// clang-format on
|
|
||||||
};
|
|
||||||
|
|
||||||
static float calculate_index(float value, const float array[NUM_LEVELS][2]) {
|
static float calculate_index(float value, const float array[NUM_LEVELS][2]) {
|
||||||
int grid_index = get_grid_index(value, array);
|
int grid_index = get_grid_index(value, array);
|
||||||
@@ -64,10 +45,7 @@ class AQICalculator : public AbstractAQICalculator {
|
|||||||
|
|
||||||
static int get_grid_index(float value, const float array[NUM_LEVELS][2]) {
|
static int get_grid_index(float value, const float array[NUM_LEVELS][2]) {
|
||||||
for (int i = 0; i < NUM_LEVELS; i++) {
|
for (int i = 0; i < NUM_LEVELS; i++) {
|
||||||
const bool in_range =
|
if (value >= array[i][0] && value <= array[i][1]) {
|
||||||
(value >= array[i][0]) && ((i == NUM_LEVELS - 1) ? (value <= array[i][1]) // last bucket inclusive
|
|
||||||
: (value < array[i][1])); // others exclusive on hi
|
|
||||||
if (in_range) {
|
|
||||||
return i;
|
return i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ class AQISensor : public sensor::Sensor, public Component {
|
|||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override { return setup_priority::DATA; }
|
||||||
|
|
||||||
void set_pm_2_5_sensor(sensor::Sensor *sensor) { this->pm_2_5_sensor_ = sensor; }
|
void set_pm_2_5_sensor(sensor::Sensor *sensor) { this->pm_2_5_sensor_ = sensor; }
|
||||||
void set_pm_10_0_sensor(sensor::Sensor *sensor) { this->pm_10_0_sensor_ = sensor; }
|
void set_pm_10_0_sensor(sensor::Sensor *sensor) { this->pm_10_0_sensor_ = sensor; }
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <algorithm>
|
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
#include <limits>
|
#include <limits>
|
||||||
#include "abstract_aqi_calculator.h"
|
#include "abstract_aqi_calculator.h"
|
||||||
@@ -13,11 +12,7 @@ class CAQICalculator : public AbstractAQICalculator {
|
|||||||
float pm2_5_index = calculate_index(pm2_5_value, PM2_5_GRID);
|
float pm2_5_index = calculate_index(pm2_5_value, PM2_5_GRID);
|
||||||
float pm10_0_index = calculate_index(pm10_0_value, PM10_0_GRID);
|
float pm10_0_index = calculate_index(pm10_0_value, PM10_0_GRID);
|
||||||
|
|
||||||
float aqi = std::max(pm2_5_index, pm10_0_index);
|
return static_cast<uint16_t>(std::round((pm2_5_index < pm10_0_index) ? pm10_0_index : pm2_5_index));
|
||||||
if (aqi < 0.0f) {
|
|
||||||
aqi = 0.0f;
|
|
||||||
}
|
|
||||||
return static_cast<uint16_t>(std::lround(aqi));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@@ -26,24 +21,10 @@ class CAQICalculator : public AbstractAQICalculator {
|
|||||||
static constexpr int INDEX_GRID[NUM_LEVELS][2] = {{0, 25}, {26, 50}, {51, 75}, {76, 100}, {101, 400}};
|
static constexpr int INDEX_GRID[NUM_LEVELS][2] = {{0, 25}, {26, 50}, {51, 75}, {76, 100}, {101, 400}};
|
||||||
|
|
||||||
static constexpr float PM2_5_GRID[NUM_LEVELS][2] = {
|
static constexpr float PM2_5_GRID[NUM_LEVELS][2] = {
|
||||||
// clang-format off
|
{0.0f, 15.0f}, {15.1f, 30.0f}, {30.1f, 55.0f}, {55.1f, 110.0f}, {110.1f, std::numeric_limits<float>::max()}};
|
||||||
{0.0f, 15.1f},
|
|
||||||
{15.1f, 30.1f},
|
|
||||||
{30.1f, 55.1f},
|
|
||||||
{55.1f, 110.1f},
|
|
||||||
{110.1f, std::numeric_limits<float>::max()}
|
|
||||||
// clang-format on
|
|
||||||
};
|
|
||||||
|
|
||||||
static constexpr float PM10_0_GRID[NUM_LEVELS][2] = {
|
static constexpr float PM10_0_GRID[NUM_LEVELS][2] = {
|
||||||
// clang-format off
|
{0.0f, 25.0f}, {25.1f, 50.0f}, {50.1f, 90.0f}, {90.1f, 180.0f}, {180.1f, std::numeric_limits<float>::max()}};
|
||||||
{0.0f, 25.1f},
|
|
||||||
{25.1f, 50.1f},
|
|
||||||
{50.1f, 90.1f},
|
|
||||||
{90.1f, 180.1f},
|
|
||||||
{180.1f, std::numeric_limits<float>::max()}
|
|
||||||
// clang-format on
|
|
||||||
};
|
|
||||||
|
|
||||||
static float calculate_index(float value, const float array[NUM_LEVELS][2]) {
|
static float calculate_index(float value, const float array[NUM_LEVELS][2]) {
|
||||||
int grid_index = get_grid_index(value, array);
|
int grid_index = get_grid_index(value, array);
|
||||||
@@ -61,10 +42,7 @@ class CAQICalculator : public AbstractAQICalculator {
|
|||||||
|
|
||||||
static int get_grid_index(float value, const float array[NUM_LEVELS][2]) {
|
static int get_grid_index(float value, const float array[NUM_LEVELS][2]) {
|
||||||
for (int i = 0; i < NUM_LEVELS; i++) {
|
for (int i = 0; i < NUM_LEVELS; i++) {
|
||||||
const bool in_range =
|
if (value >= array[i][0] && value <= array[i][1]) {
|
||||||
(value >= array[i][0]) && ((i == NUM_LEVELS - 1) ? (value <= array[i][1]) // last bucket inclusive
|
|
||||||
: (value < array[i][1])); // others exclusive on hi
|
|
||||||
if (in_range) {
|
|
||||||
return i;
|
return i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -41,6 +41,8 @@ void AS3935Component::dump_config() {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float AS3935Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void AS3935Component::loop() {
|
void AS3935Component::loop() {
|
||||||
if (!this->irq_pin_->digital_read())
|
if (!this->irq_pin_->digital_read())
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -74,6 +74,7 @@ class AS3935Component : public Component {
|
|||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void loop() override;
|
void loop() override;
|
||||||
|
|
||||||
void set_irq_pin(GPIOPin *irq_pin) { irq_pin_ = irq_pin; }
|
void set_irq_pin(GPIOPin *irq_pin) { irq_pin_ = irq_pin; }
|
||||||
|
|||||||
@@ -22,6 +22,8 @@ static const uint8_t REGISTER_STATUS = 0x0B; // 8 bytes / R
|
|||||||
static const uint8_t REGISTER_AGC = 0x1A; // 8 bytes / R
|
static const uint8_t REGISTER_AGC = 0x1A; // 8 bytes / R
|
||||||
static const uint8_t REGISTER_MAGNITUDE = 0x1B; // 16 bytes / R
|
static const uint8_t REGISTER_MAGNITUDE = 0x1B; // 16 bytes / R
|
||||||
|
|
||||||
|
float AS5600Sensor::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void AS5600Sensor::dump_config() {
|
void AS5600Sensor::dump_config() {
|
||||||
LOG_SENSOR("", "AS5600 Sensor", this);
|
LOG_SENSOR("", "AS5600 Sensor", this);
|
||||||
ESP_LOGCONFIG(TAG, " Out of Range Mode: %u", this->out_of_range_mode_);
|
ESP_LOGCONFIG(TAG, " Out of Range Mode: %u", this->out_of_range_mode_);
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ class AS5600Sensor : public PollingComponent, public Parented<AS5600Component>,
|
|||||||
public:
|
public:
|
||||||
void update() override;
|
void update() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
void set_angle_sensor(sensor::Sensor *angle_sensor) { this->angle_sensor_ = angle_sensor; }
|
void set_angle_sensor(sensor::Sensor *angle_sensor) { this->angle_sensor_ = angle_sensor; }
|
||||||
void set_raw_angle_sensor(sensor::Sensor *raw_angle_sensor) { this->raw_angle_sensor_ = raw_angle_sensor; }
|
void set_raw_angle_sensor(sensor::Sensor *raw_angle_sensor) { this->raw_angle_sensor_ = raw_angle_sensor; }
|
||||||
|
|||||||
@@ -58,6 +58,8 @@ void AS7341Component::dump_config() {
|
|||||||
LOG_SENSOR(" ", "NIR", this->nir_);
|
LOG_SENSOR(" ", "NIR", this->nir_);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float AS7341Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void AS7341Component::update() {
|
void AS7341Component::update() {
|
||||||
this->read_channels(this->channel_readings_);
|
this->read_channels(this->channel_readings_);
|
||||||
|
|
||||||
|
|||||||
@@ -78,6 +78,7 @@ class AS7341Component : public PollingComponent, public i2c::I2CDevice {
|
|||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void update() override;
|
void update() override;
|
||||||
|
|
||||||
void set_f1_sensor(sensor::Sensor *f1_sensor) { this->f1_ = f1_sensor; }
|
void set_f1_sensor(sensor::Sensor *f1_sensor) { this->f1_ = f1_sensor; }
|
||||||
|
|||||||
@@ -146,6 +146,7 @@ void ATM90E26Component::dump_config() {
|
|||||||
LOG_SENSOR(" ", "Active Reverse Energy A", this->reverse_active_energy_sensor_);
|
LOG_SENSOR(" ", "Active Reverse Energy A", this->reverse_active_energy_sensor_);
|
||||||
LOG_SENSOR(" ", "Frequency", this->freq_sensor_);
|
LOG_SENSOR(" ", "Frequency", this->freq_sensor_);
|
||||||
}
|
}
|
||||||
|
float ATM90E26Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
uint16_t ATM90E26Component::read16_(uint8_t a_register) {
|
uint16_t ATM90E26Component::read16_(uint8_t a_register) {
|
||||||
uint8_t data[2];
|
uint8_t data[2];
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ class ATM90E26Component : public PollingComponent,
|
|||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void update() override;
|
void update() override;
|
||||||
|
|
||||||
void set_voltage_sensor(sensor::Sensor *obj) { this->voltage_sensor_ = obj; }
|
void set_voltage_sensor(sensor::Sensor *obj) { this->voltage_sensor_ = obj; }
|
||||||
|
|||||||
@@ -6,7 +6,8 @@ namespace bang_bang {
|
|||||||
|
|
||||||
static const char *const TAG = "bang_bang.climate";
|
static const char *const TAG = "bang_bang.climate";
|
||||||
|
|
||||||
BangBangClimate::BangBangClimate() = default;
|
BangBangClimate::BangBangClimate()
|
||||||
|
: idle_trigger_(new Trigger<>()), cool_trigger_(new Trigger<>()), heat_trigger_(new Trigger<>()) {}
|
||||||
|
|
||||||
void BangBangClimate::setup() {
|
void BangBangClimate::setup() {
|
||||||
this->sensor_->add_on_state_callback([this](float state) {
|
this->sensor_->add_on_state_callback([this](float state) {
|
||||||
@@ -159,13 +160,13 @@ void BangBangClimate::switch_to_action_(climate::ClimateAction action) {
|
|||||||
switch (action) {
|
switch (action) {
|
||||||
case climate::CLIMATE_ACTION_OFF:
|
case climate::CLIMATE_ACTION_OFF:
|
||||||
case climate::CLIMATE_ACTION_IDLE:
|
case climate::CLIMATE_ACTION_IDLE:
|
||||||
trig = &this->idle_trigger_;
|
trig = this->idle_trigger_;
|
||||||
break;
|
break;
|
||||||
case climate::CLIMATE_ACTION_COOLING:
|
case climate::CLIMATE_ACTION_COOLING:
|
||||||
trig = &this->cool_trigger_;
|
trig = this->cool_trigger_;
|
||||||
break;
|
break;
|
||||||
case climate::CLIMATE_ACTION_HEATING:
|
case climate::CLIMATE_ACTION_HEATING:
|
||||||
trig = &this->heat_trigger_;
|
trig = this->heat_trigger_;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
trig = nullptr;
|
trig = nullptr;
|
||||||
@@ -203,9 +204,9 @@ void BangBangClimate::set_away_config(const BangBangClimateTargetTempConfig &awa
|
|||||||
void BangBangClimate::set_sensor(sensor::Sensor *sensor) { this->sensor_ = sensor; }
|
void BangBangClimate::set_sensor(sensor::Sensor *sensor) { this->sensor_ = sensor; }
|
||||||
void BangBangClimate::set_humidity_sensor(sensor::Sensor *humidity_sensor) { this->humidity_sensor_ = humidity_sensor; }
|
void BangBangClimate::set_humidity_sensor(sensor::Sensor *humidity_sensor) { this->humidity_sensor_ = humidity_sensor; }
|
||||||
|
|
||||||
Trigger<> *BangBangClimate::get_idle_trigger() { return &this->idle_trigger_; }
|
Trigger<> *BangBangClimate::get_idle_trigger() const { return this->idle_trigger_; }
|
||||||
Trigger<> *BangBangClimate::get_cool_trigger() { return &this->cool_trigger_; }
|
Trigger<> *BangBangClimate::get_cool_trigger() const { return this->cool_trigger_; }
|
||||||
Trigger<> *BangBangClimate::get_heat_trigger() { return &this->heat_trigger_; }
|
Trigger<> *BangBangClimate::get_heat_trigger() const { return this->heat_trigger_; }
|
||||||
|
|
||||||
void BangBangClimate::set_supports_cool(bool supports_cool) { this->supports_cool_ = supports_cool; }
|
void BangBangClimate::set_supports_cool(bool supports_cool) { this->supports_cool_ = supports_cool; }
|
||||||
void BangBangClimate::set_supports_heat(bool supports_heat) { this->supports_heat_ = supports_heat; }
|
void BangBangClimate::set_supports_heat(bool supports_heat) { this->supports_heat_ = supports_heat; }
|
||||||
|
|||||||
@@ -30,9 +30,9 @@ class BangBangClimate : public climate::Climate, public Component {
|
|||||||
void set_normal_config(const BangBangClimateTargetTempConfig &normal_config);
|
void set_normal_config(const BangBangClimateTargetTempConfig &normal_config);
|
||||||
void set_away_config(const BangBangClimateTargetTempConfig &away_config);
|
void set_away_config(const BangBangClimateTargetTempConfig &away_config);
|
||||||
|
|
||||||
Trigger<> *get_idle_trigger();
|
Trigger<> *get_idle_trigger() const;
|
||||||
Trigger<> *get_cool_trigger();
|
Trigger<> *get_cool_trigger() const;
|
||||||
Trigger<> *get_heat_trigger();
|
Trigger<> *get_heat_trigger() const;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/// Override control to change settings of the climate device.
|
/// Override control to change settings of the climate device.
|
||||||
@@ -57,13 +57,17 @@ class BangBangClimate : public climate::Climate, public Component {
|
|||||||
*
|
*
|
||||||
* In idle mode, the controller is assumed to have both heating and cooling disabled.
|
* In idle mode, the controller is assumed to have both heating and cooling disabled.
|
||||||
*/
|
*/
|
||||||
Trigger<> idle_trigger_;
|
Trigger<> *idle_trigger_{nullptr};
|
||||||
/** The trigger to call when the controller should switch to cooling mode.
|
/** The trigger to call when the controller should switch to cooling mode.
|
||||||
*/
|
*/
|
||||||
Trigger<> cool_trigger_;
|
Trigger<> *cool_trigger_{nullptr};
|
||||||
/** The trigger to call when the controller should switch to heating mode.
|
/** The trigger to call when the controller should switch to heating mode.
|
||||||
|
*
|
||||||
|
* A null value for this attribute means that the controller has no heating action
|
||||||
|
* For example window blinds, where only cooling (blinds closed) and not-cooling
|
||||||
|
* (blinds open) is possible.
|
||||||
*/
|
*/
|
||||||
Trigger<> heat_trigger_;
|
Trigger<> *heat_trigger_{nullptr};
|
||||||
/** A reference to the trigger that was previously active.
|
/** A reference to the trigger that was previously active.
|
||||||
*
|
*
|
||||||
* This is so that the previous trigger can be stopped before enabling a new one.
|
* This is so that the previous trigger can be stopped before enabling a new one.
|
||||||
|
|||||||
@@ -265,4 +265,6 @@ void BH1750Sensor::fail_and_reset_() {
|
|||||||
this->state_ = IDLE;
|
this->state_ = IDLE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float BH1750Sensor::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
} // namespace esphome::bh1750
|
} // namespace esphome::bh1750
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ class BH1750Sensor : public sensor::Sensor, public PollingComponent, public i2c:
|
|||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
void update() override;
|
void update() override;
|
||||||
void loop() override;
|
void loop() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
// State machine states
|
// State machine states
|
||||||
|
|||||||
@@ -5,14 +5,6 @@ namespace esphome::binary_sensor {
|
|||||||
|
|
||||||
static const char *const TAG = "binary_sensor.automation";
|
static const char *const TAG = "binary_sensor.automation";
|
||||||
|
|
||||||
// MultiClickTrigger timeout IDs.
|
|
||||||
// MultiClickTrigger is its own Component instance, so the scheduler scopes
|
|
||||||
// IDs by component pointer — no risk of collisions between instances.
|
|
||||||
constexpr uint32_t MULTICLICK_TRIGGER_ID = 0;
|
|
||||||
constexpr uint32_t MULTICLICK_COOLDOWN_ID = 1;
|
|
||||||
constexpr uint32_t MULTICLICK_IS_VALID_ID = 2;
|
|
||||||
constexpr uint32_t MULTICLICK_IS_NOT_VALID_ID = 3;
|
|
||||||
|
|
||||||
void MultiClickTrigger::on_state_(bool state) {
|
void MultiClickTrigger::on_state_(bool state) {
|
||||||
// Handle duplicate events
|
// Handle duplicate events
|
||||||
if (state == this->last_state_) {
|
if (state == this->last_state_) {
|
||||||
@@ -35,7 +27,7 @@ void MultiClickTrigger::on_state_(bool state) {
|
|||||||
evt.min_length, evt.max_length);
|
evt.min_length, evt.max_length);
|
||||||
this->at_index_ = 1;
|
this->at_index_ = 1;
|
||||||
if (this->timing_.size() == 1 && evt.max_length == 4294967294UL) {
|
if (this->timing_.size() == 1 && evt.max_length == 4294967294UL) {
|
||||||
this->set_timeout(MULTICLICK_TRIGGER_ID, evt.min_length, [this]() { this->trigger_(); });
|
this->set_timeout("trigger", evt.min_length, [this]() { this->trigger_(); });
|
||||||
} else {
|
} else {
|
||||||
this->schedule_is_valid_(evt.min_length);
|
this->schedule_is_valid_(evt.min_length);
|
||||||
this->schedule_is_not_valid_(evt.max_length);
|
this->schedule_is_not_valid_(evt.max_length);
|
||||||
@@ -65,13 +57,13 @@ void MultiClickTrigger::on_state_(bool state) {
|
|||||||
this->schedule_is_not_valid_(evt.max_length);
|
this->schedule_is_not_valid_(evt.max_length);
|
||||||
} else if (*this->at_index_ + 1 != this->timing_.size()) {
|
} else if (*this->at_index_ + 1 != this->timing_.size()) {
|
||||||
ESP_LOGV(TAG, "B i=%zu min=%" PRIu32, *this->at_index_, evt.min_length); // NOLINT
|
ESP_LOGV(TAG, "B i=%zu min=%" PRIu32, *this->at_index_, evt.min_length); // NOLINT
|
||||||
this->cancel_timeout(MULTICLICK_IS_NOT_VALID_ID);
|
this->cancel_timeout("is_not_valid");
|
||||||
this->schedule_is_valid_(evt.min_length);
|
this->schedule_is_valid_(evt.min_length);
|
||||||
} else {
|
} else {
|
||||||
ESP_LOGV(TAG, "C i=%zu min=%" PRIu32, *this->at_index_, evt.min_length); // NOLINT
|
ESP_LOGV(TAG, "C i=%zu min=%" PRIu32, *this->at_index_, evt.min_length); // NOLINT
|
||||||
this->is_valid_ = false;
|
this->is_valid_ = false;
|
||||||
this->cancel_timeout(MULTICLICK_IS_NOT_VALID_ID);
|
this->cancel_timeout("is_not_valid");
|
||||||
this->set_timeout(MULTICLICK_TRIGGER_ID, evt.min_length, [this]() { this->trigger_(); });
|
this->set_timeout("trigger", evt.min_length, [this]() { this->trigger_(); });
|
||||||
}
|
}
|
||||||
|
|
||||||
*this->at_index_ = *this->at_index_ + 1;
|
*this->at_index_ = *this->at_index_ + 1;
|
||||||
@@ -79,14 +71,14 @@ void MultiClickTrigger::on_state_(bool state) {
|
|||||||
void MultiClickTrigger::schedule_cooldown_() {
|
void MultiClickTrigger::schedule_cooldown_() {
|
||||||
ESP_LOGV(TAG, "Multi Click: Invalid length of press, starting cooldown of %" PRIu32 " ms", this->invalid_cooldown_);
|
ESP_LOGV(TAG, "Multi Click: Invalid length of press, starting cooldown of %" PRIu32 " ms", this->invalid_cooldown_);
|
||||||
this->is_in_cooldown_ = true;
|
this->is_in_cooldown_ = true;
|
||||||
this->set_timeout(MULTICLICK_COOLDOWN_ID, this->invalid_cooldown_, [this]() {
|
this->set_timeout("cooldown", this->invalid_cooldown_, [this]() {
|
||||||
ESP_LOGV(TAG, "Multi Click: Cooldown ended, matching is now enabled again.");
|
ESP_LOGV(TAG, "Multi Click: Cooldown ended, matching is now enabled again.");
|
||||||
this->is_in_cooldown_ = false;
|
this->is_in_cooldown_ = false;
|
||||||
});
|
});
|
||||||
this->at_index_.reset();
|
this->at_index_.reset();
|
||||||
this->cancel_timeout(MULTICLICK_TRIGGER_ID);
|
this->cancel_timeout("trigger");
|
||||||
this->cancel_timeout(MULTICLICK_IS_VALID_ID);
|
this->cancel_timeout("is_valid");
|
||||||
this->cancel_timeout(MULTICLICK_IS_NOT_VALID_ID);
|
this->cancel_timeout("is_not_valid");
|
||||||
}
|
}
|
||||||
void MultiClickTrigger::schedule_is_valid_(uint32_t min_length) {
|
void MultiClickTrigger::schedule_is_valid_(uint32_t min_length) {
|
||||||
if (min_length == 0) {
|
if (min_length == 0) {
|
||||||
@@ -94,13 +86,13 @@ void MultiClickTrigger::schedule_is_valid_(uint32_t min_length) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this->is_valid_ = false;
|
this->is_valid_ = false;
|
||||||
this->set_timeout(MULTICLICK_IS_VALID_ID, min_length, [this]() {
|
this->set_timeout("is_valid", min_length, [this]() {
|
||||||
ESP_LOGV(TAG, "Multi Click: You can now %s the button.", this->parent_->state ? "RELEASE" : "PRESS");
|
ESP_LOGV(TAG, "Multi Click: You can now %s the button.", this->parent_->state ? "RELEASE" : "PRESS");
|
||||||
this->is_valid_ = true;
|
this->is_valid_ = true;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
void MultiClickTrigger::schedule_is_not_valid_(uint32_t max_length) {
|
void MultiClickTrigger::schedule_is_not_valid_(uint32_t max_length) {
|
||||||
this->set_timeout(MULTICLICK_IS_NOT_VALID_ID, max_length, [this]() {
|
this->set_timeout("is_not_valid", max_length, [this]() {
|
||||||
ESP_LOGV(TAG, "Multi Click: You waited too long to %s.", this->parent_->state ? "RELEASE" : "PRESS");
|
ESP_LOGV(TAG, "Multi Click: You waited too long to %s.", this->parent_->state ? "RELEASE" : "PRESS");
|
||||||
this->is_valid_ = false;
|
this->is_valid_ = false;
|
||||||
this->schedule_cooldown_();
|
this->schedule_cooldown_();
|
||||||
@@ -114,9 +106,9 @@ void MultiClickTrigger::cancel() {
|
|||||||
void MultiClickTrigger::trigger_() {
|
void MultiClickTrigger::trigger_() {
|
||||||
ESP_LOGV(TAG, "Multi Click: Hooray, multi click is valid. Triggering!");
|
ESP_LOGV(TAG, "Multi Click: Hooray, multi click is valid. Triggering!");
|
||||||
this->at_index_.reset();
|
this->at_index_.reset();
|
||||||
this->cancel_timeout(MULTICLICK_TRIGGER_ID);
|
this->cancel_timeout("trigger");
|
||||||
this->cancel_timeout(MULTICLICK_IS_VALID_ID);
|
this->cancel_timeout("is_valid");
|
||||||
this->cancel_timeout(MULTICLICK_IS_NOT_VALID_ID);
|
this->cancel_timeout("is_not_valid");
|
||||||
this->trigger();
|
this->trigger();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -6,14 +6,6 @@ namespace esphome::binary_sensor {
|
|||||||
|
|
||||||
static const char *const TAG = "sensor.filter";
|
static const char *const TAG = "sensor.filter";
|
||||||
|
|
||||||
// Timeout IDs for filter classes.
|
|
||||||
// Each filter is its own Component instance, so the scheduler scopes
|
|
||||||
// IDs by component pointer — no risk of collisions between instances.
|
|
||||||
constexpr uint32_t FILTER_TIMEOUT_ID = 0;
|
|
||||||
// AutorepeatFilter needs two distinct IDs (both timeouts on the same component)
|
|
||||||
constexpr uint32_t AUTOREPEAT_TIMING_ID = 0;
|
|
||||||
constexpr uint32_t AUTOREPEAT_ON_OFF_ID = 1;
|
|
||||||
|
|
||||||
void Filter::output(bool value) {
|
void Filter::output(bool value) {
|
||||||
if (this->next_ == nullptr) {
|
if (this->next_ == nullptr) {
|
||||||
this->parent_->send_state_internal(value);
|
this->parent_->send_state_internal(value);
|
||||||
@@ -31,16 +23,16 @@ void Filter::input(bool value) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void TimeoutFilter::input(bool value) {
|
void TimeoutFilter::input(bool value) {
|
||||||
this->set_timeout(FILTER_TIMEOUT_ID, this->timeout_delay_.value(), [this]() { this->parent_->invalidate_state(); });
|
this->set_timeout("timeout", this->timeout_delay_.value(), [this]() { this->parent_->invalidate_state(); });
|
||||||
// we do not de-dup here otherwise changes from invalid to valid state will not be output
|
// we do not de-dup here otherwise changes from invalid to valid state will not be output
|
||||||
this->output(value);
|
this->output(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
optional<bool> DelayedOnOffFilter::new_value(bool value) {
|
optional<bool> DelayedOnOffFilter::new_value(bool value) {
|
||||||
if (value) {
|
if (value) {
|
||||||
this->set_timeout(FILTER_TIMEOUT_ID, this->on_delay_.value(), [this]() { this->output(true); });
|
this->set_timeout("ON_OFF", this->on_delay_.value(), [this]() { this->output(true); });
|
||||||
} else {
|
} else {
|
||||||
this->set_timeout(FILTER_TIMEOUT_ID, this->off_delay_.value(), [this]() { this->output(false); });
|
this->set_timeout("ON_OFF", this->off_delay_.value(), [this]() { this->output(false); });
|
||||||
}
|
}
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
@@ -49,10 +41,10 @@ float DelayedOnOffFilter::get_setup_priority() const { return setup_priority::HA
|
|||||||
|
|
||||||
optional<bool> DelayedOnFilter::new_value(bool value) {
|
optional<bool> DelayedOnFilter::new_value(bool value) {
|
||||||
if (value) {
|
if (value) {
|
||||||
this->set_timeout(FILTER_TIMEOUT_ID, this->delay_.value(), [this]() { this->output(true); });
|
this->set_timeout("ON", this->delay_.value(), [this]() { this->output(true); });
|
||||||
return {};
|
return {};
|
||||||
} else {
|
} else {
|
||||||
this->cancel_timeout(FILTER_TIMEOUT_ID);
|
this->cancel_timeout("ON");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -61,10 +53,10 @@ float DelayedOnFilter::get_setup_priority() const { return setup_priority::HARDW
|
|||||||
|
|
||||||
optional<bool> DelayedOffFilter::new_value(bool value) {
|
optional<bool> DelayedOffFilter::new_value(bool value) {
|
||||||
if (!value) {
|
if (!value) {
|
||||||
this->set_timeout(FILTER_TIMEOUT_ID, this->delay_.value(), [this]() { this->output(false); });
|
this->set_timeout("OFF", this->delay_.value(), [this]() { this->output(false); });
|
||||||
return {};
|
return {};
|
||||||
} else {
|
} else {
|
||||||
this->cancel_timeout(FILTER_TIMEOUT_ID);
|
this->cancel_timeout("OFF");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -84,8 +76,8 @@ optional<bool> AutorepeatFilter::new_value(bool value) {
|
|||||||
this->next_timing_();
|
this->next_timing_();
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
this->cancel_timeout(AUTOREPEAT_TIMING_ID);
|
this->cancel_timeout("TIMING");
|
||||||
this->cancel_timeout(AUTOREPEAT_ON_OFF_ID);
|
this->cancel_timeout("ON_OFF");
|
||||||
this->active_timing_ = 0;
|
this->active_timing_ = 0;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -96,10 +88,8 @@ void AutorepeatFilter::next_timing_() {
|
|||||||
// 1st time: starts waiting the first delay
|
// 1st time: starts waiting the first delay
|
||||||
// 2nd time: starts waiting the second delay and starts toggling with the first time_off / _on
|
// 2nd time: starts waiting the second delay and starts toggling with the first time_off / _on
|
||||||
// last time: no delay to start but have to bump the index to reflect the last
|
// last time: no delay to start but have to bump the index to reflect the last
|
||||||
if (this->active_timing_ < this->timings_.size()) {
|
if (this->active_timing_ < this->timings_.size())
|
||||||
this->set_timeout(AUTOREPEAT_TIMING_ID, this->timings_[this->active_timing_].delay,
|
this->set_timeout("TIMING", this->timings_[this->active_timing_].delay, [this]() { this->next_timing_(); });
|
||||||
[this]() { this->next_timing_(); });
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this->active_timing_ <= this->timings_.size()) {
|
if (this->active_timing_ <= this->timings_.size()) {
|
||||||
this->active_timing_++;
|
this->active_timing_++;
|
||||||
@@ -114,8 +104,7 @@ void AutorepeatFilter::next_timing_() {
|
|||||||
void AutorepeatFilter::next_value_(bool val) {
|
void AutorepeatFilter::next_value_(bool val) {
|
||||||
const AutorepeatFilterTiming &timing = this->timings_[this->active_timing_ - 2];
|
const AutorepeatFilterTiming &timing = this->timings_[this->active_timing_ - 2];
|
||||||
this->output(val); // This is at least the second one so not initial
|
this->output(val); // This is at least the second one so not initial
|
||||||
this->set_timeout(AUTOREPEAT_ON_OFF_ID, val ? timing.time_on : timing.time_off,
|
this->set_timeout("ON_OFF", val ? timing.time_on : timing.time_off, [this, val]() { this->next_value_(!val); });
|
||||||
[this, val]() { this->next_value_(!val); });
|
|
||||||
}
|
}
|
||||||
|
|
||||||
float AutorepeatFilter::get_setup_priority() const { return setup_priority::HARDWARE; }
|
float AutorepeatFilter::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||||
@@ -126,7 +115,7 @@ optional<bool> LambdaFilter::new_value(bool value) { return this->f_(value); }
|
|||||||
|
|
||||||
optional<bool> SettleFilter::new_value(bool value) {
|
optional<bool> SettleFilter::new_value(bool value) {
|
||||||
if (!this->steady_) {
|
if (!this->steady_) {
|
||||||
this->set_timeout(FILTER_TIMEOUT_ID, this->delay_.value(), [this, value]() {
|
this->set_timeout("SETTLE", this->delay_.value(), [this, value]() {
|
||||||
this->steady_ = true;
|
this->steady_ = true;
|
||||||
this->output(value);
|
this->output(value);
|
||||||
});
|
});
|
||||||
@@ -134,7 +123,7 @@ optional<bool> SettleFilter::new_value(bool value) {
|
|||||||
} else {
|
} else {
|
||||||
this->steady_ = false;
|
this->steady_ = false;
|
||||||
this->output(value);
|
this->output(value);
|
||||||
this->set_timeout(FILTER_TIMEOUT_ID, this->delay_.value(), [this]() { this->steady_ = true; });
|
this->set_timeout("SETTLE", this->delay_.value(), [this]() { this->steady_ = true; });
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -46,16 +46,16 @@ static const uint32_t PKT_TIMEOUT_MS = 200;
|
|||||||
|
|
||||||
void BL0942::loop() {
|
void BL0942::loop() {
|
||||||
DataPacket buffer;
|
DataPacket buffer;
|
||||||
size_t avail = this->available();
|
int avail = this->available();
|
||||||
|
|
||||||
if (!avail) {
|
if (!avail) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (avail < sizeof(buffer)) {
|
if (static_cast<size_t>(avail) < sizeof(buffer)) {
|
||||||
if (!this->rx_start_) {
|
if (!this->rx_start_) {
|
||||||
this->rx_start_ = millis();
|
this->rx_start_ = millis();
|
||||||
} else if (millis() > this->rx_start_ + PKT_TIMEOUT_MS) {
|
} else if (millis() > this->rx_start_ + PKT_TIMEOUT_MS) {
|
||||||
ESP_LOGW(TAG, "Junk on wire. Throwing away partial message (%zu bytes)", avail);
|
ESP_LOGW(TAG, "Junk on wire. Throwing away partial message (%d bytes)", avail);
|
||||||
this->read_array((uint8_t *) &buffer, avail);
|
this->read_array((uint8_t *) &buffer, avail);
|
||||||
this->rx_start_ = 0;
|
this->rx_start_ = 0;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -199,6 +199,7 @@ void BME280Component::dump_config() {
|
|||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
ESP_LOGCONFIG(TAG, " Oversampling: %s", oversampling_to_str(this->humidity_oversampling_));
|
ESP_LOGCONFIG(TAG, " Oversampling: %s", oversampling_to_str(this->humidity_oversampling_));
|
||||||
}
|
}
|
||||||
|
float BME280Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
inline uint8_t oversampling_to_time(BME280Oversampling over_sampling) { return (1 << uint8_t(over_sampling)) >> 1; }
|
inline uint8_t oversampling_to_time(BME280Oversampling over_sampling) { return (1 << uint8_t(over_sampling)) >> 1; }
|
||||||
|
|
||||||
|
|||||||
@@ -76,6 +76,7 @@ class BME280Component : public PollingComponent {
|
|||||||
// (In most use cases you won't need these)
|
// (In most use cases you won't need these)
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void update() override;
|
void update() override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|||||||
@@ -233,6 +233,8 @@ void BME680Component::dump_config() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float BME680Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void BME680Component::update() {
|
void BME680Component::update() {
|
||||||
uint8_t meas_control = 0; // No need to fetch, we're setting all fields
|
uint8_t meas_control = 0; // No need to fetch, we're setting all fields
|
||||||
meas_control |= (this->temperature_oversampling_ & 0b111) << 5;
|
meas_control |= (this->temperature_oversampling_ & 0b111) << 5;
|
||||||
|
|||||||
@@ -99,6 +99,7 @@ class BME680Component : public PollingComponent, public i2c::I2CDevice {
|
|||||||
// (In most use cases you won't need these)
|
// (In most use cases you won't need these)
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void update() override;
|
void update() override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|||||||
@@ -89,9 +89,8 @@ async def to_code(config):
|
|||||||
var.set_state_save_interval(config[CONF_STATE_SAVE_INTERVAL].total_milliseconds)
|
var.set_state_save_interval(config[CONF_STATE_SAVE_INTERVAL].total_milliseconds)
|
||||||
)
|
)
|
||||||
|
|
||||||
# Although this component does not use SPI/Wire directly, the BSEC library requires them
|
# Although this component does not use SPI, the BSEC library requires the SPI library
|
||||||
cg.add_library("SPI", None)
|
cg.add_library("SPI", None)
|
||||||
cg.add_library("Wire", None)
|
|
||||||
|
|
||||||
cg.add_define("USE_BSEC")
|
cg.add_define("USE_BSEC")
|
||||||
cg.add_library("boschsensortec/BSEC Software Library", "1.6.1480")
|
cg.add_library("boschsensortec/BSEC Software Library", "1.6.1480")
|
||||||
|
|||||||
@@ -181,6 +181,8 @@ void BME680BSECComponent::dump_config() {
|
|||||||
LOG_SENSOR(" ", "Breath VOC Equivalent", this->breath_voc_equivalent_sensor_);
|
LOG_SENSOR(" ", "Breath VOC Equivalent", this->breath_voc_equivalent_sensor_);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float BME680BSECComponent::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void BME680BSECComponent::loop() {
|
void BME680BSECComponent::loop() {
|
||||||
this->run_();
|
this->run_();
|
||||||
|
|
||||||
|
|||||||
@@ -64,6 +64,7 @@ class BME680BSECComponent : public Component, public i2c::I2CDevice {
|
|||||||
|
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void loop() override;
|
void loop() override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|||||||
@@ -106,6 +106,8 @@ void BME68xBSEC2Component::dump_config() {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float BME68xBSEC2Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void BME68xBSEC2Component::loop() {
|
void BME68xBSEC2Component::loop() {
|
||||||
this->run_();
|
this->run_();
|
||||||
|
|
||||||
|
|||||||
@@ -48,6 +48,7 @@ class BME68xBSEC2Component : public Component {
|
|||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void loop() override;
|
void loop() override;
|
||||||
|
|
||||||
void set_algorithm_output(AlgorithmOutput algorithm_output) { this->algorithm_output_ = algorithm_output; }
|
void set_algorithm_output(AlgorithmOutput algorithm_output) { this->algorithm_output_ = algorithm_output; }
|
||||||
|
|||||||
@@ -263,6 +263,7 @@ void BMI160Component::update() {
|
|||||||
|
|
||||||
this->status_clear_warning();
|
this->status_clear_warning();
|
||||||
}
|
}
|
||||||
|
float BMI160Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
} // namespace bmi160
|
} // namespace bmi160
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -14,6 +14,8 @@ class BMI160Component : public PollingComponent, public i2c::I2CDevice {
|
|||||||
|
|
||||||
void update() override;
|
void update() override;
|
||||||
|
|
||||||
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
void set_accel_x_sensor(sensor::Sensor *accel_x_sensor) { accel_x_sensor_ = accel_x_sensor; }
|
void set_accel_x_sensor(sensor::Sensor *accel_x_sensor) { accel_x_sensor_ = accel_x_sensor; }
|
||||||
void set_accel_y_sensor(sensor::Sensor *accel_y_sensor) { accel_y_sensor_ = accel_y_sensor; }
|
void set_accel_y_sensor(sensor::Sensor *accel_y_sensor) { accel_y_sensor_ = accel_y_sensor; }
|
||||||
void set_accel_z_sensor(sensor::Sensor *accel_z_sensor) { accel_z_sensor_ = accel_z_sensor; }
|
void set_accel_z_sensor(sensor::Sensor *accel_z_sensor) { accel_z_sensor_ = accel_z_sensor; }
|
||||||
|
|||||||
@@ -131,6 +131,7 @@ bool BMP085Component::set_mode_(uint8_t mode) {
|
|||||||
ESP_LOGV(TAG, "Setting mode to 0x%02X", mode);
|
ESP_LOGV(TAG, "Setting mode to 0x%02X", mode);
|
||||||
return this->write_byte(BMP085_REGISTER_CONTROL, mode);
|
return this->write_byte(BMP085_REGISTER_CONTROL, mode);
|
||||||
}
|
}
|
||||||
|
float BMP085Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
} // namespace bmp085
|
} // namespace bmp085
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -18,6 +18,8 @@ class BMP085Component : public PollingComponent, public i2c::I2CDevice {
|
|||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
|
||||||
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
struct CalibrationData {
|
struct CalibrationData {
|
||||||
int16_t ac1, ac2, ac3;
|
int16_t ac1, ac2, ac3;
|
||||||
|
|||||||
@@ -148,6 +148,7 @@ void BMP280Component::dump_config() {
|
|||||||
LOG_SENSOR(" ", "Pressure", this->pressure_sensor_);
|
LOG_SENSOR(" ", "Pressure", this->pressure_sensor_);
|
||||||
ESP_LOGCONFIG(TAG, " Oversampling: %s", oversampling_to_str(this->pressure_oversampling_));
|
ESP_LOGCONFIG(TAG, " Oversampling: %s", oversampling_to_str(this->pressure_oversampling_));
|
||||||
}
|
}
|
||||||
|
float BMP280Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
inline uint8_t oversampling_to_time(BMP280Oversampling over_sampling) { return (1 << uint8_t(over_sampling)) >> 1; }
|
inline uint8_t oversampling_to_time(BMP280Oversampling over_sampling) { return (1 << uint8_t(over_sampling)) >> 1; }
|
||||||
|
|
||||||
|
|||||||
@@ -64,6 +64,7 @@ class BMP280Component : public PollingComponent {
|
|||||||
|
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void update() override;
|
void update() override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|||||||
@@ -179,6 +179,7 @@ void BMP3XXComponent::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, " Oversampling: %s", LOG_STR_ARG(oversampling_to_str(this->pressure_oversampling_)));
|
ESP_LOGCONFIG(TAG, " Oversampling: %s", LOG_STR_ARG(oversampling_to_str(this->pressure_oversampling_)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
float BMP3XXComponent::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
inline uint8_t oversampling_to_time(Oversampling over_sampling) { return (1 << uint8_t(over_sampling)); }
|
inline uint8_t oversampling_to_time(Oversampling over_sampling) { return (1 << uint8_t(over_sampling)); }
|
||||||
|
|
||||||
|
|||||||
@@ -73,6 +73,7 @@ class BMP3XXComponent : public PollingComponent {
|
|||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void update() override;
|
void update() override;
|
||||||
|
|
||||||
void set_temperature_sensor(sensor::Sensor *temperature_sensor) { temperature_sensor_ = temperature_sensor; }
|
void set_temperature_sensor(sensor::Sensor *temperature_sensor) { temperature_sensor_ = temperature_sensor; }
|
||||||
|
|||||||
@@ -156,7 +156,7 @@ void CC1101Component::call_listeners_(const std::vector<uint8_t> &packet, float
|
|||||||
for (auto &listener : this->listeners_) {
|
for (auto &listener : this->listeners_) {
|
||||||
listener->on_packet(packet, freq_offset, rssi, lqi);
|
listener->on_packet(packet, freq_offset, rssi, lqi);
|
||||||
}
|
}
|
||||||
this->packet_trigger_.trigger(packet, freq_offset, rssi, lqi);
|
this->packet_trigger_->trigger(packet, freq_offset, rssi, lqi);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CC1101Component::loop() {
|
void CC1101Component::loop() {
|
||||||
|
|||||||
@@ -79,7 +79,7 @@ class CC1101Component : public Component,
|
|||||||
// Packet mode operations
|
// Packet mode operations
|
||||||
CC1101Error transmit_packet(const std::vector<uint8_t> &packet);
|
CC1101Error transmit_packet(const std::vector<uint8_t> &packet);
|
||||||
void register_listener(CC1101Listener *listener) { this->listeners_.push_back(listener); }
|
void register_listener(CC1101Listener *listener) { this->listeners_.push_back(listener); }
|
||||||
Trigger<std::vector<uint8_t>, float, float, uint8_t> *get_packet_trigger() { return &this->packet_trigger_; }
|
Trigger<std::vector<uint8_t>, float, float, uint8_t> *get_packet_trigger() const { return this->packet_trigger_; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
uint16_t chip_id_{0};
|
uint16_t chip_id_{0};
|
||||||
@@ -96,7 +96,8 @@ class CC1101Component : public Component,
|
|||||||
|
|
||||||
// Packet handling
|
// Packet handling
|
||||||
void call_listeners_(const std::vector<uint8_t> &packet, float freq_offset, float rssi, uint8_t lqi);
|
void call_listeners_(const std::vector<uint8_t> &packet, float freq_offset, float rssi, uint8_t lqi);
|
||||||
Trigger<std::vector<uint8_t>, float, float, uint8_t> packet_trigger_;
|
Trigger<std::vector<uint8_t>, float, float, uint8_t> *packet_trigger_{
|
||||||
|
new Trigger<std::vector<uint8_t>, float, float, uint8_t>()};
|
||||||
std::vector<uint8_t> packet_;
|
std::vector<uint8_t> packet_;
|
||||||
std::vector<CC1101Listener *> listeners_;
|
std::vector<CC1101Listener *> listeners_;
|
||||||
|
|
||||||
|
|||||||
@@ -7,6 +7,8 @@ namespace cd74hc4067 {
|
|||||||
|
|
||||||
static const char *const TAG = "cd74hc4067";
|
static const char *const TAG = "cd74hc4067";
|
||||||
|
|
||||||
|
float CD74HC4067Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void CD74HC4067Component::setup() {
|
void CD74HC4067Component::setup() {
|
||||||
this->pin_s0_->setup();
|
this->pin_s0_->setup();
|
||||||
this->pin_s1_->setup();
|
this->pin_s1_->setup();
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ class CD74HC4067Component : public Component {
|
|||||||
/// Set up the internal sensor array.
|
/// Set up the internal sensor array.
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
/// setting pin active by setting the right combination of the four multiplexer input pins
|
/// setting pin active by setting the right combination of the four multiplexer input pins
|
||||||
void activate_pin(uint8_t pin);
|
void activate_pin(uint8_t pin);
|
||||||
|
|||||||
@@ -1,44 +1,109 @@
|
|||||||
#include "climate_mode.h"
|
#include "climate_mode.h"
|
||||||
#include "esphome/core/progmem.h"
|
|
||||||
|
|
||||||
namespace esphome::climate {
|
namespace esphome::climate {
|
||||||
|
|
||||||
// Climate mode strings indexed by ClimateMode enum (0-6): OFF, HEAT_COOL, COOL, HEAT, FAN_ONLY, DRY, AUTO
|
|
||||||
PROGMEM_STRING_TABLE(ClimateModeStrings, "OFF", "HEAT_COOL", "COOL", "HEAT", "FAN_ONLY", "DRY", "AUTO", "UNKNOWN");
|
|
||||||
|
|
||||||
const LogString *climate_mode_to_string(ClimateMode mode) {
|
const LogString *climate_mode_to_string(ClimateMode mode) {
|
||||||
return ClimateModeStrings::get_log_str(static_cast<uint8_t>(mode), ClimateModeStrings::LAST_INDEX);
|
switch (mode) {
|
||||||
|
case CLIMATE_MODE_OFF:
|
||||||
|
return LOG_STR("OFF");
|
||||||
|
case CLIMATE_MODE_HEAT_COOL:
|
||||||
|
return LOG_STR("HEAT_COOL");
|
||||||
|
case CLIMATE_MODE_AUTO:
|
||||||
|
return LOG_STR("AUTO");
|
||||||
|
case CLIMATE_MODE_COOL:
|
||||||
|
return LOG_STR("COOL");
|
||||||
|
case CLIMATE_MODE_HEAT:
|
||||||
|
return LOG_STR("HEAT");
|
||||||
|
case CLIMATE_MODE_FAN_ONLY:
|
||||||
|
return LOG_STR("FAN_ONLY");
|
||||||
|
case CLIMATE_MODE_DRY:
|
||||||
|
return LOG_STR("DRY");
|
||||||
|
default:
|
||||||
|
return LOG_STR("UNKNOWN");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Climate action strings indexed by ClimateAction enum (0,2-6): OFF, (gap), COOLING, HEATING, IDLE, DRYING, FAN
|
|
||||||
PROGMEM_STRING_TABLE(ClimateActionStrings, "OFF", "UNKNOWN", "COOLING", "HEATING", "IDLE", "DRYING", "FAN", "UNKNOWN");
|
|
||||||
|
|
||||||
const LogString *climate_action_to_string(ClimateAction action) {
|
const LogString *climate_action_to_string(ClimateAction action) {
|
||||||
return ClimateActionStrings::get_log_str(static_cast<uint8_t>(action), ClimateActionStrings::LAST_INDEX);
|
switch (action) {
|
||||||
|
case CLIMATE_ACTION_OFF:
|
||||||
|
return LOG_STR("OFF");
|
||||||
|
case CLIMATE_ACTION_COOLING:
|
||||||
|
return LOG_STR("COOLING");
|
||||||
|
case CLIMATE_ACTION_HEATING:
|
||||||
|
return LOG_STR("HEATING");
|
||||||
|
case CLIMATE_ACTION_IDLE:
|
||||||
|
return LOG_STR("IDLE");
|
||||||
|
case CLIMATE_ACTION_DRYING:
|
||||||
|
return LOG_STR("DRYING");
|
||||||
|
case CLIMATE_ACTION_FAN:
|
||||||
|
return LOG_STR("FAN");
|
||||||
|
default:
|
||||||
|
return LOG_STR("UNKNOWN");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Climate fan mode strings indexed by ClimateFanMode enum (0-9): ON, OFF, AUTO, LOW, MEDIUM, HIGH, MIDDLE, FOCUS,
|
|
||||||
// DIFFUSE, QUIET
|
|
||||||
PROGMEM_STRING_TABLE(ClimateFanModeStrings, "ON", "OFF", "AUTO", "LOW", "MEDIUM", "HIGH", "MIDDLE", "FOCUS", "DIFFUSE",
|
|
||||||
"QUIET", "UNKNOWN");
|
|
||||||
|
|
||||||
const LogString *climate_fan_mode_to_string(ClimateFanMode fan_mode) {
|
const LogString *climate_fan_mode_to_string(ClimateFanMode fan_mode) {
|
||||||
return ClimateFanModeStrings::get_log_str(static_cast<uint8_t>(fan_mode), ClimateFanModeStrings::LAST_INDEX);
|
switch (fan_mode) {
|
||||||
|
case climate::CLIMATE_FAN_ON:
|
||||||
|
return LOG_STR("ON");
|
||||||
|
case climate::CLIMATE_FAN_OFF:
|
||||||
|
return LOG_STR("OFF");
|
||||||
|
case climate::CLIMATE_FAN_AUTO:
|
||||||
|
return LOG_STR("AUTO");
|
||||||
|
case climate::CLIMATE_FAN_LOW:
|
||||||
|
return LOG_STR("LOW");
|
||||||
|
case climate::CLIMATE_FAN_MEDIUM:
|
||||||
|
return LOG_STR("MEDIUM");
|
||||||
|
case climate::CLIMATE_FAN_HIGH:
|
||||||
|
return LOG_STR("HIGH");
|
||||||
|
case climate::CLIMATE_FAN_MIDDLE:
|
||||||
|
return LOG_STR("MIDDLE");
|
||||||
|
case climate::CLIMATE_FAN_FOCUS:
|
||||||
|
return LOG_STR("FOCUS");
|
||||||
|
case climate::CLIMATE_FAN_DIFFUSE:
|
||||||
|
return LOG_STR("DIFFUSE");
|
||||||
|
case climate::CLIMATE_FAN_QUIET:
|
||||||
|
return LOG_STR("QUIET");
|
||||||
|
default:
|
||||||
|
return LOG_STR("UNKNOWN");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Climate swing mode strings indexed by ClimateSwingMode enum (0-3): OFF, BOTH, VERTICAL, HORIZONTAL
|
|
||||||
PROGMEM_STRING_TABLE(ClimateSwingModeStrings, "OFF", "BOTH", "VERTICAL", "HORIZONTAL", "UNKNOWN");
|
|
||||||
|
|
||||||
const LogString *climate_swing_mode_to_string(ClimateSwingMode swing_mode) {
|
const LogString *climate_swing_mode_to_string(ClimateSwingMode swing_mode) {
|
||||||
return ClimateSwingModeStrings::get_log_str(static_cast<uint8_t>(swing_mode), ClimateSwingModeStrings::LAST_INDEX);
|
switch (swing_mode) {
|
||||||
|
case climate::CLIMATE_SWING_OFF:
|
||||||
|
return LOG_STR("OFF");
|
||||||
|
case climate::CLIMATE_SWING_BOTH:
|
||||||
|
return LOG_STR("BOTH");
|
||||||
|
case climate::CLIMATE_SWING_VERTICAL:
|
||||||
|
return LOG_STR("VERTICAL");
|
||||||
|
case climate::CLIMATE_SWING_HORIZONTAL:
|
||||||
|
return LOG_STR("HORIZONTAL");
|
||||||
|
default:
|
||||||
|
return LOG_STR("UNKNOWN");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Climate preset strings indexed by ClimatePreset enum (0-7): NONE, HOME, AWAY, BOOST, COMFORT, ECO, SLEEP, ACTIVITY
|
|
||||||
PROGMEM_STRING_TABLE(ClimatePresetStrings, "NONE", "HOME", "AWAY", "BOOST", "COMFORT", "ECO", "SLEEP", "ACTIVITY",
|
|
||||||
"UNKNOWN");
|
|
||||||
|
|
||||||
const LogString *climate_preset_to_string(ClimatePreset preset) {
|
const LogString *climate_preset_to_string(ClimatePreset preset) {
|
||||||
return ClimatePresetStrings::get_log_str(static_cast<uint8_t>(preset), ClimatePresetStrings::LAST_INDEX);
|
switch (preset) {
|
||||||
|
case climate::CLIMATE_PRESET_NONE:
|
||||||
|
return LOG_STR("NONE");
|
||||||
|
case climate::CLIMATE_PRESET_HOME:
|
||||||
|
return LOG_STR("HOME");
|
||||||
|
case climate::CLIMATE_PRESET_ECO:
|
||||||
|
return LOG_STR("ECO");
|
||||||
|
case climate::CLIMATE_PRESET_AWAY:
|
||||||
|
return LOG_STR("AWAY");
|
||||||
|
case climate::CLIMATE_PRESET_BOOST:
|
||||||
|
return LOG_STR("BOOST");
|
||||||
|
case climate::CLIMATE_PRESET_COMFORT:
|
||||||
|
return LOG_STR("COMFORT");
|
||||||
|
case climate::CLIMATE_PRESET_SLEEP:
|
||||||
|
return LOG_STR("SLEEP");
|
||||||
|
case climate::CLIMATE_PRESET_ACTIVITY:
|
||||||
|
return LOG_STR("ACTIVITY");
|
||||||
|
default:
|
||||||
|
return LOG_STR("UNKNOWN");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace esphome::climate
|
} // namespace esphome::climate
|
||||||
|
|||||||
@@ -10,6 +10,8 @@ namespace cm1106 {
|
|||||||
|
|
||||||
class CM1106Component : public PollingComponent, public uart::UARTDevice {
|
class CM1106Component : public PollingComponent, public uart::UARTDevice {
|
||||||
public:
|
public:
|
||||||
|
float get_setup_priority() const override { return esphome::setup_priority::DATA; }
|
||||||
|
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void update() override;
|
void update() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
|||||||
@@ -10,6 +10,8 @@ namespace combination {
|
|||||||
|
|
||||||
class CombinationComponent : public Component, public sensor::Sensor {
|
class CombinationComponent : public Component, public sensor::Sensor {
|
||||||
public:
|
public:
|
||||||
|
float get_setup_priority() const override { return esphome::setup_priority::DATA; }
|
||||||
|
|
||||||
/// @brief Logs all source sensor's names
|
/// @brief Logs all source sensor's names
|
||||||
virtual void log_source_sensors() = 0;
|
virtual void log_source_sensors() = 0;
|
||||||
|
|
||||||
|
|||||||
@@ -17,9 +17,3 @@ CONF_ON_STATE_CHANGE = "on_state_change"
|
|||||||
CONF_REQUEST_HEADERS = "request_headers"
|
CONF_REQUEST_HEADERS = "request_headers"
|
||||||
CONF_ROWS = "rows"
|
CONF_ROWS = "rows"
|
||||||
CONF_USE_PSRAM = "use_psram"
|
CONF_USE_PSRAM = "use_psram"
|
||||||
|
|
||||||
ICON_CURRENT_DC = "mdi:current-dc"
|
|
||||||
ICON_SOLAR_PANEL = "mdi:solar-panel"
|
|
||||||
ICON_SOLAR_POWER = "mdi:solar-power"
|
|
||||||
|
|
||||||
UNIT_AMPERE_HOUR = "Ah"
|
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
import logging
|
|
||||||
|
|
||||||
from esphome import automation
|
from esphome import automation
|
||||||
from esphome.automation import Condition, maybe_simple_id
|
from esphome.automation import Condition, maybe_simple_id
|
||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
@@ -11,7 +9,6 @@ from esphome.const import (
|
|||||||
CONF_ICON,
|
CONF_ICON,
|
||||||
CONF_ID,
|
CONF_ID,
|
||||||
CONF_MQTT_ID,
|
CONF_MQTT_ID,
|
||||||
CONF_ON_IDLE,
|
|
||||||
CONF_ON_OPEN,
|
CONF_ON_OPEN,
|
||||||
CONF_POSITION,
|
CONF_POSITION,
|
||||||
CONF_POSITION_COMMAND_TOPIC,
|
CONF_POSITION_COMMAND_TOPIC,
|
||||||
@@ -35,10 +32,9 @@ from esphome.const import (
|
|||||||
DEVICE_CLASS_SHUTTER,
|
DEVICE_CLASS_SHUTTER,
|
||||||
DEVICE_CLASS_WINDOW,
|
DEVICE_CLASS_WINDOW,
|
||||||
)
|
)
|
||||||
from esphome.core import CORE, ID, CoroPriority, coroutine_with_priority
|
from esphome.core import CORE, CoroPriority, coroutine_with_priority
|
||||||
from esphome.core.entity_helpers import entity_duplicate_validator, setup_entity
|
from esphome.core.entity_helpers import entity_duplicate_validator, setup_entity
|
||||||
from esphome.cpp_generator import MockObj, MockObjClass
|
from esphome.cpp_generator import MockObjClass
|
||||||
from esphome.types import ConfigType, TemplateArgsType
|
|
||||||
|
|
||||||
IS_PLATFORM_COMPONENT = True
|
IS_PLATFORM_COMPONENT = True
|
||||||
|
|
||||||
@@ -57,8 +53,6 @@ DEVICE_CLASSES = [
|
|||||||
DEVICE_CLASS_WINDOW,
|
DEVICE_CLASS_WINDOW,
|
||||||
]
|
]
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
cover_ns = cg.esphome_ns.namespace("cover")
|
cover_ns = cg.esphome_ns.namespace("cover")
|
||||||
|
|
||||||
Cover = cover_ns.class_("Cover", cg.EntityBase)
|
Cover = cover_ns.class_("Cover", cg.EntityBase)
|
||||||
@@ -89,29 +83,14 @@ ControlAction = cover_ns.class_("ControlAction", automation.Action)
|
|||||||
CoverPublishAction = cover_ns.class_("CoverPublishAction", automation.Action)
|
CoverPublishAction = cover_ns.class_("CoverPublishAction", automation.Action)
|
||||||
CoverIsOpenCondition = cover_ns.class_("CoverIsOpenCondition", Condition)
|
CoverIsOpenCondition = cover_ns.class_("CoverIsOpenCondition", Condition)
|
||||||
CoverIsClosedCondition = cover_ns.class_("CoverIsClosedCondition", Condition)
|
CoverIsClosedCondition = cover_ns.class_("CoverIsClosedCondition", Condition)
|
||||||
CoverOpenedTrigger = cover_ns.class_(
|
|
||||||
"CoverOpenedTrigger", automation.Trigger.template()
|
# Triggers
|
||||||
)
|
CoverOpenTrigger = cover_ns.class_("CoverOpenTrigger", automation.Trigger.template())
|
||||||
CoverClosedTrigger = cover_ns.class_(
|
CoverClosedTrigger = cover_ns.class_(
|
||||||
"CoverClosedTrigger", automation.Trigger.template()
|
"CoverClosedTrigger", automation.Trigger.template()
|
||||||
)
|
)
|
||||||
CoverTrigger = cover_ns.class_("CoverTrigger", automation.Trigger.template())
|
|
||||||
|
|
||||||
# Cover-specific constants
|
|
||||||
CONF_ON_CLOSED = "on_closed"
|
CONF_ON_CLOSED = "on_closed"
|
||||||
CONF_ON_OPENED = "on_opened"
|
|
||||||
CONF_ON_OPENING = "on_opening"
|
|
||||||
CONF_ON_CLOSING = "on_closing"
|
|
||||||
|
|
||||||
TRIGGERS = {
|
|
||||||
CONF_ON_OPEN: CoverOpenedTrigger, # Deprecated, use on_opened
|
|
||||||
CONF_ON_OPENED: CoverOpenedTrigger,
|
|
||||||
CONF_ON_CLOSED: CoverClosedTrigger,
|
|
||||||
CONF_ON_CLOSING: CoverTrigger.template(CoverOperation.COVER_OPERATION_CLOSING),
|
|
||||||
CONF_ON_OPENING: CoverTrigger.template(CoverOperation.COVER_OPERATION_OPENING),
|
|
||||||
CONF_ON_IDLE: CoverTrigger.template(CoverOperation.COVER_OPERATION_IDLE),
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
_COVER_SCHEMA = (
|
_COVER_SCHEMA = (
|
||||||
cv.ENTITY_BASE_SCHEMA.extend(web_server.WEBSERVER_SORTING_SCHEMA)
|
cv.ENTITY_BASE_SCHEMA.extend(web_server.WEBSERVER_SORTING_SCHEMA)
|
||||||
@@ -132,14 +111,16 @@ _COVER_SCHEMA = (
|
|||||||
cv.Optional(CONF_TILT_STATE_TOPIC): cv.All(
|
cv.Optional(CONF_TILT_STATE_TOPIC): cv.All(
|
||||||
cv.requires_component("mqtt"), cv.subscribe_topic
|
cv.requires_component("mqtt"), cv.subscribe_topic
|
||||||
),
|
),
|
||||||
**{
|
cv.Optional(CONF_ON_OPEN): automation.validate_automation(
|
||||||
cv.Optional(conf): automation.validate_automation(
|
|
||||||
{
|
{
|
||||||
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(trigger_class),
|
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(CoverOpenTrigger),
|
||||||
}
|
}
|
||||||
)
|
),
|
||||||
for conf, trigger_class in TRIGGERS.items()
|
cv.Optional(CONF_ON_CLOSED): automation.validate_automation(
|
||||||
},
|
{
|
||||||
|
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(CoverClosedTrigger),
|
||||||
|
}
|
||||||
|
),
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
@@ -176,12 +157,10 @@ async def setup_cover_core_(var, config):
|
|||||||
if (device_class := config.get(CONF_DEVICE_CLASS)) is not None:
|
if (device_class := config.get(CONF_DEVICE_CLASS)) is not None:
|
||||||
cg.add(var.set_device_class(device_class))
|
cg.add(var.set_device_class(device_class))
|
||||||
|
|
||||||
if CONF_ON_OPEN in config:
|
for conf in config.get(CONF_ON_OPEN, []):
|
||||||
_LOGGER.warning(
|
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
||||||
"'on_open' is deprecated, use 'on_opened'. Will be removed in 2026.8.0"
|
await automation.build_automation(trigger, [], conf)
|
||||||
)
|
for conf in config.get(CONF_ON_CLOSED, []):
|
||||||
for trigger_conf in TRIGGERS:
|
|
||||||
for conf in config.get(trigger_conf, []):
|
|
||||||
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
||||||
await automation.build_automation(trigger, [], conf)
|
await automation.build_automation(trigger, [], conf)
|
||||||
|
|
||||||
@@ -279,26 +258,6 @@ async def cover_control_to_code(config, action_id, template_arg, args):
|
|||||||
return var
|
return var
|
||||||
|
|
||||||
|
|
||||||
COVER_CONDITION_SCHEMA = cv.maybe_simple_value(
|
|
||||||
{cv.Required(CONF_ID): cv.use_id(Cover)}, key=CONF_ID
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
async def cover_condition_to_code(
|
|
||||||
config: ConfigType, condition_id: ID, template_arg: MockObj, args: TemplateArgsType
|
|
||||||
) -> MockObj:
|
|
||||||
paren = await cg.get_variable(config[CONF_ID])
|
|
||||||
return cg.new_Pvariable(condition_id, template_arg, paren)
|
|
||||||
|
|
||||||
|
|
||||||
automation.register_condition(
|
|
||||||
"cover.is_open", CoverIsOpenCondition, COVER_CONDITION_SCHEMA
|
|
||||||
)(cover_condition_to_code)
|
|
||||||
automation.register_condition(
|
|
||||||
"cover.is_closed", CoverIsClosedCondition, COVER_CONDITION_SCHEMA
|
|
||||||
)(cover_condition_to_code)
|
|
||||||
|
|
||||||
|
|
||||||
@coroutine_with_priority(CoroPriority.CORE)
|
@coroutine_with_priority(CoroPriority.CORE)
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
cg.add_global(cover_ns.using)
|
cg.add_global(cover_ns.using)
|
||||||
|
|||||||
@@ -90,53 +90,44 @@ template<typename... Ts> class CoverPublishAction : public Action<Ts...> {
|
|||||||
Cover *cover_;
|
Cover *cover_;
|
||||||
};
|
};
|
||||||
|
|
||||||
template<bool OPEN, typename... Ts> class CoverPositionCondition : public Condition<Ts...> {
|
template<typename... Ts> class CoverIsOpenCondition : public Condition<Ts...> {
|
||||||
public:
|
public:
|
||||||
CoverPositionCondition(Cover *cover) : cover_(cover) {}
|
CoverIsOpenCondition(Cover *cover) : cover_(cover) {}
|
||||||
|
bool check(const Ts &...x) override { return this->cover_->is_fully_open(); }
|
||||||
bool check(const Ts &...x) override { return this->cover_->position == (OPEN ? COVER_OPEN : COVER_CLOSED); }
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
Cover *cover_;
|
Cover *cover_;
|
||||||
};
|
};
|
||||||
|
|
||||||
template<typename... Ts> using CoverIsOpenCondition = CoverPositionCondition<true, Ts...>;
|
template<typename... Ts> class CoverIsClosedCondition : public Condition<Ts...> {
|
||||||
template<typename... Ts> using CoverIsClosedCondition = CoverPositionCondition<false, Ts...>;
|
|
||||||
|
|
||||||
template<bool OPEN> class CoverPositionTrigger : public Trigger<> {
|
|
||||||
public:
|
public:
|
||||||
CoverPositionTrigger(Cover *a_cover) {
|
CoverIsClosedCondition(Cover *cover) : cover_(cover) {}
|
||||||
|
bool check(const Ts &...x) override { return this->cover_->is_fully_closed(); }
|
||||||
|
|
||||||
|
protected:
|
||||||
|
Cover *cover_;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CoverOpenTrigger : public Trigger<> {
|
||||||
|
public:
|
||||||
|
CoverOpenTrigger(Cover *a_cover) {
|
||||||
a_cover->add_on_state_callback([this, a_cover]() {
|
a_cover->add_on_state_callback([this, a_cover]() {
|
||||||
if (a_cover->position != this->last_position_) {
|
if (a_cover->is_fully_open()) {
|
||||||
this->last_position_ = a_cover->position;
|
|
||||||
if (a_cover->position == (OPEN ? COVER_OPEN : COVER_CLOSED))
|
|
||||||
this->trigger();
|
this->trigger();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
protected:
|
|
||||||
float last_position_{NAN};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
using CoverOpenedTrigger = CoverPositionTrigger<true>;
|
class CoverClosedTrigger : public Trigger<> {
|
||||||
using CoverClosedTrigger = CoverPositionTrigger<false>;
|
|
||||||
|
|
||||||
template<CoverOperation OP> class CoverTrigger : public Trigger<> {
|
|
||||||
public:
|
public:
|
||||||
CoverTrigger(Cover *a_cover) {
|
CoverClosedTrigger(Cover *a_cover) {
|
||||||
a_cover->add_on_state_callback([this, a_cover]() {
|
a_cover->add_on_state_callback([this, a_cover]() {
|
||||||
auto current_op = a_cover->current_operation;
|
if (a_cover->is_fully_closed()) {
|
||||||
if (current_op == OP) {
|
|
||||||
if (!this->last_operation_.has_value() || this->last_operation_.value() != OP) {
|
|
||||||
this->trigger();
|
this->trigger();
|
||||||
}
|
}
|
||||||
}
|
|
||||||
this->last_operation_ = current_op;
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
protected:
|
|
||||||
optional<CoverOperation> last_operation_{};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace esphome::cover
|
} // namespace esphome::cover
|
||||||
|
|||||||
@@ -10,6 +10,9 @@ namespace esphome::cover {
|
|||||||
|
|
||||||
static const char *const TAG = "cover";
|
static const char *const TAG = "cover";
|
||||||
|
|
||||||
|
const float COVER_OPEN = 1.0f;
|
||||||
|
const float COVER_CLOSED = 0.0f;
|
||||||
|
|
||||||
const LogString *cover_command_to_str(float pos) {
|
const LogString *cover_command_to_str(float pos) {
|
||||||
if (pos == COVER_OPEN) {
|
if (pos == COVER_OPEN) {
|
||||||
return LOG_STR("OPEN");
|
return LOG_STR("OPEN");
|
||||||
@@ -19,11 +22,17 @@ const LogString *cover_command_to_str(float pos) {
|
|||||||
return LOG_STR("UNKNOWN");
|
return LOG_STR("UNKNOWN");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Cover operation strings indexed by CoverOperation enum (0-2): IDLE, OPENING, CLOSING, plus UNKNOWN
|
|
||||||
PROGMEM_STRING_TABLE(CoverOperationStrings, "IDLE", "OPENING", "CLOSING", "UNKNOWN");
|
|
||||||
|
|
||||||
const LogString *cover_operation_to_str(CoverOperation op) {
|
const LogString *cover_operation_to_str(CoverOperation op) {
|
||||||
return CoverOperationStrings::get_log_str(static_cast<uint8_t>(op), CoverOperationStrings::LAST_INDEX);
|
switch (op) {
|
||||||
|
case COVER_OPERATION_IDLE:
|
||||||
|
return LOG_STR("IDLE");
|
||||||
|
case COVER_OPERATION_OPENING:
|
||||||
|
return LOG_STR("OPENING");
|
||||||
|
case COVER_OPERATION_CLOSING:
|
||||||
|
return LOG_STR("CLOSING");
|
||||||
|
default:
|
||||||
|
return LOG_STR("UNKNOWN");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Cover::Cover() : position{COVER_OPEN} {}
|
Cover::Cover() : position{COVER_OPEN} {}
|
||||||
|
|||||||
@@ -10,8 +10,8 @@
|
|||||||
|
|
||||||
namespace esphome::cover {
|
namespace esphome::cover {
|
||||||
|
|
||||||
static constexpr float COVER_OPEN = 1.0f;
|
const extern float COVER_OPEN;
|
||||||
static constexpr float COVER_CLOSED = 0.0f;
|
const extern float COVER_CLOSED;
|
||||||
|
|
||||||
#define LOG_COVER(prefix, type, obj) \
|
#define LOG_COVER(prefix, type, obj) \
|
||||||
if ((obj) != nullptr) { \
|
if ((obj) != nullptr) { \
|
||||||
|
|||||||
@@ -62,6 +62,8 @@ void CSE7761Component::dump_config() {
|
|||||||
this->check_uart_settings(38400, 1, uart::UART_CONFIG_PARITY_EVEN, 8);
|
this->check_uart_settings(38400, 1, uart::UART_CONFIG_PARITY_EVEN, 8);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float CSE7761Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void CSE7761Component::update() {
|
void CSE7761Component::update() {
|
||||||
if (this->data_.ready) {
|
if (this->data_.ready) {
|
||||||
this->get_data_();
|
this->get_data_();
|
||||||
|
|||||||
@@ -28,6 +28,7 @@ class CSE7761Component : public PollingComponent, public uart::UARTDevice {
|
|||||||
void set_current_2_sensor(sensor::Sensor *current_sensor_2) { current_sensor_2_ = current_sensor_2; }
|
void set_current_2_sensor(sensor::Sensor *current_sensor_2) { current_sensor_2_ = current_sensor_2; }
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void update() override;
|
void update() override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ namespace esphome {
|
|||||||
namespace cse7766 {
|
namespace cse7766 {
|
||||||
|
|
||||||
static const char *const TAG = "cse7766";
|
static const char *const TAG = "cse7766";
|
||||||
|
static constexpr size_t CSE7766_RAW_DATA_SIZE = 24;
|
||||||
|
|
||||||
void CSE7766Component::loop() {
|
void CSE7766Component::loop() {
|
||||||
const uint32_t now = App.get_loop_component_start_time();
|
const uint32_t now = App.get_loop_component_start_time();
|
||||||
@@ -15,41 +16,28 @@ void CSE7766Component::loop() {
|
|||||||
this->raw_data_index_ = 0;
|
this->raw_data_index_ = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Early return prevents updating last_transmission_ when no data is available.
|
if (this->available() == 0) {
|
||||||
size_t avail = this->available();
|
|
||||||
if (avail == 0) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this->last_transmission_ = now;
|
this->last_transmission_ = now;
|
||||||
|
while (this->available() != 0) {
|
||||||
// Read all available bytes in batches to reduce UART call overhead.
|
this->read_byte(&this->raw_data_[this->raw_data_index_]);
|
||||||
// At 4800 baud (~480 bytes/sec) with ~122 Hz loop rate, typically ~4 bytes per call.
|
|
||||||
uint8_t buf[CSE7766_RAW_DATA_SIZE];
|
|
||||||
while (avail > 0) {
|
|
||||||
size_t to_read = std::min(avail, sizeof(buf));
|
|
||||||
if (!this->read_array(buf, to_read)) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
avail -= to_read;
|
|
||||||
|
|
||||||
for (size_t i = 0; i < to_read; i++) {
|
|
||||||
this->raw_data_[this->raw_data_index_] = buf[i];
|
|
||||||
if (!this->check_byte_()) {
|
if (!this->check_byte_()) {
|
||||||
this->raw_data_index_ = 0;
|
this->raw_data_index_ = 0;
|
||||||
this->status_set_warning();
|
this->status_set_warning();
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->raw_data_index_ == CSE7766_RAW_DATA_SIZE - 1) {
|
if (this->raw_data_index_ == 23) {
|
||||||
this->parse_data_();
|
this->parse_data_();
|
||||||
this->status_clear_warning();
|
this->status_clear_warning();
|
||||||
}
|
}
|
||||||
|
|
||||||
this->raw_data_index_ = (this->raw_data_index_ + 1) % CSE7766_RAW_DATA_SIZE;
|
this->raw_data_index_ = (this->raw_data_index_ + 1) % 24;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
float CSE7766Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
bool CSE7766Component::check_byte_() {
|
bool CSE7766Component::check_byte_() {
|
||||||
uint8_t index = this->raw_data_index_;
|
uint8_t index = this->raw_data_index_;
|
||||||
@@ -66,15 +54,14 @@ bool CSE7766Component::check_byte_() {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (index == CSE7766_RAW_DATA_SIZE - 1) {
|
if (index == 23) {
|
||||||
uint8_t checksum = 0;
|
uint8_t checksum = 0;
|
||||||
for (uint8_t i = 2; i < CSE7766_RAW_DATA_SIZE - 1; i++) {
|
for (uint8_t i = 2; i < 23; i++) {
|
||||||
checksum += this->raw_data_[i];
|
checksum += this->raw_data_[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (checksum != this->raw_data_[CSE7766_RAW_DATA_SIZE - 1]) {
|
if (checksum != this->raw_data_[23]) {
|
||||||
ESP_LOGW(TAG, "Invalid checksum from CSE7766: 0x%02X != 0x%02X", checksum,
|
ESP_LOGW(TAG, "Invalid checksum from CSE7766: 0x%02X != 0x%02X", checksum, this->raw_data_[23]);
|
||||||
this->raw_data_[CSE7766_RAW_DATA_SIZE - 1]);
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
@@ -165,10 +152,6 @@ void CSE7766Component::parse_data_() {
|
|||||||
if (this->power_sensor_ != nullptr) {
|
if (this->power_sensor_ != nullptr) {
|
||||||
this->power_sensor_->publish_state(power);
|
this->power_sensor_->publish_state(power);
|
||||||
}
|
}
|
||||||
} else if (this->power_sensor_ != nullptr) {
|
|
||||||
// No valid power measurement from chip - publish 0W to avoid stale readings
|
|
||||||
// This typically happens when current is below the measurable threshold (~50mA)
|
|
||||||
this->power_sensor_->publish_state(0.0f);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
float current = 0.0f;
|
float current = 0.0f;
|
||||||
|
|||||||
@@ -8,8 +8,6 @@
|
|||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace cse7766 {
|
namespace cse7766 {
|
||||||
|
|
||||||
static constexpr size_t CSE7766_RAW_DATA_SIZE = 24;
|
|
||||||
|
|
||||||
class CSE7766Component : public Component, public uart::UARTDevice {
|
class CSE7766Component : public Component, public uart::UARTDevice {
|
||||||
public:
|
public:
|
||||||
void set_voltage_sensor(sensor::Sensor *voltage_sensor) { voltage_sensor_ = voltage_sensor; }
|
void set_voltage_sensor(sensor::Sensor *voltage_sensor) { voltage_sensor_ = voltage_sensor; }
|
||||||
@@ -25,6 +23,7 @@ class CSE7766Component : public Component, public uart::UARTDevice {
|
|||||||
void set_power_factor_sensor(sensor::Sensor *power_factor_sensor) { power_factor_sensor_ = power_factor_sensor; }
|
void set_power_factor_sensor(sensor::Sensor *power_factor_sensor) { power_factor_sensor_ = power_factor_sensor; }
|
||||||
|
|
||||||
void loop() override;
|
void loop() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@@ -35,7 +34,7 @@ class CSE7766Component : public Component, public uart::UARTDevice {
|
|||||||
this->raw_data_[start_index + 2]);
|
this->raw_data_[start_index + 2]);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t raw_data_[CSE7766_RAW_DATA_SIZE];
|
uint8_t raw_data_[24];
|
||||||
uint8_t raw_data_index_{0};
|
uint8_t raw_data_index_{0};
|
||||||
uint32_t last_transmission_{0};
|
uint32_t last_transmission_{0};
|
||||||
sensor::Sensor *voltage_sensor_{nullptr};
|
sensor::Sensor *voltage_sensor_{nullptr};
|
||||||
|
|||||||
@@ -66,7 +66,7 @@ void CurrentBasedCover::loop() {
|
|||||||
if (this->current_operation == COVER_OPERATION_OPENING) {
|
if (this->current_operation == COVER_OPERATION_OPENING) {
|
||||||
if (this->malfunction_detection_ && this->is_closing_()) { // Malfunction
|
if (this->malfunction_detection_ && this->is_closing_()) { // Malfunction
|
||||||
this->direction_idle_();
|
this->direction_idle_();
|
||||||
this->malfunction_trigger_.trigger();
|
this->malfunction_trigger_->trigger();
|
||||||
ESP_LOGI(TAG, "'%s' - Malfunction detected during opening. Current flow detected in close circuit",
|
ESP_LOGI(TAG, "'%s' - Malfunction detected during opening. Current flow detected in close circuit",
|
||||||
this->name_.c_str());
|
this->name_.c_str());
|
||||||
} else if (this->is_opening_blocked_()) { // Blocked
|
} else if (this->is_opening_blocked_()) { // Blocked
|
||||||
@@ -87,7 +87,7 @@ void CurrentBasedCover::loop() {
|
|||||||
} else if (this->current_operation == COVER_OPERATION_CLOSING) {
|
} else if (this->current_operation == COVER_OPERATION_CLOSING) {
|
||||||
if (this->malfunction_detection_ && this->is_opening_()) { // Malfunction
|
if (this->malfunction_detection_ && this->is_opening_()) { // Malfunction
|
||||||
this->direction_idle_();
|
this->direction_idle_();
|
||||||
this->malfunction_trigger_.trigger();
|
this->malfunction_trigger_->trigger();
|
||||||
ESP_LOGI(TAG, "'%s' - Malfunction detected during closing. Current flow detected in open circuit",
|
ESP_LOGI(TAG, "'%s' - Malfunction detected during closing. Current flow detected in open circuit",
|
||||||
this->name_.c_str());
|
this->name_.c_str());
|
||||||
} else if (this->is_closing_blocked_()) { // Blocked
|
} else if (this->is_closing_blocked_()) { // Blocked
|
||||||
@@ -159,6 +159,7 @@ void CurrentBasedCover::dump_config() {
|
|||||||
this->start_sensing_delay_ / 1e3f, YESNO(this->malfunction_detection_));
|
this->start_sensing_delay_ / 1e3f, YESNO(this->malfunction_detection_));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float CurrentBasedCover::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
void CurrentBasedCover::stop_prev_trigger_() {
|
void CurrentBasedCover::stop_prev_trigger_() {
|
||||||
if (this->prev_command_trigger_ != nullptr) {
|
if (this->prev_command_trigger_ != nullptr) {
|
||||||
this->prev_command_trigger_->stop_action();
|
this->prev_command_trigger_->stop_action();
|
||||||
@@ -220,15 +221,15 @@ void CurrentBasedCover::start_direction_(CoverOperation dir) {
|
|||||||
Trigger<> *trig;
|
Trigger<> *trig;
|
||||||
switch (dir) {
|
switch (dir) {
|
||||||
case COVER_OPERATION_IDLE:
|
case COVER_OPERATION_IDLE:
|
||||||
trig = &this->stop_trigger_;
|
trig = this->stop_trigger_;
|
||||||
break;
|
break;
|
||||||
case COVER_OPERATION_OPENING:
|
case COVER_OPERATION_OPENING:
|
||||||
this->last_operation_ = dir;
|
this->last_operation_ = dir;
|
||||||
trig = &this->open_trigger_;
|
trig = this->open_trigger_;
|
||||||
break;
|
break;
|
||||||
case COVER_OPERATION_CLOSING:
|
case COVER_OPERATION_CLOSING:
|
||||||
this->last_operation_ = dir;
|
this->last_operation_ = dir;
|
||||||
trig = &this->close_trigger_;
|
trig = this->close_trigger_;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -14,10 +14,11 @@ class CurrentBasedCover : public cover::Cover, public Component {
|
|||||||
void setup() override;
|
void setup() override;
|
||||||
void loop() override;
|
void loop() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
Trigger<> *get_stop_trigger() { return &this->stop_trigger_; }
|
Trigger<> *get_stop_trigger() const { return this->stop_trigger_; }
|
||||||
|
|
||||||
Trigger<> *get_open_trigger() { return &this->open_trigger_; }
|
Trigger<> *get_open_trigger() const { return this->open_trigger_; }
|
||||||
void set_open_sensor(sensor::Sensor *open_sensor) { this->open_sensor_ = open_sensor; }
|
void set_open_sensor(sensor::Sensor *open_sensor) { this->open_sensor_ = open_sensor; }
|
||||||
void set_open_moving_current_threshold(float open_moving_current_threshold) {
|
void set_open_moving_current_threshold(float open_moving_current_threshold) {
|
||||||
this->open_moving_current_threshold_ = open_moving_current_threshold;
|
this->open_moving_current_threshold_ = open_moving_current_threshold;
|
||||||
@@ -27,7 +28,7 @@ class CurrentBasedCover : public cover::Cover, public Component {
|
|||||||
}
|
}
|
||||||
void set_open_duration(uint32_t open_duration) { this->open_duration_ = open_duration; }
|
void set_open_duration(uint32_t open_duration) { this->open_duration_ = open_duration; }
|
||||||
|
|
||||||
Trigger<> *get_close_trigger() { return &this->close_trigger_; }
|
Trigger<> *get_close_trigger() const { return this->close_trigger_; }
|
||||||
void set_close_sensor(sensor::Sensor *close_sensor) { this->close_sensor_ = close_sensor; }
|
void set_close_sensor(sensor::Sensor *close_sensor) { this->close_sensor_ = close_sensor; }
|
||||||
void set_close_moving_current_threshold(float close_moving_current_threshold) {
|
void set_close_moving_current_threshold(float close_moving_current_threshold) {
|
||||||
this->close_moving_current_threshold_ = close_moving_current_threshold;
|
this->close_moving_current_threshold_ = close_moving_current_threshold;
|
||||||
@@ -43,7 +44,7 @@ class CurrentBasedCover : public cover::Cover, public Component {
|
|||||||
void set_malfunction_detection(bool malfunction_detection) { this->malfunction_detection_ = malfunction_detection; }
|
void set_malfunction_detection(bool malfunction_detection) { this->malfunction_detection_ = malfunction_detection; }
|
||||||
void set_start_sensing_delay(uint32_t start_sensing_delay) { this->start_sensing_delay_ = start_sensing_delay; }
|
void set_start_sensing_delay(uint32_t start_sensing_delay) { this->start_sensing_delay_ = start_sensing_delay; }
|
||||||
|
|
||||||
Trigger<> *get_malfunction_trigger() { return &this->malfunction_trigger_; }
|
Trigger<> *get_malfunction_trigger() const { return this->malfunction_trigger_; }
|
||||||
|
|
||||||
cover::CoverTraits get_traits() override;
|
cover::CoverTraits get_traits() override;
|
||||||
|
|
||||||
@@ -63,23 +64,23 @@ class CurrentBasedCover : public cover::Cover, public Component {
|
|||||||
|
|
||||||
void recompute_position_();
|
void recompute_position_();
|
||||||
|
|
||||||
Trigger<> stop_trigger_;
|
Trigger<> *stop_trigger_{new Trigger<>()};
|
||||||
|
|
||||||
sensor::Sensor *open_sensor_{nullptr};
|
sensor::Sensor *open_sensor_{nullptr};
|
||||||
Trigger<> open_trigger_;
|
Trigger<> *open_trigger_{new Trigger<>()};
|
||||||
float open_moving_current_threshold_;
|
float open_moving_current_threshold_;
|
||||||
float open_obstacle_current_threshold_{FLT_MAX};
|
float open_obstacle_current_threshold_{FLT_MAX};
|
||||||
uint32_t open_duration_;
|
uint32_t open_duration_;
|
||||||
|
|
||||||
sensor::Sensor *close_sensor_{nullptr};
|
sensor::Sensor *close_sensor_{nullptr};
|
||||||
Trigger<> close_trigger_;
|
Trigger<> *close_trigger_{new Trigger<>()};
|
||||||
float close_moving_current_threshold_;
|
float close_moving_current_threshold_;
|
||||||
float close_obstacle_current_threshold_{FLT_MAX};
|
float close_obstacle_current_threshold_{FLT_MAX};
|
||||||
uint32_t close_duration_;
|
uint32_t close_duration_;
|
||||||
|
|
||||||
uint32_t max_duration_{UINT32_MAX};
|
uint32_t max_duration_{UINT32_MAX};
|
||||||
bool malfunction_detection_{true};
|
bool malfunction_detection_{true};
|
||||||
Trigger<> malfunction_trigger_;
|
Trigger<> *malfunction_trigger_{new Trigger<>()};
|
||||||
uint32_t start_sensing_delay_;
|
uint32_t start_sensing_delay_;
|
||||||
float obstacle_rollback_;
|
float obstacle_rollback_;
|
||||||
|
|
||||||
|
|||||||
@@ -104,6 +104,8 @@ void DalyBmsComponent::loop() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float DalyBmsComponent::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void DalyBmsComponent::request_data_(uint8_t data_id) {
|
void DalyBmsComponent::request_data_(uint8_t data_id) {
|
||||||
uint8_t request_message[DALY_FRAME_SIZE];
|
uint8_t request_message[DALY_FRAME_SIZE];
|
||||||
|
|
||||||
|
|||||||
@@ -72,6 +72,7 @@ class DalyBmsComponent : public PollingComponent, public uart::UARTDevice {
|
|||||||
void update() override;
|
void update() override;
|
||||||
void loop() override;
|
void loop() override;
|
||||||
|
|
||||||
|
float get_setup_priority() const override;
|
||||||
void set_address(uint8_t address) { this->addr_ = address; }
|
void set_address(uint8_t address) { this->addr_ = address; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
from esphome.components import sensor
|
from esphome.components import sensor
|
||||||
from esphome.components.const import ICON_CURRENT_DC, UNIT_AMPERE_HOUR
|
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
CONF_BATTERY_LEVEL,
|
CONF_BATTERY_LEVEL,
|
||||||
@@ -56,11 +55,14 @@ CONF_CELL_15_VOLTAGE = "cell_15_voltage"
|
|||||||
CONF_CELL_16_VOLTAGE = "cell_16_voltage"
|
CONF_CELL_16_VOLTAGE = "cell_16_voltage"
|
||||||
CONF_CELL_17_VOLTAGE = "cell_17_voltage"
|
CONF_CELL_17_VOLTAGE = "cell_17_voltage"
|
||||||
CONF_CELL_18_VOLTAGE = "cell_18_voltage"
|
CONF_CELL_18_VOLTAGE = "cell_18_voltage"
|
||||||
|
ICON_CURRENT_DC = "mdi:current-dc"
|
||||||
ICON_BATTERY_OUTLINE = "mdi:battery-outline"
|
ICON_BATTERY_OUTLINE = "mdi:battery-outline"
|
||||||
ICON_THERMOMETER_CHEVRON_UP = "mdi:thermometer-chevron-up"
|
ICON_THERMOMETER_CHEVRON_UP = "mdi:thermometer-chevron-up"
|
||||||
ICON_THERMOMETER_CHEVRON_DOWN = "mdi:thermometer-chevron-down"
|
ICON_THERMOMETER_CHEVRON_DOWN = "mdi:thermometer-chevron-down"
|
||||||
ICON_CAR_BATTERY = "mdi:car-battery"
|
ICON_CAR_BATTERY = "mdi:car-battery"
|
||||||
|
|
||||||
|
UNIT_AMPERE_HOUR = "Ah"
|
||||||
|
|
||||||
TYPES = [
|
TYPES = [
|
||||||
CONF_VOLTAGE,
|
CONF_VOLTAGE,
|
||||||
CONF_CURRENT,
|
CONF_CURRENT,
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
#include "dfplayer.h"
|
#include "dfplayer.h"
|
||||||
#include "esphome/core/helpers.h"
|
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
@@ -132,17 +131,10 @@ void DFPlayer::send_cmd_(uint8_t cmd, uint16_t argument) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void DFPlayer::loop() {
|
void DFPlayer::loop() {
|
||||||
// Read all available bytes in batches to reduce UART call overhead.
|
// Read message
|
||||||
size_t avail = this->available();
|
while (this->available()) {
|
||||||
uint8_t buf[64];
|
uint8_t byte;
|
||||||
while (avail > 0) {
|
this->read_byte(&byte);
|
||||||
size_t to_read = std::min(avail, sizeof(buf));
|
|
||||||
if (!this->read_array(buf, to_read)) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
avail -= to_read;
|
|
||||||
for (size_t bi = 0; bi < to_read; bi++) {
|
|
||||||
uint8_t byte = buf[bi];
|
|
||||||
|
|
||||||
if (this->read_pos_ == DFPLAYER_READ_BUFFER_LENGTH)
|
if (this->read_pos_ == DFPLAYER_READ_BUFFER_LENGTH)
|
||||||
this->read_pos_ = 0;
|
this->read_pos_ = 0;
|
||||||
@@ -237,8 +229,7 @@ void DFPlayer::loop() {
|
|||||||
this->is_playing_ = false;
|
this->is_playing_ = false;
|
||||||
break;
|
break;
|
||||||
case 0x07:
|
case 0x07:
|
||||||
ESP_LOGE(TAG,
|
ESP_LOGE(TAG, "Insertion error (an inserting operation only can be done when a track is being played)");
|
||||||
"Insertion error (an inserting operation only can be done when a track is being played)");
|
|
||||||
break;
|
break;
|
||||||
case 0x08:
|
case 0x08:
|
||||||
ESP_LOGE(TAG, "SD card reading failed (SD card pulled out or damaged)");
|
ESP_LOGE(TAG, "SD card reading failed (SD card pulled out or damaged)");
|
||||||
@@ -276,7 +267,6 @@ void DFPlayer::loop() {
|
|||||||
this->read_pos_++;
|
this->read_pos_++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
void DFPlayer::dump_config() {
|
void DFPlayer::dump_config() {
|
||||||
ESP_LOGCONFIG(TAG, "DFPlayer:");
|
ESP_LOGCONFIG(TAG, "DFPlayer:");
|
||||||
this->check_uart_settings(9600);
|
this->check_uart_settings(9600);
|
||||||
|
|||||||
@@ -63,6 +63,8 @@ void DHT::update() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float DHT::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void DHT::set_dht_model(DHTModel model) {
|
void DHT::set_dht_model(DHTModel model) {
|
||||||
this->model_ = model;
|
this->model_ = model;
|
||||||
this->is_auto_detect_ = model == DHT_MODEL_AUTO_DETECT;
|
this->is_auto_detect_ = model == DHT_MODEL_AUTO_DETECT;
|
||||||
|
|||||||
@@ -51,6 +51,8 @@ class DHT : public PollingComponent {
|
|||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
/// Update sensor values and push them to the frontend.
|
/// Update sensor values and push them to the frontend.
|
||||||
void update() override;
|
void update() override;
|
||||||
|
/// HARDWARE_LATE setup priority.
|
||||||
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool read_sensor_(float *temperature, float *humidity, bool report_errors);
|
bool read_sensor_(float *temperature, float *humidity, bool report_errors);
|
||||||
|
|||||||
@@ -49,7 +49,7 @@ void DHT12Component::dump_config() {
|
|||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
}
|
}
|
||||||
|
float DHT12Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
bool DHT12Component::read_data_(uint8_t *data) {
|
bool DHT12Component::read_data_(uint8_t *data) {
|
||||||
if (!this->read_bytes(0, data, 5)) {
|
if (!this->read_bytes(0, data, 5)) {
|
||||||
ESP_LOGW(TAG, "Updating DHT12 failed!");
|
ESP_LOGW(TAG, "Updating DHT12 failed!");
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ class DHT12Component : public PollingComponent, public i2c::I2CDevice {
|
|||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
void update() override;
|
void update() override;
|
||||||
|
|
||||||
void set_temperature_sensor(sensor::Sensor *temperature_sensor) { temperature_sensor_ = temperature_sensor; }
|
void set_temperature_sensor(sensor::Sensor *temperature_sensor) { temperature_sensor_ = temperature_sensor; }
|
||||||
|
|||||||
@@ -1,57 +0,0 @@
|
|||||||
import esphome.codegen as cg
|
|
||||||
from esphome.components import uart
|
|
||||||
import esphome.config_validation as cv
|
|
||||||
from esphome.const import CONF_ID, PLATFORM_ESP32, PLATFORM_ESP8266
|
|
||||||
|
|
||||||
CODEOWNERS = ["@SimonFischer04"]
|
|
||||||
DEPENDENCIES = ["uart"]
|
|
||||||
|
|
||||||
CONF_DLMS_METER_ID = "dlms_meter_id"
|
|
||||||
CONF_DECRYPTION_KEY = "decryption_key"
|
|
||||||
CONF_PROVIDER = "provider"
|
|
||||||
|
|
||||||
PROVIDERS = {"generic": 0, "netznoe": 1}
|
|
||||||
|
|
||||||
dlms_meter_component_ns = cg.esphome_ns.namespace("dlms_meter")
|
|
||||||
DlmsMeterComponent = dlms_meter_component_ns.class_(
|
|
||||||
"DlmsMeterComponent", cg.Component, uart.UARTDevice
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def validate_key(value):
|
|
||||||
value = cv.string_strict(value)
|
|
||||||
if len(value) != 32:
|
|
||||||
raise cv.Invalid("Decryption key must be 32 hex characters (16 bytes)")
|
|
||||||
try:
|
|
||||||
return [int(value[i : i + 2], 16) for i in range(0, 32, 2)]
|
|
||||||
except ValueError as exc:
|
|
||||||
raise cv.Invalid("Decryption key must be hex values from 00 to FF") from exc
|
|
||||||
|
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.All(
|
|
||||||
cv.Schema(
|
|
||||||
{
|
|
||||||
cv.GenerateID(): cv.declare_id(DlmsMeterComponent),
|
|
||||||
cv.Required(CONF_DECRYPTION_KEY): validate_key,
|
|
||||||
cv.Optional(CONF_PROVIDER, default="generic"): cv.enum(
|
|
||||||
PROVIDERS, lower=True
|
|
||||||
),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
.extend(uart.UART_DEVICE_SCHEMA)
|
|
||||||
.extend(cv.COMPONENT_SCHEMA),
|
|
||||||
cv.only_on([PLATFORM_ESP8266, PLATFORM_ESP32]),
|
|
||||||
)
|
|
||||||
|
|
||||||
FINAL_VALIDATE_SCHEMA = uart.final_validate_device_schema(
|
|
||||||
"dlms_meter", baud_rate=2400, require_rx=True
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
async def to_code(config):
|
|
||||||
var = cg.new_Pvariable(config[CONF_ID])
|
|
||||||
await cg.register_component(var, config)
|
|
||||||
await uart.register_uart_device(var, config)
|
|
||||||
key = ", ".join(str(b) for b in config[CONF_DECRYPTION_KEY])
|
|
||||||
cg.add(var.set_decryption_key(cg.RawExpression(f"{{{key}}}")))
|
|
||||||
cg.add(var.set_provider(PROVIDERS[config[CONF_PROVIDER]]))
|
|
||||||
@@ -1,71 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <cstdint>
|
|
||||||
|
|
||||||
namespace esphome::dlms_meter {
|
|
||||||
|
|
||||||
/*
|
|
||||||
+-------------------------------+
|
|
||||||
| Ciphering Service |
|
|
||||||
+-------------------------------+
|
|
||||||
| System Title Length |
|
|
||||||
+-------------------------------+
|
|
||||||
| |
|
|
||||||
| |
|
|
||||||
| |
|
|
||||||
| System |
|
|
||||||
| Title |
|
|
||||||
| |
|
|
||||||
| |
|
|
||||||
| |
|
|
||||||
+-------------------------------+
|
|
||||||
| Length | (1 or 3 Bytes)
|
|
||||||
+-------------------------------+
|
|
||||||
| Security Control Byte |
|
|
||||||
+-------------------------------+
|
|
||||||
| |
|
|
||||||
| Frame |
|
|
||||||
| Counter |
|
|
||||||
| |
|
|
||||||
+-------------------------------+
|
|
||||||
| |
|
|
||||||
~ ~
|
|
||||||
Encrypted Payload
|
|
||||||
~ ~
|
|
||||||
| |
|
|
||||||
+-------------------------------+
|
|
||||||
|
|
||||||
Ciphering Service: 0xDB (General-Glo-Ciphering)
|
|
||||||
System Title Length: 0x08
|
|
||||||
System Title: Unique ID of meter
|
|
||||||
Length: 1 Byte=Length <= 127, 3 Bytes=Length > 127 (0x82 & 2 Bytes length)
|
|
||||||
Security Control Byte:
|
|
||||||
- Bit 3…0: Security_Suite_Id
|
|
||||||
- Bit 4: "A" subfield: indicates that authentication is applied
|
|
||||||
- Bit 5: "E" subfield: indicates that encryption is applied
|
|
||||||
- Bit 6: Key_Set subfield: 0 = Unicast, 1 = Broadcast
|
|
||||||
- Bit 7: Indicates the use of compression.
|
|
||||||
*/
|
|
||||||
|
|
||||||
static constexpr uint8_t DLMS_HEADER_LENGTH = 16;
|
|
||||||
static constexpr uint8_t DLMS_HEADER_EXT_OFFSET = 2; // Extra offset for extended length header
|
|
||||||
static constexpr uint8_t DLMS_CIPHER_OFFSET = 0;
|
|
||||||
static constexpr uint8_t DLMS_SYST_OFFSET = 1;
|
|
||||||
static constexpr uint8_t DLMS_LENGTH_OFFSET = 10;
|
|
||||||
static constexpr uint8_t TWO_BYTE_LENGTH = 0x82;
|
|
||||||
static constexpr uint8_t DLMS_LENGTH_CORRECTION = 5; // Header bytes included in length field
|
|
||||||
static constexpr uint8_t DLMS_SECBYTE_OFFSET = 11;
|
|
||||||
static constexpr uint8_t DLMS_FRAMECOUNTER_OFFSET = 12;
|
|
||||||
static constexpr uint8_t DLMS_FRAMECOUNTER_LENGTH = 4;
|
|
||||||
static constexpr uint8_t DLMS_PAYLOAD_OFFSET = 16;
|
|
||||||
static constexpr uint8_t GLO_CIPHERING = 0xDB;
|
|
||||||
static constexpr uint8_t DATA_NOTIFICATION = 0x0F;
|
|
||||||
static constexpr uint8_t TIMESTAMP_DATETIME = 0x0C;
|
|
||||||
static constexpr uint16_t MAX_MESSAGE_LENGTH = 512; // Maximum size of message (when having 2 bytes length in header).
|
|
||||||
|
|
||||||
// Provider specific quirks
|
|
||||||
static constexpr uint8_t NETZ_NOE_MAGIC_BYTE = 0x81; // Magic length byte used by Netz NOE
|
|
||||||
static constexpr uint8_t NETZ_NOE_EXPECTED_MESSAGE_LENGTH = 0xF8;
|
|
||||||
static constexpr uint8_t NETZ_NOE_EXPECTED_SECURITY_CONTROL_BYTE = 0x20;
|
|
||||||
|
|
||||||
} // namespace esphome::dlms_meter
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user