mirror of
https://github.com/esphome/esphome.git
synced 2025-10-24 12:43:51 +01:00
976 lines
35 KiB
Python
Executable File
976 lines
35 KiB
Python
Executable File
#!/usr/bin/env python3
|
||
"""Test ESPHome component builds with intelligent grouping.
|
||
|
||
This script replaces the bash test_build_components script with Python,
|
||
adding support for intelligent component grouping based on shared bus
|
||
configurations to reduce CI build time.
|
||
|
||
Features:
|
||
- Analyzes components for shared common bus configs
|
||
- Groups compatible components together
|
||
- Merges configs for grouped components
|
||
- Uses --testing-mode for grouped tests
|
||
- Maintains backward compatibility with single component testing
|
||
"""
|
||
|
||
from __future__ import annotations
|
||
|
||
import argparse
|
||
from collections import defaultdict
|
||
import hashlib
|
||
import os
|
||
from pathlib import Path
|
||
import subprocess
|
||
import sys
|
||
|
||
# Add esphome to path
|
||
sys.path.insert(0, str(Path(__file__).parent.parent))
|
||
|
||
# pylint: disable=wrong-import-position
|
||
from script.analyze_component_buses import (
|
||
BASE_BUS_COMPONENTS,
|
||
ISOLATED_COMPONENTS,
|
||
NO_BUSES_SIGNATURE,
|
||
analyze_all_components,
|
||
create_grouping_signature,
|
||
is_platform_component,
|
||
uses_local_file_references,
|
||
)
|
||
from script.merge_component_configs import merge_component_configs
|
||
|
||
# Platform-specific maximum group sizes
|
||
# ESP8266 has limited IRAM and can't handle large component groups
|
||
PLATFORM_MAX_GROUP_SIZE = {
|
||
"esp8266-ard": 10, # ESP8266 Arduino has limited IRAM
|
||
"esp8266-idf": 10, # ESP8266 IDF also has limited IRAM
|
||
# BK72xx now uses BK7252 board (1.62MB flash vs 1.03MB) - no limit needed
|
||
# Other platforms can handle larger groups
|
||
}
|
||
|
||
|
||
def show_disk_space_if_ci(esphome_command: str) -> None:
|
||
"""Show disk space usage if running in CI during compile.
|
||
|
||
Args:
|
||
esphome_command: The esphome command being run (config/compile/clean)
|
||
"""
|
||
if os.environ.get("GITHUB_ACTIONS") and esphome_command == "compile":
|
||
print("\n" + "=" * 80)
|
||
print("Disk Space After Build:")
|
||
print("=" * 80)
|
||
subprocess.run(["df", "-h"], check=False)
|
||
print("=" * 80 + "\n")
|
||
|
||
|
||
def find_component_tests(
|
||
components_dir: Path, component_pattern: str = "*"
|
||
) -> dict[str, list[Path]]:
|
||
"""Find all component test files.
|
||
|
||
Args:
|
||
components_dir: Path to tests/components directory
|
||
component_pattern: Glob pattern for component names
|
||
|
||
Returns:
|
||
Dictionary mapping component name to list of test files
|
||
"""
|
||
component_tests = defaultdict(list)
|
||
|
||
for comp_dir in components_dir.glob(component_pattern):
|
||
if not comp_dir.is_dir():
|
||
continue
|
||
|
||
for test_file in comp_dir.glob("test.*.yaml"):
|
||
component_tests[comp_dir.name].append(test_file)
|
||
|
||
return dict(component_tests)
|
||
|
||
|
||
def parse_test_filename(test_file: Path) -> tuple[str, str]:
|
||
"""Parse test filename to extract test name and platform.
|
||
|
||
Args:
|
||
test_file: Path to test file
|
||
|
||
Returns:
|
||
Tuple of (test_name, platform)
|
||
"""
|
||
parts = test_file.stem.split(".")
|
||
if len(parts) == 2:
|
||
return parts[0], parts[1] # test, platform
|
||
return parts[0], "all"
|
||
|
||
|
||
def get_platform_base_files(base_dir: Path) -> dict[str, list[Path]]:
|
||
"""Get all platform base files.
|
||
|
||
Args:
|
||
base_dir: Path to test_build_components directory
|
||
|
||
Returns:
|
||
Dictionary mapping platform to list of base files (for version variants)
|
||
"""
|
||
platform_files = defaultdict(list)
|
||
|
||
for base_file in base_dir.glob("build_components_base.*.yaml"):
|
||
# Extract platform from filename
|
||
# e.g., build_components_base.esp32-idf.yaml -> esp32-idf
|
||
# or build_components_base.esp32-idf-50.yaml -> esp32-idf
|
||
filename = base_file.stem
|
||
parts = filename.replace("build_components_base.", "").split("-")
|
||
|
||
# Platform is everything before version number (if present)
|
||
# Check if last part is a number (version)
|
||
platform = "-".join(parts[:-1]) if parts[-1].isdigit() else "-".join(parts)
|
||
|
||
platform_files[platform].append(base_file)
|
||
|
||
return dict(platform_files)
|
||
|
||
|
||
def extract_platform_with_version(base_file: Path) -> str:
|
||
"""Extract platform with version from base filename.
|
||
|
||
Args:
|
||
base_file: Path to base file
|
||
|
||
Returns:
|
||
Platform with version (e.g., "esp32-idf-50" or "esp32-idf")
|
||
"""
|
||
# Remove "build_components_base." prefix and ".yaml" suffix
|
||
return base_file.stem.replace("build_components_base.", "")
|
||
|
||
|
||
def run_esphome_test(
|
||
component: str,
|
||
test_file: Path,
|
||
platform: str,
|
||
platform_with_version: str,
|
||
base_file: Path,
|
||
build_dir: Path,
|
||
esphome_command: str,
|
||
continue_on_fail: bool,
|
||
use_testing_mode: bool = False,
|
||
) -> tuple[bool, str]:
|
||
"""Run esphome test for a single component.
|
||
|
||
Args:
|
||
component: Component name
|
||
test_file: Path to component test file
|
||
platform: Platform name (e.g., "esp32-idf")
|
||
platform_with_version: Platform with version (e.g., "esp32-idf-50")
|
||
base_file: Path to platform base file
|
||
build_dir: Path to build directory
|
||
esphome_command: ESPHome command (config/compile)
|
||
continue_on_fail: Whether to continue on failure
|
||
use_testing_mode: Whether to use --testing-mode flag
|
||
|
||
Returns:
|
||
Tuple of (success status, command string)
|
||
"""
|
||
test_name = test_file.stem.split(".")[0]
|
||
|
||
# Create dynamic test file in build directory
|
||
output_file = build_dir / f"{component}.{test_name}.{platform_with_version}.yaml"
|
||
|
||
# Copy base file and substitute component test file reference
|
||
base_content = base_file.read_text()
|
||
# Get relative path from build dir to test file
|
||
repo_root = Path(__file__).parent.parent
|
||
component_test_ref = f"../../{test_file.relative_to(repo_root / 'tests')}"
|
||
output_content = base_content.replace("$component_test_file", component_test_ref)
|
||
output_file.write_text(output_content)
|
||
|
||
# Build esphome command
|
||
cmd = [
|
||
sys.executable,
|
||
"-m",
|
||
"esphome",
|
||
]
|
||
|
||
# Add --testing-mode if needed (must be before subcommand)
|
||
if use_testing_mode:
|
||
cmd.append("--testing-mode")
|
||
|
||
# Add substitutions
|
||
cmd.extend(
|
||
[
|
||
"-s",
|
||
"component_name",
|
||
component,
|
||
"-s",
|
||
"component_dir",
|
||
f"../../components/{component}",
|
||
"-s",
|
||
"test_name",
|
||
test_name,
|
||
"-s",
|
||
"target_platform",
|
||
platform,
|
||
]
|
||
)
|
||
|
||
# Add command and config file
|
||
cmd.extend([esphome_command, str(output_file)])
|
||
|
||
# Build command string for display/logging
|
||
cmd_str = " ".join(cmd)
|
||
|
||
# Run command
|
||
print(f"> [{component}] [{test_name}] [{platform_with_version}]")
|
||
if use_testing_mode:
|
||
print(" (using --testing-mode)")
|
||
|
||
try:
|
||
result = subprocess.run(cmd, check=False)
|
||
success = result.returncode == 0
|
||
|
||
# Show disk space after build in CI during compile
|
||
show_disk_space_if_ci(esphome_command)
|
||
|
||
if not success and not continue_on_fail:
|
||
# Print command immediately for failed tests
|
||
print(f"\n{'=' * 80}")
|
||
print("FAILED - Command to reproduce:")
|
||
print(f"{'=' * 80}")
|
||
print(cmd_str)
|
||
print()
|
||
raise subprocess.CalledProcessError(result.returncode, cmd)
|
||
return success, cmd_str
|
||
except subprocess.CalledProcessError:
|
||
# Re-raise if we're not continuing on fail
|
||
if not continue_on_fail:
|
||
raise
|
||
return False, cmd_str
|
||
|
||
|
||
def run_grouped_test(
|
||
components: list[str],
|
||
platform: str,
|
||
platform_with_version: str,
|
||
base_file: Path,
|
||
build_dir: Path,
|
||
tests_dir: Path,
|
||
esphome_command: str,
|
||
continue_on_fail: bool,
|
||
) -> tuple[bool, str]:
|
||
"""Run esphome test for a group of components with shared bus configs.
|
||
|
||
Args:
|
||
components: List of component names to test together
|
||
platform: Platform name (e.g., "esp32-idf")
|
||
platform_with_version: Platform with version (e.g., "esp32-idf-50")
|
||
base_file: Path to platform base file
|
||
build_dir: Path to build directory
|
||
tests_dir: Path to tests/components directory
|
||
esphome_command: ESPHome command (config/compile)
|
||
continue_on_fail: Whether to continue on failure
|
||
|
||
Returns:
|
||
Tuple of (success status, command string)
|
||
"""
|
||
# Create merged config
|
||
group_name = "_".join(components[:3]) # Use first 3 components for name
|
||
if len(components) > 3:
|
||
group_name += f"_plus_{len(components) - 3}"
|
||
|
||
# Create unique device name by hashing sorted component list + platform
|
||
# This prevents conflicts when different component groups are tested
|
||
sorted_components = sorted(components)
|
||
hash_input = "_".join(sorted_components) + "_" + platform
|
||
group_hash = hashlib.md5(hash_input.encode()).hexdigest()[:8]
|
||
device_name = f"comptest{platform.replace('-', '')}{group_hash}"
|
||
|
||
merged_config_file = build_dir / f"merged_{group_name}.{platform_with_version}.yaml"
|
||
|
||
try:
|
||
merge_component_configs(
|
||
component_names=components,
|
||
platform=platform_with_version,
|
||
tests_dir=tests_dir,
|
||
output_file=merged_config_file,
|
||
)
|
||
except Exception as e: # pylint: disable=broad-exception-caught
|
||
print(f"Error merging configs for {components}: {e}")
|
||
if not continue_on_fail:
|
||
raise
|
||
# Return empty command string since we failed before building the command
|
||
return False, f"# Failed during config merge: {e}"
|
||
|
||
# Create test file that includes merged config
|
||
output_file = build_dir / f"test_{group_name}.{platform_with_version}.yaml"
|
||
base_content = base_file.read_text()
|
||
merged_ref = merged_config_file.name
|
||
output_content = base_content.replace("$component_test_file", merged_ref)
|
||
output_file.write_text(output_content)
|
||
|
||
# Build esphome command with --testing-mode
|
||
cmd = [
|
||
sys.executable,
|
||
"-m",
|
||
"esphome",
|
||
"--testing-mode", # Required for grouped tests
|
||
"-s",
|
||
"component_name",
|
||
device_name, # Use unique hash-based device name
|
||
"-s",
|
||
"component_dir",
|
||
"../../components",
|
||
"-s",
|
||
"test_name",
|
||
"merged",
|
||
"-s",
|
||
"target_platform",
|
||
platform,
|
||
esphome_command,
|
||
str(output_file),
|
||
]
|
||
|
||
# Build command string for display/logging
|
||
cmd_str = " ".join(cmd)
|
||
|
||
# Run command
|
||
components_str = ", ".join(components)
|
||
print(f"> [GROUPED: {components_str}] [{platform_with_version}]")
|
||
print(" (using --testing-mode)")
|
||
|
||
try:
|
||
result = subprocess.run(cmd, check=False)
|
||
success = result.returncode == 0
|
||
|
||
# Show disk space after build in CI during compile
|
||
show_disk_space_if_ci(esphome_command)
|
||
|
||
if not success and not continue_on_fail:
|
||
# Print command immediately for failed tests
|
||
print(f"\n{'=' * 80}")
|
||
print("FAILED - Command to reproduce:")
|
||
print(f"{'=' * 80}")
|
||
print(cmd_str)
|
||
print()
|
||
raise subprocess.CalledProcessError(result.returncode, cmd)
|
||
return success, cmd_str
|
||
except subprocess.CalledProcessError:
|
||
# Re-raise if we're not continuing on fail
|
||
if not continue_on_fail:
|
||
raise
|
||
return False, cmd_str
|
||
|
||
|
||
def run_grouped_component_tests(
|
||
all_tests: dict[str, list[Path]],
|
||
platform_filter: str | None,
|
||
platform_bases: dict[str, list[Path]],
|
||
tests_dir: Path,
|
||
build_dir: Path,
|
||
esphome_command: str,
|
||
continue_on_fail: bool,
|
||
additional_isolated: set[str] | None = None,
|
||
) -> tuple[set[tuple[str, str]], list[str], list[str], dict[str, str]]:
|
||
"""Run grouped component tests.
|
||
|
||
Args:
|
||
all_tests: Dictionary mapping component names to test files
|
||
platform_filter: Optional platform to filter by
|
||
platform_bases: Platform base files mapping
|
||
tests_dir: Path to tests/components directory
|
||
build_dir: Path to build directory
|
||
esphome_command: ESPHome command (config/compile)
|
||
continue_on_fail: Whether to continue on failure
|
||
additional_isolated: Additional components to treat as isolated (not grouped)
|
||
|
||
Returns:
|
||
Tuple of (tested_components, passed_tests, failed_tests, failed_commands)
|
||
"""
|
||
tested_components = set()
|
||
passed_tests = []
|
||
failed_tests = []
|
||
failed_commands = {} # Map test_id to command string
|
||
|
||
# Group components by platform and bus signature
|
||
grouped_components: dict[tuple[str, str], list[str]] = defaultdict(list)
|
||
print("\n" + "=" * 80)
|
||
print("Analyzing components for intelligent grouping...")
|
||
print("=" * 80)
|
||
component_buses, non_groupable, direct_bus_components = analyze_all_components(
|
||
tests_dir
|
||
)
|
||
|
||
# Track why components can't be grouped (for detailed output)
|
||
non_groupable_reasons = {}
|
||
|
||
# Merge additional isolated components with predefined ones
|
||
# ISOLATED COMPONENTS are tested individually WITHOUT --testing-mode
|
||
# This is critical because:
|
||
# - Grouped tests use --testing-mode which disables pin conflict checks and other validation
|
||
# - These checks are disabled to allow config merging (multiple components in one build)
|
||
# - For directly changed components (via --isolate), we need full validation to catch issues
|
||
# - Dependencies are safe to group since they weren't modified in the PR
|
||
all_isolated = set(ISOLATED_COMPONENTS.keys())
|
||
if additional_isolated:
|
||
all_isolated.update(additional_isolated)
|
||
|
||
# Group by (platform, bus_signature)
|
||
for component, platforms in component_buses.items():
|
||
if component not in all_tests:
|
||
continue
|
||
|
||
# Skip components that must be tested in isolation
|
||
# These are shown separately and should not be in non_groupable_reasons
|
||
if component in all_isolated:
|
||
continue
|
||
|
||
# Skip base bus components (these test the bus platforms themselves)
|
||
if component in BASE_BUS_COMPONENTS:
|
||
continue
|
||
|
||
# Skip components that use local file references or direct bus configs
|
||
if component in non_groupable:
|
||
# Track the reason (using pre-calculated results to avoid expensive re-analysis)
|
||
if component not in non_groupable_reasons:
|
||
if component in direct_bus_components:
|
||
non_groupable_reasons[component] = (
|
||
"Defines buses directly (not via packages) - NEEDS MIGRATION"
|
||
)
|
||
elif uses_local_file_references(tests_dir / component):
|
||
non_groupable_reasons[component] = (
|
||
"Uses local file references ($component_dir)"
|
||
)
|
||
elif is_platform_component(tests_dir / component):
|
||
non_groupable_reasons[component] = (
|
||
"Platform component (abstract base class)"
|
||
)
|
||
else:
|
||
non_groupable_reasons[component] = (
|
||
"Uses !extend or !remove directives"
|
||
)
|
||
continue
|
||
|
||
for platform, buses in platforms.items():
|
||
# Skip if platform doesn't match filter
|
||
if platform_filter and not platform.startswith(platform_filter):
|
||
continue
|
||
|
||
# Create signature for this component's bus configuration
|
||
# Components with no buses get NO_BUSES_SIGNATURE so they can be grouped together
|
||
if buses:
|
||
signature = create_grouping_signature({platform: buses}, platform)
|
||
else:
|
||
signature = NO_BUSES_SIGNATURE
|
||
|
||
# Add to grouped components (including those with no buses)
|
||
if signature:
|
||
grouped_components[(platform, signature)].append(component)
|
||
|
||
# Print detailed grouping plan
|
||
print("\nGrouping Plan:")
|
||
print("-" * 80)
|
||
|
||
# Show isolated components (must test individually due to known issues or direct changes)
|
||
isolated_in_tests = [c for c in all_isolated if c in all_tests]
|
||
if isolated_in_tests:
|
||
predefined_isolated = [c for c in isolated_in_tests if c in ISOLATED_COMPONENTS]
|
||
additional_in_tests = [
|
||
c for c in isolated_in_tests if c in (additional_isolated or set())
|
||
]
|
||
|
||
if predefined_isolated:
|
||
print(
|
||
f"\n⚠ {len(predefined_isolated)} components must be tested in isolation (known build issues):"
|
||
)
|
||
for comp in sorted(predefined_isolated):
|
||
reason = ISOLATED_COMPONENTS[comp]
|
||
print(f" - {comp}: {reason}")
|
||
|
||
if additional_in_tests:
|
||
print(
|
||
f"\n✓ {len(additional_in_tests)} components tested in isolation (directly changed in PR):"
|
||
)
|
||
for comp in sorted(additional_in_tests):
|
||
print(f" - {comp}")
|
||
|
||
# Show base bus components (test the bus platform implementations)
|
||
base_bus_in_tests = [c for c in BASE_BUS_COMPONENTS if c in all_tests]
|
||
if base_bus_in_tests:
|
||
print(
|
||
f"\n○ {len(base_bus_in_tests)} base bus platform components (tested individually):"
|
||
)
|
||
for comp in sorted(base_bus_in_tests):
|
||
print(f" - {comp}")
|
||
|
||
# Show excluded components with detailed reasons
|
||
if non_groupable_reasons:
|
||
excluded_in_tests = [c for c in non_groupable_reasons if c in all_tests]
|
||
if excluded_in_tests:
|
||
print(
|
||
f"\n⚠ {len(excluded_in_tests)} components excluded from grouping (each needs individual build):"
|
||
)
|
||
# Group by reason to show summary
|
||
direct_bus = [
|
||
c
|
||
for c in excluded_in_tests
|
||
if "NEEDS MIGRATION" in non_groupable_reasons.get(c, "")
|
||
]
|
||
if direct_bus:
|
||
print(
|
||
f"\n ⚠⚠⚠ {len(direct_bus)} DEFINE BUSES DIRECTLY - NEED MIGRATION TO PACKAGES:"
|
||
)
|
||
for comp in sorted(direct_bus):
|
||
print(f" - {comp}")
|
||
|
||
other_reasons = [
|
||
c
|
||
for c in excluded_in_tests
|
||
if "NEEDS MIGRATION" not in non_groupable_reasons.get(c, "")
|
||
]
|
||
if other_reasons and len(other_reasons) <= 10:
|
||
print("\n Other non-groupable components:")
|
||
for comp in sorted(other_reasons):
|
||
reason = non_groupable_reasons[comp]
|
||
print(f" - {comp}: {reason}")
|
||
elif other_reasons:
|
||
print(
|
||
f"\n Other non-groupable components: {len(other_reasons)} components"
|
||
)
|
||
|
||
# Distribute no_buses components into other groups to maximize efficiency
|
||
# Components with no buses can merge with any bus group since they have no conflicting requirements
|
||
no_buses_by_platform: dict[str, list[str]] = {}
|
||
for (platform, signature), components in list(grouped_components.items()):
|
||
if signature == NO_BUSES_SIGNATURE:
|
||
no_buses_by_platform[platform] = components
|
||
# Remove from grouped_components - we'll distribute them
|
||
del grouped_components[(platform, signature)]
|
||
|
||
# Distribute no_buses components into existing groups for each platform
|
||
for platform, no_buses_comps in no_buses_by_platform.items():
|
||
# Find all non-empty groups for this platform (excluding no_buses)
|
||
platform_groups = [
|
||
(sig, comps)
|
||
for (plat, sig), comps in grouped_components.items()
|
||
if plat == platform and sig != NO_BUSES_SIGNATURE
|
||
]
|
||
|
||
if platform_groups:
|
||
# Distribute no_buses components round-robin across existing groups
|
||
for i, comp in enumerate(no_buses_comps):
|
||
sig, _ = platform_groups[i % len(platform_groups)]
|
||
grouped_components[(platform, sig)].append(comp)
|
||
else:
|
||
# No other groups for this platform - keep no_buses components together
|
||
grouped_components[(platform, NO_BUSES_SIGNATURE)] = no_buses_comps
|
||
|
||
# Split groups that exceed platform-specific maximum sizes
|
||
# ESP8266 has limited IRAM and can't handle large component groups
|
||
split_groups = {}
|
||
for (platform, signature), components in list(grouped_components.items()):
|
||
max_size = PLATFORM_MAX_GROUP_SIZE.get(platform)
|
||
if max_size and len(components) > max_size:
|
||
# Split this group into smaller groups
|
||
print(
|
||
f"\n ℹ️ Splitting {platform} group (signature: {signature}) "
|
||
f"from {len(components)} to max {max_size} components per group"
|
||
)
|
||
# Remove original group
|
||
del grouped_components[(platform, signature)]
|
||
# Create split groups
|
||
for i in range(0, len(components), max_size):
|
||
split_components = components[i : i + max_size]
|
||
# Create unique signature for each split group
|
||
split_signature = f"{signature}_split{i // max_size + 1}"
|
||
split_groups[(platform, split_signature)] = split_components
|
||
# Add split groups back
|
||
grouped_components.update(split_groups)
|
||
|
||
groups_to_test = []
|
||
individual_tests = set() # Use set to avoid duplicates
|
||
|
||
for (platform, signature), components in sorted(grouped_components.items()):
|
||
if len(components) > 1:
|
||
groups_to_test.append((platform, signature, components))
|
||
# Note: Don't add single-component groups to individual_tests here
|
||
# They'll be added below when we check for ungrouped components
|
||
|
||
# Add components that weren't grouped on any platform
|
||
for component in all_tests:
|
||
if component not in [c for _, _, comps in groups_to_test for c in comps]:
|
||
individual_tests.add(component)
|
||
|
||
if groups_to_test:
|
||
print(f"\n✓ {len(groups_to_test)} groups will be tested together:")
|
||
for platform, signature, components in groups_to_test:
|
||
component_list = ", ".join(sorted(components))
|
||
print(f" [{platform}] [{signature}]: {component_list}")
|
||
print(
|
||
f" → {len(components)} components in 1 build (saves {len(components) - 1} builds)"
|
||
)
|
||
|
||
if individual_tests:
|
||
print(f"\n○ {len(individual_tests)} components will be tested individually:")
|
||
sorted_individual = sorted(individual_tests)
|
||
for comp in sorted_individual[:10]:
|
||
print(f" - {comp}")
|
||
if len(individual_tests) > 10:
|
||
print(f" ... and {len(individual_tests) - 10} more")
|
||
|
||
# Calculate actual build counts based on test files, not component counts
|
||
# Without grouping: every test file would be built separately
|
||
total_test_files = sum(len(test_files) for test_files in all_tests.values())
|
||
|
||
# With grouping:
|
||
# - 1 build per group (regardless of how many components)
|
||
# - Individual components still need all their platform builds
|
||
individual_test_file_count = sum(
|
||
len(all_tests[comp]) for comp in individual_tests if comp in all_tests
|
||
)
|
||
|
||
total_grouped_components = sum(len(comps) for _, _, comps in groups_to_test)
|
||
total_builds_with_grouping = len(groups_to_test) + individual_test_file_count
|
||
builds_saved = total_test_files - total_builds_with_grouping
|
||
|
||
print(f"\n{'=' * 80}")
|
||
print(
|
||
f"Summary: {total_builds_with_grouping} builds total (vs {total_test_files} without grouping)"
|
||
)
|
||
print(
|
||
f" • {len(groups_to_test)} grouped builds ({total_grouped_components} components)"
|
||
)
|
||
print(
|
||
f" • {individual_test_file_count} individual builds ({len(individual_tests)} components)"
|
||
)
|
||
if total_test_files > 0:
|
||
reduction_pct = (builds_saved / total_test_files) * 100
|
||
print(f" • Saves {builds_saved} builds ({reduction_pct:.1f}% reduction)")
|
||
print("=" * 80 + "\n")
|
||
|
||
# Execute grouped tests
|
||
for (platform, signature), components in grouped_components.items():
|
||
# Only group if we have multiple components with same signature
|
||
if len(components) <= 1:
|
||
continue
|
||
|
||
# Filter out components not in our test list
|
||
components_to_group = [c for c in components if c in all_tests]
|
||
if len(components_to_group) <= 1:
|
||
continue
|
||
|
||
# Get platform base files
|
||
if platform not in platform_bases:
|
||
continue
|
||
|
||
for base_file in platform_bases[platform]:
|
||
platform_with_version = extract_platform_with_version(base_file)
|
||
|
||
# Skip if platform filter doesn't match
|
||
if platform_filter and platform != platform_filter:
|
||
continue
|
||
if (
|
||
platform_filter
|
||
and platform_with_version != platform_filter
|
||
and not platform_with_version.startswith(f"{platform_filter}-")
|
||
):
|
||
continue
|
||
|
||
# Run grouped test
|
||
success, cmd_str = run_grouped_test(
|
||
components=components_to_group,
|
||
platform=platform,
|
||
platform_with_version=platform_with_version,
|
||
base_file=base_file,
|
||
build_dir=build_dir,
|
||
tests_dir=tests_dir,
|
||
esphome_command=esphome_command,
|
||
continue_on_fail=continue_on_fail,
|
||
)
|
||
|
||
# Mark all components as tested
|
||
for comp in components_to_group:
|
||
tested_components.add((comp, platform_with_version))
|
||
|
||
# Record result for each component - show all components in grouped tests
|
||
test_id = (
|
||
f"GROUPED[{','.join(components_to_group)}].{platform_with_version}"
|
||
)
|
||
if success:
|
||
passed_tests.append(test_id)
|
||
else:
|
||
failed_tests.append(test_id)
|
||
failed_commands[test_id] = cmd_str
|
||
|
||
return tested_components, passed_tests, failed_tests, failed_commands
|
||
|
||
|
||
def run_individual_component_test(
|
||
component: str,
|
||
test_file: Path,
|
||
platform: str,
|
||
platform_with_version: str,
|
||
base_file: Path,
|
||
build_dir: Path,
|
||
esphome_command: str,
|
||
continue_on_fail: bool,
|
||
tested_components: set[tuple[str, str]],
|
||
passed_tests: list[str],
|
||
failed_tests: list[str],
|
||
failed_commands: dict[str, str],
|
||
) -> None:
|
||
"""Run an individual component test if not already tested in a group.
|
||
|
||
Args:
|
||
component: Component name
|
||
test_file: Test file path
|
||
platform: Platform name
|
||
platform_with_version: Platform with version
|
||
base_file: Base file for platform
|
||
build_dir: Build directory
|
||
esphome_command: ESPHome command
|
||
continue_on_fail: Whether to continue on failure
|
||
tested_components: Set of already tested components
|
||
passed_tests: List to append passed test IDs
|
||
failed_tests: List to append failed test IDs
|
||
failed_commands: Dict to store failed test commands
|
||
"""
|
||
# Skip if already tested in a group
|
||
if (component, platform_with_version) in tested_components:
|
||
return
|
||
|
||
test_name = test_file.stem.split(".")[0]
|
||
success, cmd_str = run_esphome_test(
|
||
component=component,
|
||
test_file=test_file,
|
||
platform=platform,
|
||
platform_with_version=platform_with_version,
|
||
base_file=base_file,
|
||
build_dir=build_dir,
|
||
esphome_command=esphome_command,
|
||
continue_on_fail=continue_on_fail,
|
||
)
|
||
test_id = f"{component}.{test_name}.{platform_with_version}"
|
||
if success:
|
||
passed_tests.append(test_id)
|
||
else:
|
||
failed_tests.append(test_id)
|
||
failed_commands[test_id] = cmd_str
|
||
|
||
|
||
def test_components(
|
||
component_patterns: list[str],
|
||
platform_filter: str | None,
|
||
esphome_command: str,
|
||
continue_on_fail: bool,
|
||
enable_grouping: bool = True,
|
||
isolated_components: set[str] | None = None,
|
||
) -> int:
|
||
"""Test components with optional intelligent grouping.
|
||
|
||
Args:
|
||
component_patterns: List of component name patterns
|
||
platform_filter: Optional platform to filter by
|
||
esphome_command: ESPHome command (config/compile)
|
||
continue_on_fail: Whether to continue on failure
|
||
enable_grouping: Whether to enable component grouping
|
||
isolated_components: Set of component names to test in isolation (not grouped).
|
||
These are tested WITHOUT --testing-mode to enable full validation
|
||
(pin conflicts, etc). This is used in CI for directly changed components
|
||
to catch issues that would be missed with --testing-mode.
|
||
|
||
Returns:
|
||
Exit code (0 for success, 1 for failure)
|
||
"""
|
||
# Setup paths
|
||
repo_root = Path(__file__).parent.parent
|
||
tests_dir = repo_root / "tests" / "components"
|
||
build_components_dir = repo_root / "tests" / "test_build_components"
|
||
build_dir = build_components_dir / "build"
|
||
build_dir.mkdir(parents=True, exist_ok=True)
|
||
|
||
# Get platform base files
|
||
platform_bases = get_platform_base_files(build_components_dir)
|
||
|
||
# Find all component tests
|
||
all_tests = {}
|
||
for pattern in component_patterns:
|
||
all_tests.update(find_component_tests(tests_dir, pattern))
|
||
|
||
if not all_tests:
|
||
print(f"No components found matching: {component_patterns}")
|
||
return 1
|
||
|
||
print(f"Found {len(all_tests)} components to test")
|
||
|
||
# Run tests
|
||
failed_tests = []
|
||
passed_tests = []
|
||
tested_components = set() # Track which components were tested in groups
|
||
failed_commands = {} # Track commands for failed tests
|
||
|
||
# First, run grouped tests if grouping is enabled
|
||
if enable_grouping:
|
||
(
|
||
tested_components,
|
||
passed_tests,
|
||
failed_tests,
|
||
failed_commands,
|
||
) = run_grouped_component_tests(
|
||
all_tests=all_tests,
|
||
platform_filter=platform_filter,
|
||
platform_bases=platform_bases,
|
||
tests_dir=tests_dir,
|
||
build_dir=build_dir,
|
||
esphome_command=esphome_command,
|
||
continue_on_fail=continue_on_fail,
|
||
additional_isolated=isolated_components,
|
||
)
|
||
|
||
# Then run individual tests for components not in groups
|
||
for component, test_files in sorted(all_tests.items()):
|
||
for test_file in test_files:
|
||
test_name, platform = parse_test_filename(test_file)
|
||
|
||
# Handle "all" platform tests
|
||
if platform == "all":
|
||
# Run for all platforms
|
||
for plat, base_files in platform_bases.items():
|
||
if platform_filter and plat != platform_filter:
|
||
continue
|
||
|
||
for base_file in base_files:
|
||
platform_with_version = extract_platform_with_version(base_file)
|
||
run_individual_component_test(
|
||
component=component,
|
||
test_file=test_file,
|
||
platform=plat,
|
||
platform_with_version=platform_with_version,
|
||
base_file=base_file,
|
||
build_dir=build_dir,
|
||
esphome_command=esphome_command,
|
||
continue_on_fail=continue_on_fail,
|
||
tested_components=tested_components,
|
||
passed_tests=passed_tests,
|
||
failed_tests=failed_tests,
|
||
failed_commands=failed_commands,
|
||
)
|
||
else:
|
||
# Platform-specific test
|
||
if platform_filter and platform != platform_filter:
|
||
continue
|
||
|
||
if platform not in platform_bases:
|
||
print(f"No base file for platform: {platform}")
|
||
continue
|
||
|
||
for base_file in platform_bases[platform]:
|
||
platform_with_version = extract_platform_with_version(base_file)
|
||
|
||
# Skip if requested platform doesn't match
|
||
if (
|
||
platform_filter
|
||
and platform_with_version != platform_filter
|
||
and not platform_with_version.startswith(f"{platform_filter}-")
|
||
):
|
||
continue
|
||
|
||
run_individual_component_test(
|
||
component=component,
|
||
test_file=test_file,
|
||
platform=platform,
|
||
platform_with_version=platform_with_version,
|
||
base_file=base_file,
|
||
build_dir=build_dir,
|
||
esphome_command=esphome_command,
|
||
continue_on_fail=continue_on_fail,
|
||
tested_components=tested_components,
|
||
passed_tests=passed_tests,
|
||
failed_tests=failed_tests,
|
||
failed_commands=failed_commands,
|
||
)
|
||
|
||
# Print summary
|
||
print("\n" + "=" * 80)
|
||
print(f"Test Summary: {len(passed_tests)} passed, {len(failed_tests)} failed")
|
||
print("=" * 80)
|
||
|
||
if failed_tests:
|
||
print("\nFailed tests:")
|
||
for test in failed_tests:
|
||
print(f" - {test}")
|
||
|
||
# Print failed commands at the end for easy copy-paste from CI logs
|
||
print("\n" + "=" * 80)
|
||
print("Failed test commands (copy-paste to reproduce locally):")
|
||
print("=" * 80)
|
||
for test in failed_tests:
|
||
if test in failed_commands:
|
||
print(f"\n# {test}")
|
||
print(failed_commands[test])
|
||
print()
|
||
|
||
return 1
|
||
|
||
return 0
|
||
|
||
|
||
def main() -> int:
|
||
"""Main entry point."""
|
||
parser = argparse.ArgumentParser(
|
||
description="Test ESPHome component builds with intelligent grouping"
|
||
)
|
||
parser.add_argument(
|
||
"-e",
|
||
"--esphome-command",
|
||
default="compile",
|
||
choices=["config", "compile", "clean"],
|
||
help="ESPHome command to run (default: compile)",
|
||
)
|
||
parser.add_argument(
|
||
"-c",
|
||
"--components",
|
||
default="*",
|
||
help="Component pattern(s) to test (default: *). Comma-separated.",
|
||
)
|
||
parser.add_argument(
|
||
"-t",
|
||
"--target",
|
||
help="Target platform to test (e.g., esp32-idf)",
|
||
)
|
||
parser.add_argument(
|
||
"-f",
|
||
"--continue-on-fail",
|
||
action="store_true",
|
||
help="Continue testing even if a test fails",
|
||
)
|
||
parser.add_argument(
|
||
"--no-grouping",
|
||
action="store_true",
|
||
help="Disable component grouping (test each component individually)",
|
||
)
|
||
parser.add_argument(
|
||
"--isolate",
|
||
help="Comma-separated list of components to test in isolation (not grouped with others). "
|
||
"These are tested WITHOUT --testing-mode to enable full validation. "
|
||
"Used in CI for directly changed components to catch pin conflicts and other issues.",
|
||
)
|
||
|
||
args = parser.parse_args()
|
||
|
||
# Parse component patterns
|
||
component_patterns = [p.strip() for p in args.components.split(",")]
|
||
|
||
# Parse isolated components
|
||
isolated_components = None
|
||
if args.isolate:
|
||
isolated_components = {c.strip() for c in args.isolate.split(",") if c.strip()}
|
||
|
||
return test_components(
|
||
component_patterns=component_patterns,
|
||
platform_filter=args.target,
|
||
esphome_command=args.esphome_command,
|
||
continue_on_fail=args.continue_on_fail,
|
||
enable_grouping=not args.no_grouping,
|
||
isolated_components=isolated_components,
|
||
)
|
||
|
||
|
||
if __name__ == "__main__":
|
||
sys.exit(main())
|