mirror of
https://github.com/ChuckBuilds/LEDMatrix.git
synced 2026-04-10 21:03:01 +00:00
- Use get_logger() from src.logging_config instead of logging.getLogger() in visual_display_manager.py to match project logging conventions - Eliminate duplicate public/private weather draw methods — public draw_sun/ draw_cloud/draw_rain/draw_snow now delegate to the private _draw_* variants so plugins get consistent pixel output in tests vs production - Default install_deps=False in dev_server.py and render_plugin.py — dev scripts don't need to run pip install; developers are expected to have plugin deps installed in their venv already - Guard plugins_dir fixture against PermissionError during directory iteration - Fix PluginInstallManager.updateAll() to fall back to window.installedPlugins when PluginStateManager.installedPlugins is empty (plugins_manager.js populates window.installedPlugins independently of PluginStateManager) - Remove 5 debug console.log statements from plugins_manager.js button setup and initialization code Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
132 lines
3.5 KiB
Python
132 lines
3.5 KiB
Python
"""
|
|
Pytest fixtures for plugin integration tests.
|
|
"""
|
|
|
|
import pytest
|
|
import os
|
|
import sys
|
|
import json
|
|
from pathlib import Path
|
|
from unittest.mock import MagicMock, Mock
|
|
from typing import Dict, Any
|
|
|
|
# Add project root to path
|
|
project_root = Path(__file__).parent.parent.parent
|
|
if str(project_root) not in sys.path:
|
|
sys.path.insert(0, str(project_root))
|
|
|
|
# Set emulator mode
|
|
os.environ['EMULATOR'] = 'true'
|
|
|
|
|
|
@pytest.fixture
|
|
def plugins_dir():
|
|
"""Get the plugins directory path.
|
|
|
|
Checks plugins/ first, then falls back to plugin-repos/
|
|
for monorepo development environments.
|
|
"""
|
|
plugins_path = project_root / 'plugins'
|
|
plugin_repos_path = project_root / 'plugin-repos'
|
|
|
|
# Prefer plugins/ if it has actual plugin directories
|
|
if plugins_path.exists():
|
|
try:
|
|
has_plugins = any(
|
|
p for p in plugins_path.iterdir()
|
|
if p.is_dir() and not p.name.startswith('.')
|
|
)
|
|
if has_plugins:
|
|
return plugins_path
|
|
except PermissionError:
|
|
pass
|
|
if plugin_repos_path.exists():
|
|
return plugin_repos_path
|
|
return plugins_path
|
|
|
|
|
|
@pytest.fixture
|
|
def mock_display_manager():
|
|
"""Create a mock DisplayManager for plugin tests."""
|
|
mock = MagicMock()
|
|
mock.width = 128
|
|
mock.height = 32
|
|
mock.clear = Mock()
|
|
mock.draw_text = Mock()
|
|
mock.draw_image = Mock()
|
|
mock.update_display = Mock()
|
|
mock.get_font = Mock(return_value=None)
|
|
# Some plugins access matrix.width/height
|
|
mock.matrix = MagicMock()
|
|
mock.matrix.width = 128
|
|
mock.matrix.height = 32
|
|
return mock
|
|
|
|
|
|
@pytest.fixture
|
|
def mock_cache_manager():
|
|
"""Create a mock CacheManager for plugin tests."""
|
|
mock = MagicMock()
|
|
mock._memory_cache = {}
|
|
|
|
def mock_get(key: str, max_age: int = 300) -> Any:
|
|
return mock._memory_cache.get(key)
|
|
|
|
def mock_set(key: str, data: Any, ttl: int = None) -> None:
|
|
mock._memory_cache[key] = data
|
|
|
|
def mock_clear(key: str = None) -> None:
|
|
if key:
|
|
mock._memory_cache.pop(key, None)
|
|
else:
|
|
mock._memory_cache.clear()
|
|
|
|
mock.get = Mock(side_effect=mock_get)
|
|
mock.set = Mock(side_effect=mock_set)
|
|
mock.clear = Mock(side_effect=mock_clear)
|
|
return mock
|
|
|
|
|
|
@pytest.fixture
|
|
def mock_plugin_manager():
|
|
"""Create a mock PluginManager for plugin tests."""
|
|
mock = MagicMock()
|
|
mock.plugins = {}
|
|
mock.plugin_manifests = {}
|
|
return mock
|
|
|
|
|
|
@pytest.fixture
|
|
def base_plugin_config():
|
|
"""Base configuration for plugins."""
|
|
return {
|
|
'enabled': True,
|
|
'update_interval': 300
|
|
}
|
|
|
|
|
|
def load_plugin_manifest(plugin_id: str, plugins_dir: Path) -> Dict[str, Any]:
|
|
"""Load plugin manifest.json."""
|
|
manifest_path = plugins_dir / plugin_id / 'manifest.json'
|
|
if not manifest_path.exists():
|
|
pytest.skip(f"Manifest not found for {plugin_id}")
|
|
|
|
with open(manifest_path, 'r') as f:
|
|
return json.load(f)
|
|
|
|
|
|
def get_plugin_config_schema(plugin_id: str, plugins_dir: Path) -> Dict[str, Any]:
|
|
"""Load plugin config_schema.json if it exists."""
|
|
schema_path = plugins_dir / plugin_id / 'config_schema.json'
|
|
if schema_path.exists():
|
|
with open(schema_path, 'r') as f:
|
|
return json.load(f)
|
|
return None
|
|
|
|
|
|
@pytest.fixture
|
|
def visual_display_manager():
|
|
"""Create a VisualTestDisplayManager that renders real pixels for visual testing."""
|
|
from src.plugin_system.testing import VisualTestDisplayManager
|
|
return VisualTestDisplayManager(width=128, height=32)
|