mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-25 13:13:48 +01:00 
			
		
		
		
	Co-authored-by: Jesse Hills <3060199+jesserockz@users.noreply.github.com> Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com>
		
			
				
	
	
		
			166 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			166 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """Fixtures for component tests."""
 | |
| 
 | |
| from __future__ import annotations
 | |
| 
 | |
| from collections.abc import Callable, Generator
 | |
| from pathlib import Path
 | |
| import sys
 | |
| from typing import Any
 | |
| from unittest import mock
 | |
| 
 | |
| import pytest
 | |
| 
 | |
| from esphome import config, final_validate
 | |
| from esphome.const import (
 | |
|     KEY_CORE,
 | |
|     KEY_TARGET_FRAMEWORK,
 | |
|     KEY_TARGET_PLATFORM,
 | |
|     PlatformFramework,
 | |
| )
 | |
| from esphome.types import ConfigType
 | |
| from esphome.util import OrderedDict
 | |
| 
 | |
| # Add package root to python path
 | |
| here = Path(__file__).parent
 | |
| package_root = here.parent.parent
 | |
| sys.path.insert(0, package_root.as_posix())
 | |
| 
 | |
| from esphome.__main__ import generate_cpp_contents  # noqa: E402
 | |
| from esphome.config import Config, read_config  # noqa: E402
 | |
| from esphome.core import CORE  # noqa: E402
 | |
| 
 | |
| from .types import SetCoreConfigCallable  # noqa: E402
 | |
| 
 | |
| 
 | |
| @pytest.fixture(autouse=True)
 | |
| def config_path(request: pytest.FixtureRequest) -> Generator[None]:
 | |
|     """Set CORE.config_path to the component's config directory and reset it after the test."""
 | |
|     original_path = CORE.config_path
 | |
|     config_dir = Path(request.fspath).parent / "config"
 | |
| 
 | |
|     # Check if config directory exists, if not use parent directory
 | |
|     if config_dir.exists():
 | |
|         # Set config_path to a dummy yaml file in the config directory
 | |
|         # This ensures CORE.config_dir points to the config directory
 | |
|         CORE.config_path = config_dir / "dummy.yaml"
 | |
|     else:
 | |
|         CORE.config_path = Path(request.fspath).parent / "dummy.yaml"
 | |
| 
 | |
|     yield
 | |
|     CORE.config_path = original_path
 | |
| 
 | |
| 
 | |
| @pytest.fixture(autouse=True)
 | |
| def reset_core() -> Generator[None]:
 | |
|     """Reset CORE after each test."""
 | |
|     yield
 | |
|     CORE.reset()
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def set_core_config() -> Generator[SetCoreConfigCallable]:
 | |
|     """Fixture to set up the core configuration for tests."""
 | |
| 
 | |
|     def setter(
 | |
|         platform_framework: PlatformFramework,
 | |
|         /,
 | |
|         *,
 | |
|         core_data: ConfigType | None = None,
 | |
|         platform_data: ConfigType | None = None,
 | |
|         full_config: dict[str, ConfigType] | None = None,
 | |
|     ) -> None:
 | |
|         platform, framework = platform_framework.value
 | |
| 
 | |
|         # Set base core configuration
 | |
|         CORE.data[KEY_CORE] = {
 | |
|             KEY_TARGET_PLATFORM: platform.value,
 | |
|             KEY_TARGET_FRAMEWORK: framework.value,
 | |
|         }
 | |
| 
 | |
|         # Update with any additional core data
 | |
|         if core_data:
 | |
|             CORE.data[KEY_CORE].update(core_data)
 | |
| 
 | |
|         # Set platform-specific data
 | |
|         if platform_data:
 | |
|             CORE.data[platform.value] = platform_data
 | |
| 
 | |
|         config.path_context.set([])
 | |
|         final_validate.full_config.set(full_config or Config())
 | |
| 
 | |
|     yield setter
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def set_component_config() -> Callable[[str, Any], None]:
 | |
|     """
 | |
|     Fixture to set a component configuration in the mock config.
 | |
|     This must be used after the core configuration has been set up.
 | |
|     """
 | |
| 
 | |
|     def setter(name: str, value: Any) -> None:
 | |
|         final_validate.full_config.get()[name] = value
 | |
| 
 | |
|     return setter
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def component_fixture_path(request: pytest.FixtureRequest) -> Callable[[str], Path]:
 | |
|     """Return a function to get absolute paths relative to the component's fixtures directory."""
 | |
| 
 | |
|     def _get_path(file_name: str) -> Path:
 | |
|         """Get the absolute path of a file relative to the component's fixtures directory."""
 | |
|         return (Path(request.fspath).parent / "fixtures" / file_name).absolute()
 | |
| 
 | |
|     return _get_path
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def component_config_path(request: pytest.FixtureRequest) -> Callable[[str], Path]:
 | |
|     """Return a function to get absolute paths relative to the component's config directory."""
 | |
| 
 | |
|     def _get_path(file_name: str) -> Path:
 | |
|         """Get the absolute path of a file relative to the component's config directory."""
 | |
|         return (Path(request.fspath).parent / "config" / file_name).absolute()
 | |
| 
 | |
|     return _get_path
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def generate_main() -> Generator[Callable[[str | Path], str]]:
 | |
|     """Generates the C++ main.cpp from a given yaml file and returns it in string form."""
 | |
| 
 | |
|     def generator(path: str | Path) -> str:
 | |
|         CORE.config_path = Path(path)
 | |
|         CORE.config = read_config({})
 | |
|         generate_cpp_contents(CORE.config)
 | |
|         return CORE.cpp_main_section
 | |
| 
 | |
|     yield generator
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def mock_clone_or_update() -> Generator[Any]:
 | |
|     """Mock git.clone_or_update for testing."""
 | |
|     with mock.patch("esphome.git.clone_or_update") as mock_func:
 | |
|         # Default return value
 | |
|         mock_func.return_value = (Path("/tmp/test"), None)
 | |
|         yield mock_func
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def mock_load_yaml() -> Generator[Any]:
 | |
|     """Mock yaml_util.load_yaml for testing."""
 | |
| 
 | |
|     with mock.patch("esphome.yaml_util.load_yaml") as mock_func:
 | |
|         # Default return value
 | |
|         mock_func.return_value = OrderedDict({"sensor": []})
 | |
|         yield mock_func
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def mock_install_meta_finder() -> Generator[Any]:
 | |
|     """Mock loader.install_meta_finder for testing."""
 | |
|     with mock.patch("esphome.loader.install_meta_finder") as mock_func:
 | |
|         yield mock_func
 |