mirror of
https://github.com/esphome/esphome.git
synced 2026-03-26 17:50:28 +08:00
[core] cpp tests: Allow customizing code generation during tests (#14681)
Co-authored-by: J. Nick Koston <nick@home-assistant.io> Co-authored-by: J. Nick Koston <nick@koston.org>
This commit is contained in:
460
script/build_helpers.py
Normal file
460
script/build_helpers.py
Normal file
@@ -0,0 +1,460 @@
|
||||
"""Shared helpers for C++ unit test and benchmark build scripts."""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
from collections.abc import Callable
|
||||
import hashlib
|
||||
import importlib.util
|
||||
import os
|
||||
from pathlib import Path
|
||||
import subprocess
|
||||
import sys
|
||||
|
||||
from helpers import get_all_dependencies, root_path as _root_path
|
||||
import yaml
|
||||
|
||||
# Ensure the repo root is on sys.path so that ``tests.testing_helpers`` and
|
||||
# override ``__init__.py`` modules can ``from tests.testing_helpers import ...``.
|
||||
if _root_path not in sys.path:
|
||||
sys.path.insert(0, _root_path)
|
||||
|
||||
from esphome.__main__ import command_compile, parse_args
|
||||
from esphome.config import validate_config
|
||||
from esphome.const import CONF_PLATFORM
|
||||
from esphome.core import CORE
|
||||
from esphome.loader import get_component, get_platform
|
||||
from esphome.platformio_api import get_idedata
|
||||
from tests.testing_helpers import ComponentManifestOverride, set_testing_manifest
|
||||
|
||||
# This must coincide with the version in /platformio.ini
|
||||
PLATFORMIO_GOOGLE_TEST_LIB = "google/googletest@^1.15.2"
|
||||
|
||||
# Google Benchmark library for PlatformIO
|
||||
# Format: name=repository_url (see esphome/core/config.py library parsing)
|
||||
PLATFORMIO_GOOGLE_BENCHMARK_LIB = (
|
||||
"benchmark=https://github.com/google/benchmark.git#v1.9.1"
|
||||
)
|
||||
|
||||
# Key names for the base config sections
|
||||
ESPHOME_KEY = "esphome"
|
||||
HOST_KEY = "host"
|
||||
LOGGER_KEY = "logger"
|
||||
|
||||
# Exit codes
|
||||
EXIT_OK = 0
|
||||
EXIT_SKIPPED = 1
|
||||
EXIT_COMPILE_ERROR = 2
|
||||
EXIT_CONFIG_ERROR = 3
|
||||
EXIT_NO_EXECUTABLE = 4
|
||||
|
||||
# Name of the per-component YAML config file in benchmark directories
|
||||
BENCHMARK_YAML_FILENAME = "benchmark.yaml"
|
||||
|
||||
|
||||
def hash_components(components: list[str]) -> str:
|
||||
"""Create a short hash of component names for unique config naming."""
|
||||
key = ",".join(components)
|
||||
return hashlib.sha256(key.encode()).hexdigest()[:16]
|
||||
|
||||
|
||||
def filter_components_with_files(components: list[str], tests_dir: Path) -> list[str]:
|
||||
"""Filter out components that do not have .cpp or .h files in the tests dir.
|
||||
|
||||
Args:
|
||||
components: List of component names to check
|
||||
tests_dir: Base directory containing component test/benchmark folders
|
||||
|
||||
Returns:
|
||||
Filtered list of components that have test files
|
||||
"""
|
||||
filtered_components: list[str] = []
|
||||
for component in components:
|
||||
test_dir = tests_dir / component
|
||||
if test_dir.is_dir() and (
|
||||
any(test_dir.glob("*.cpp")) or any(test_dir.glob("*.h"))
|
||||
):
|
||||
filtered_components.append(component)
|
||||
else:
|
||||
print(
|
||||
f"WARNING: No files found for component '{component}' in {test_dir}, skipping.",
|
||||
file=sys.stderr,
|
||||
)
|
||||
return filtered_components
|
||||
|
||||
|
||||
def get_platform_components(components: list[str], tests_dir: Path) -> list[str]:
|
||||
"""Discover platform sub-components referenced by test directory structure.
|
||||
|
||||
For each component, any sub-directory named after a platform domain
|
||||
(e.g. ``sensor``, ``binary_sensor``) is treated as a request to include
|
||||
that ``<domain>.<component>`` platform in the build.
|
||||
|
||||
Args:
|
||||
components: List of component names to scan
|
||||
tests_dir: Base directory containing component test/benchmark folders
|
||||
|
||||
Returns:
|
||||
List of ``"domain.component"`` strings
|
||||
"""
|
||||
platform_components: list[str] = []
|
||||
for component in components:
|
||||
test_dir = tests_dir / component
|
||||
if not test_dir.is_dir():
|
||||
continue
|
||||
for domain_dir in test_dir.iterdir():
|
||||
if not domain_dir.is_dir():
|
||||
continue
|
||||
domain = domain_dir.name
|
||||
if domain.startswith("__"):
|
||||
continue
|
||||
domain_module = get_component(domain)
|
||||
if domain_module is None or not domain_module.is_platform_component:
|
||||
raise ValueError(
|
||||
f"Component '{component}' references non-existing or invalid domain '{domain}'"
|
||||
f" in its directory structure. See ({tests_dir / component / domain})."
|
||||
)
|
||||
platform_components.append(f"{domain}.{component}")
|
||||
return platform_components
|
||||
|
||||
|
||||
def load_component_yaml_configs(components: list[str], tests_dir: Path) -> dict:
|
||||
"""Load and merge benchmark.yaml files from component directories.
|
||||
|
||||
Each component directory may contain a ``benchmark.yaml`` file that
|
||||
declares additional ESPHome components needed for the build (e.g.
|
||||
``api:``, ``sensor:``). These get merged into the base config before
|
||||
validation so that dependencies are properly resolved with defaults.
|
||||
|
||||
The ``esphome:`` key is special: its sub-keys are merged into the
|
||||
existing esphome config (e.g. to add ``areas:`` or ``devices:``).
|
||||
Keys already present in the base config (e.g. ``host:``, ``logger:``)
|
||||
are protected by ``setdefault`` in the caller.
|
||||
|
||||
Args:
|
||||
components: List of component directory names
|
||||
tests_dir: Base directory containing component folders
|
||||
|
||||
Returns:
|
||||
Merged dict of component configs to add to the base config
|
||||
"""
|
||||
merged: dict = {}
|
||||
for component in components:
|
||||
yaml_path = tests_dir / component / BENCHMARK_YAML_FILENAME
|
||||
if not yaml_path.is_file():
|
||||
continue
|
||||
with open(yaml_path) as f:
|
||||
component_config = yaml.safe_load(f)
|
||||
if component_config and isinstance(component_config, dict):
|
||||
for key, value in component_config.items():
|
||||
if key == ESPHOME_KEY and isinstance(value, dict):
|
||||
# Merge esphome sub-keys rather than replacing
|
||||
esphome_extra = merged.setdefault(ESPHOME_KEY, {})
|
||||
for sub_key, sub_value in value.items():
|
||||
esphome_extra.setdefault(sub_key, sub_value)
|
||||
continue
|
||||
merged.setdefault(key, value)
|
||||
return merged
|
||||
|
||||
|
||||
def create_host_config(
|
||||
config_name: str,
|
||||
friendly_name: str,
|
||||
libraries: str | list[str],
|
||||
includes: list[str],
|
||||
platformio_options: dict,
|
||||
) -> dict:
|
||||
"""Create an ESPHome host configuration for C++ builds.
|
||||
|
||||
Args:
|
||||
config_name: Unique name for this configuration
|
||||
friendly_name: Human-readable name
|
||||
libraries: PlatformIO library specification(s)
|
||||
includes: List of include folders for the build
|
||||
platformio_options: Dict of platformio_options to set
|
||||
|
||||
Returns:
|
||||
Configuration dict for ESPHome
|
||||
"""
|
||||
return {
|
||||
ESPHOME_KEY: {
|
||||
"name": config_name,
|
||||
"friendly_name": friendly_name,
|
||||
"libraries": libraries,
|
||||
"platformio_options": platformio_options,
|
||||
"includes": includes,
|
||||
},
|
||||
HOST_KEY: {},
|
||||
LOGGER_KEY: {"level": "DEBUG"},
|
||||
}
|
||||
|
||||
|
||||
def _wrap_manifest(
|
||||
comp_name: str,
|
||||
) -> ComponentManifestOverride | None:
|
||||
"""Wrap a component manifest in a ComponentManifestOverride with to_code suppressed.
|
||||
|
||||
If the manifest is already wrapped or not found, returns None.
|
||||
Otherwise returns the newly created override after installing it.
|
||||
"""
|
||||
if "." in comp_name:
|
||||
domain, component = comp_name.split(".", maxsplit=1)
|
||||
manifest = get_platform(domain, component)
|
||||
cache_key = f"{component}.{domain}"
|
||||
else:
|
||||
manifest = get_component(comp_name)
|
||||
cache_key = comp_name
|
||||
|
||||
if manifest is None or isinstance(manifest, ComponentManifestOverride):
|
||||
return None
|
||||
|
||||
override = ComponentManifestOverride(manifest)
|
||||
override.to_code = None # suppress by default
|
||||
set_testing_manifest(cache_key, override)
|
||||
return override
|
||||
|
||||
|
||||
def load_test_manifest_overrides(
|
||||
components: list[str],
|
||||
tests_dir: Path,
|
||||
) -> None:
|
||||
"""Apply per-component manifest overrides from test ``__init__.py`` files.
|
||||
|
||||
For every component, wraps its manifest and suppresses ``to_code``.
|
||||
If the component's test directory contains an ``__init__.py`` that
|
||||
defines ``override_manifest(manifest)``, it is called to customise
|
||||
the override (e.g. ``manifest.enable_codegen()``).
|
||||
"""
|
||||
for comp_name in components:
|
||||
override = _wrap_manifest(comp_name)
|
||||
if override is None:
|
||||
continue
|
||||
|
||||
if "." in comp_name:
|
||||
domain, component = comp_name.split(".", maxsplit=1)
|
||||
cache_key = f"{component}.{domain}"
|
||||
test_init = tests_dir / component / domain / "__init__.py"
|
||||
else:
|
||||
cache_key = comp_name
|
||||
test_init = tests_dir / comp_name / "__init__.py"
|
||||
|
||||
if not test_init.is_file():
|
||||
continue
|
||||
spec = importlib.util.spec_from_file_location(
|
||||
f"_test_manifest_override.{cache_key}", test_init
|
||||
)
|
||||
if spec is None or spec.loader is None:
|
||||
continue
|
||||
mod = importlib.util.module_from_spec(spec)
|
||||
spec.loader.exec_module(mod)
|
||||
override_fn = getattr(mod, "override_manifest", None)
|
||||
if override_fn is not None:
|
||||
override_fn(override)
|
||||
|
||||
|
||||
# Type alias for manifest override loaders
|
||||
ManifestOverrideLoader = Callable[[list[str]], None]
|
||||
|
||||
|
||||
def compile_and_get_binary(
|
||||
config: dict,
|
||||
components: list[str],
|
||||
tests_dir: Path,
|
||||
manifest_override_loader: ManifestOverrideLoader,
|
||||
label: str = "build",
|
||||
) -> tuple[int, str | None]:
|
||||
"""Compile an ESPHome configuration and return the binary path.
|
||||
|
||||
Args:
|
||||
config: ESPHome configuration dict (already created via create_host_config)
|
||||
components: List of components to include in the build
|
||||
tests_dir: Base directory for test files (used as config_path base)
|
||||
manifest_override_loader: Callback to apply manifest overrides for components
|
||||
label: Label for log messages (e.g. "unit tests", "benchmarks")
|
||||
|
||||
Returns:
|
||||
Tuple of (exit_code, program_path_or_none)
|
||||
"""
|
||||
# Load any benchmark.yaml files from component directories and merge
|
||||
# them into the config BEFORE dependency resolution and validation.
|
||||
# This allows each benchmark/test dir to declare which ESPHome components
|
||||
# it needs (e.g. api:) so they get proper config defaults.
|
||||
extra_config = load_component_yaml_configs(components, tests_dir)
|
||||
for key, value in extra_config.items():
|
||||
if key == ESPHOME_KEY and isinstance(value, dict):
|
||||
# Merge esphome sub-keys into existing esphome config.
|
||||
# For list values (e.g. libraries), extend rather than replace.
|
||||
for sub_key, sub_value in value.items():
|
||||
existing = config[ESPHOME_KEY].get(sub_key)
|
||||
if existing is not None and isinstance(sub_value, list):
|
||||
# Ensure existing is a list, then extend
|
||||
if not isinstance(existing, list):
|
||||
config[ESPHOME_KEY][sub_key] = [existing]
|
||||
config[ESPHOME_KEY][sub_key].extend(sub_value)
|
||||
else:
|
||||
config[ESPHOME_KEY].setdefault(sub_key, sub_value)
|
||||
else:
|
||||
config.setdefault(key, value)
|
||||
|
||||
# Apply manifest overrides before dependency resolution so that any
|
||||
# dependency additions made by override_manifest() are picked up.
|
||||
manifest_override_loader(components)
|
||||
|
||||
# Obtain possible dependencies BEFORE validate_config, because
|
||||
# get_all_dependencies calls CORE.reset() which clears build_path.
|
||||
components_with_dependencies: list[str] = sorted(
|
||||
get_all_dependencies(set(components))
|
||||
)
|
||||
|
||||
# Apply overrides for any transitively discovered dependencies.
|
||||
manifest_override_loader(components_with_dependencies)
|
||||
|
||||
CORE.config_path = tests_dir / "dummy.yaml"
|
||||
CORE.dashboard = None
|
||||
|
||||
# Validate config will expand the above with defaults:
|
||||
config = validate_config(config, {})
|
||||
|
||||
# Add remaining components and dependencies to the configuration after
|
||||
# validation, so their source files are included in the build.
|
||||
for component_name in components_with_dependencies:
|
||||
if "." in component_name:
|
||||
domain, component = component_name.split(".", maxsplit=1)
|
||||
domain_list = config.setdefault(domain, [])
|
||||
CORE.testing_ensure_platform_registered(domain)
|
||||
domain_list.append({CONF_PLATFORM: component})
|
||||
# Skip "core" — it's a pseudo-component handled by the build
|
||||
# system, not a real loadable component (get_component returns None)
|
||||
elif get_component(component_name) is not None:
|
||||
config.setdefault(component_name, [])
|
||||
|
||||
# Register platforms from the extra config (benchmark.yaml) so
|
||||
# USE_SENSOR, USE_LIGHT, etc. defines are emitted without needing
|
||||
# real entity instances.
|
||||
for key in extra_config:
|
||||
if key == ESPHOME_KEY:
|
||||
continue
|
||||
comp = get_component(key)
|
||||
if comp is not None and comp.is_platform_component:
|
||||
CORE.testing_ensure_platform_registered(key)
|
||||
|
||||
dependencies = set(components_with_dependencies) - set(components)
|
||||
deps_str = ", ".join(dependencies) if dependencies else "None"
|
||||
print(f"Building {label}: {', '.join(components)}. Dependencies: {deps_str}")
|
||||
CORE.config = config
|
||||
args = parse_args(["program", "compile", str(CORE.config_path)])
|
||||
try:
|
||||
exit_code: int = command_compile(args, config)
|
||||
|
||||
if exit_code != 0:
|
||||
print(f"Error compiling {label} for {', '.join(components)}")
|
||||
return exit_code, None
|
||||
except Exception as e:
|
||||
print(f"Error compiling {label} for {', '.join(components)}: {e}")
|
||||
return EXIT_COMPILE_ERROR, None
|
||||
|
||||
# After a successful compilation, locate the executable:
|
||||
idedata = get_idedata(config)
|
||||
if idedata is None:
|
||||
print("Cannot find executable")
|
||||
return EXIT_NO_EXECUTABLE, None
|
||||
|
||||
program_path: str = idedata.raw["prog_path"]
|
||||
return EXIT_OK, program_path
|
||||
|
||||
|
||||
def build_and_run(
|
||||
selected_components: list[str],
|
||||
tests_dir: Path,
|
||||
manifest_override_loader: ManifestOverrideLoader,
|
||||
config_prefix: str,
|
||||
friendly_name: str,
|
||||
libraries: str | list[str],
|
||||
platformio_options: dict,
|
||||
main_entry: str,
|
||||
label: str = "build",
|
||||
build_only: bool = False,
|
||||
extra_run_args: list[str] | None = None,
|
||||
extra_include_dirs: list[Path] | None = None,
|
||||
) -> int:
|
||||
"""Build and optionally run a C++ test/benchmark binary.
|
||||
|
||||
This is the main orchestration function shared between unit tests
|
||||
and benchmarks.
|
||||
|
||||
Args:
|
||||
selected_components: Components to include (directory names in tests_dir)
|
||||
tests_dir: Directory containing test/benchmark files
|
||||
manifest_override_loader: Callback to apply manifest overrides for components
|
||||
config_prefix: Prefix for the config name (e.g. "cpptests", "cppbench")
|
||||
friendly_name: Human-readable name for the config
|
||||
libraries: PlatformIO library specification(s)
|
||||
platformio_options: PlatformIO options dict
|
||||
main_entry: Name of the main entry file (e.g. "main.cpp")
|
||||
label: Label for log messages
|
||||
build_only: If True, print binary path and return without running
|
||||
extra_run_args: Extra arguments to pass to the binary
|
||||
extra_include_dirs: Additional directories whose .cpp files
|
||||
should be compiled (resolved relative to tests_dir if possible)
|
||||
|
||||
Returns:
|
||||
Exit code
|
||||
"""
|
||||
# Skip on Windows
|
||||
if os.name == "nt":
|
||||
print(f"Skipping {label} on Windows", file=sys.stderr)
|
||||
return EXIT_SKIPPED
|
||||
|
||||
# Remove components that do not have files
|
||||
components = filter_components_with_files(selected_components, tests_dir)
|
||||
|
||||
if len(components) == 0:
|
||||
print(
|
||||
f"No components specified or no files found for {label}.",
|
||||
file=sys.stderr,
|
||||
)
|
||||
return EXIT_OK
|
||||
|
||||
components = sorted(components)
|
||||
|
||||
# Build include list: main entry point + component folders + extra dirs
|
||||
includes: list[str] = [main_entry] + components
|
||||
if extra_include_dirs:
|
||||
for d in extra_include_dirs:
|
||||
if d.is_dir() and (any(d.glob("*.cpp")) or any(d.glob("*.h"))):
|
||||
# ESPHome includes are relative to the config directory (tests_dir)
|
||||
rel = os.path.relpath(d, tests_dir)
|
||||
includes.append(rel)
|
||||
|
||||
# Discover platform sub-components
|
||||
try:
|
||||
platform_components = get_platform_components(components, tests_dir)
|
||||
except ValueError as e:
|
||||
print(f"Error obtaining platform components: {e}")
|
||||
return EXIT_CONFIG_ERROR
|
||||
|
||||
components = sorted(components + platform_components)
|
||||
|
||||
# Create unique config name
|
||||
config_name: str = f"{config_prefix}-" + hash_components(components)
|
||||
|
||||
config = create_host_config(
|
||||
config_name, friendly_name, libraries, includes, platformio_options
|
||||
)
|
||||
|
||||
exit_code, program_path = compile_and_get_binary(
|
||||
config, components, tests_dir, manifest_override_loader, label
|
||||
)
|
||||
|
||||
if exit_code != EXIT_OK or program_path is None:
|
||||
return exit_code
|
||||
|
||||
if build_only:
|
||||
print(f"BUILD_BINARY={program_path}")
|
||||
return EXIT_OK
|
||||
|
||||
# Run the binary
|
||||
run_cmd: list[str] = [program_path]
|
||||
if extra_run_args:
|
||||
run_cmd.extend(extra_run_args)
|
||||
run_proc = subprocess.run(run_cmd, check=False)
|
||||
return run_proc.returncode
|
||||
Reference in New Issue
Block a user