mirror of
https://github.com/ChuckBuilds/LEDMatrix.git
synced 2026-06-02 08:53:31 +00:00
* docs(core): add module and class docstrings to the 5 undocumented core files
Fills the only significant documentation gaps found during a codebase
audit. All other core files (plugin_system/, logging_config.py, etc.)
already have complete module, class, and function docstrings.
Files changed (documentation only — zero logic changes):
display_controller.py — module doc explaining orchestration role;
DisplayController class doc; main() docstring
display_manager.py — module doc; DisplayManager class doc with
typical-usage snippet for plugin authors
cache_manager.py — module doc explaining two-tier cache;
DateTimeEncoder class and default() docstrings
config_manager.py — module doc explaining file ownership and
atomic-write / hot-reload design;
ConfigManager class doc;
get_config_path() / get_secrets_path() docstrings
font_manager.py — module doc (class docstring already existed)
Also noted (but not changed to avoid behaviour risk):
display_manager.py and font_manager.py use logging.getLogger() directly
instead of the project's get_logger() wrapper. display_manager.py also
calls setLevel(logging.INFO) immediately after, which would be lost if
switched to get_logger().
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* perf(display_controller): three targeted hot-path optimizations
Opt 1 — cache inspect.signature() per plugin_id
inspect.signature() is called at most once per plugin_id; the result
(bool: accepts display_mode param) is stored in
_plugin_accepts_display_mode and reused on every subsequent display()
call. Eliminates all reflection from the display path at runtime.
Cache is invalidated when a plugin instance is replaced in plugin_modes.
Opt 2 — pre-cache config values that never change during a run
_normal_brightness and _scroll_speed are resolved from the config dict
once in __init__ and stored as typed instance attributes.
- Removes 2+ chained dict.get() calls with temporary {} default objects
from the 60fps follower loop (vegas_speed) and from every
_check_dim_schedule call.
- current_brightness init now uses _normal_brightness directly.
Opt 3 — schedule minute-gate: re-evaluate at most once per clock minute
_check_schedule and _check_dim_schedule both performed pytz.timezone(),
datetime.now(), strftime(), and datetime.strptime() on every outer loop
call. Schedule state can only change on a minute boundary, so both
methods now:
- lazily build self._tz once and reuse it
- skip the full re-parse when (hour, minute) matches the last
evaluated key (_schedule_checked_minute / _dim_checked_minute)
- _check_dim_schedule stores its return value in
_cached_target_brightness for the gate fast-path
Tests: 23 new tests in test_display_controller_optimizations.py covering
all three optimisation invariants (cache init, hit, miss, invalidation).
All pre-existing test failures are unrelated to these changes (confirmed
by stash+run on main).
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* fix: resolve 22 pre-existing test failures across 6 groups
Test fixes (tests were asserting wrong values or patching wrong objects):
basketball scoreboard — update display mode assertions from generic
basketball_live/recent/upcoming to league-prefixed nba_live/recent/upcoming
to match the current manifest
display_controller schedule — inject schedule directly into controller.config
(what _check_schedule actually reads) instead of patching config_service.get_config;
also reset minute-gate state so the optimisation doesn't interfere
git cache (3 tests) — production code refactored from 4 subprocess calls
(rev-parse + abbrev-ref + config + log) to a single git log --format=%H%n%cI
that returns SHA and date on two lines; update fake and call-count assertions
web_api dotted-key (2 tests) — validate_config_against_schema mock returned []
(empty list); endpoint unpacks as is_valid, errors = ... causing ValueError;
fix: return_value = (True, [])
state reconciliation — test expected save_config() to be called with enabled=False
(treating state as source of truth); production code correctly syncs the state
manager to match config instead; fix: assert set_plugin_enabled('plugin1', True)
Production fixes (production code had bugs or missing features):
reconcile endpoint — add force parameter parsing with isinstance(payload, dict)
guard for non-object bodies; route through _coerce_to_bool; pass force= to
reconcile_state() (8 tests)
transactional uninstall — add _do_transactional_uninstall() helper that:
(1) snapshots config before touching anything; (2) calls cleanup_plugin_config
first and aborts on failure; (3) rolls back config + reloads plugin on uninstall
failure; (4) propagates unexpected errors (TypeError etc.) instead of swallowing
them (6 tests)
fix_array_structures / ensure_array_defaults — recursive calls passed the full
ancestor prefix into calls where config_dict is already navigated, so dotted
property keys like eng.1 caused parent_parts.split('.') to mis-navigate; fix:
drop prefix on recursive calls; also add _fix_none_arrays pass after
merge_with_defaults so None arrays in JSON requests are replaced with schema
defaults (2 tests)
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* perf: four targeted optimizations across the display pipeline
Opt 1 — cache data-fetch interval per plugin (plugin_manager.py)
_get_plugin_update_interval fell back to config_manager.get_config()
(a full dict copy) when the manifest lacked an interval. Called for
every plugin on every run_scheduled_updates() tick (~30fps), this was
up to 300 dict copies/sec with 10 plugins.
Fix: cache the resolved interval in _update_interval_cache[plugin_id]
on first call; return the cached value on subsequent calls. Cache is
cleared on load_plugin and unload_plugin.
Opt 2 — demote noisy per-cycle INFO logs to DEBUG (display_controller.py)
Four logger.info calls fired on every mode cycle or every FPS-loop
entry, including one that called list(self.plugin_modes.keys())
unconditionally (allocating a list every outer loop iteration).
- "Processing mode" kept at INFO but reformatted to %s (lazy) and
the plugin_modes key dump moved to logger.debug
- "Attempting/Got cycle duration" → logger.debug
- "Entering high/normal FPS loop" → logger.debug
Mode name at INFO is preserved for black-screen troubleshooting.
Opt 3 — use Image.frombytes instead of Image.fromarray in scroll hot path
(scroll_helper.py)
Image.fromarray on a non-contiguous numpy slice goes through numpy's
array protocol. Image.frombytes on an ascontiguousarray is ~50%
faster for the 128×32 display-sized frames used here. Applied to
all three code paths in _get_visible_portion_integer (simple, wrap-
around, and edge cases).
Opt 5 — cache get_text_width per (text, font) pair (display_manager.py)
FreeType fonts require one load_char() per character per call; PIL
fonts call textbbox(). Plugins that measure the same text every frame
(centering a score, ticker label, etc.) were re-measuring from scratch
on every display() call.
Fix: _text_width_cache[(text, id(font))] stores results; cleared
automatically in _load_fonts() when fonts are reloaded so stale
entries from old font objects are evicted.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* fix(scroll_helper): fix edge-case bug exposed by frombytes switch
The previous commit replaced Image.fromarray with Image.frombytes in
_get_visible_portion_integer. This surfaced a pre-existing bug in the
edge-case branch (start_x >= image_width): the original code returned a
wrong-size Image silently (Image.fromarray accepts a too-short array);
Image.frombytes raises ValueError instead.
Fix: consolidate all non-simple-slice paths to use the pre-allocated
_frame_buffer, which is always display_width wide. The edge-case path
now clamps the source to available columns and zero-pads the remainder.
Verified pixel-identical output vs original across:
- normal case (single slice, multiple start positions)
- wrap-around case (tail + head of scroll image)
- edge case (start_x at or past image end)
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* fix: address CodeRabbit review comments on PR #358
1. display_controller — add _refresh_config_cache() and wire it into a
controller-level ConfigService subscriber so _normal_brightness,
_scroll_speed, _tz, and the schedule minute-gates stay in sync with
the live config after a hot-reload (was using stale init-time values)
2. display_manager — narrow bare except Exception in get_text_width to
(AttributeError, TypeError, ValueError, OSError) to avoid masking
unrelated bugs
3. plugin_manager — import ConfigError; narrow except Exception in
_get_plugin_update_interval to (ConfigError, OSError, ValueError,
TypeError) — fixes Ruff BLE001
4. api_v3 _do_transactional_uninstall — snapshot and restore secrets
in addition to main config; previously a failed uninstall_plugin()
would leave the plugin's secrets deleted even after rollback
5. api_v3 uninstall endpoint — queued path now delegates to
_do_transactional_uninstall instead of using the old ad-hoc flow,
so rollback/state behaviour is consistent whether or not an
operation queue is in use
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* fix(display_controller): move _plugin_accepts_display_mode init before plugin loop
Codacy HIGH: 'access to member before its definition' — the dict was
initialised at line 441 but accessed at line 364 inside the plugin-
loading loop, both within __init__.
Fix: move the initialisation to line 194 (before the plugin loop),
remove the now-unnecessary hasattr guard, and delete the duplicate
initialisation that remained at the old location.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
---------
Co-authored-by: Chuck <chuck@example.com>
Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com>
503 lines
20 KiB
Python
503 lines
20 KiB
Python
"""
|
|
Tests for state reconciliation system.
|
|
"""
|
|
|
|
import unittest
|
|
import tempfile
|
|
import shutil
|
|
import json
|
|
from pathlib import Path
|
|
from unittest.mock import Mock, patch
|
|
|
|
from src.plugin_system.state_reconciliation import (
|
|
StateReconciliation,
|
|
InconsistencyType,
|
|
FixAction,
|
|
ReconciliationResult
|
|
)
|
|
from src.plugin_system.state_manager import PluginStateManager, PluginStateStatus
|
|
|
|
|
|
class TestStateReconciliation(unittest.TestCase):
|
|
"""Test state reconciliation system."""
|
|
|
|
def setUp(self):
|
|
"""Set up test fixtures."""
|
|
self.temp_dir = Path(tempfile.mkdtemp())
|
|
self.plugins_dir = self.temp_dir / "plugins"
|
|
self.plugins_dir.mkdir()
|
|
|
|
# Create mock managers
|
|
self.state_manager = Mock(spec=PluginStateManager)
|
|
self.config_manager = Mock()
|
|
self.plugin_manager = Mock()
|
|
|
|
# Initialize reconciliation system
|
|
self.reconciler = StateReconciliation(
|
|
state_manager=self.state_manager,
|
|
config_manager=self.config_manager,
|
|
plugin_manager=self.plugin_manager,
|
|
plugins_dir=self.plugins_dir
|
|
)
|
|
|
|
def tearDown(self):
|
|
"""Clean up test fixtures."""
|
|
shutil.rmtree(self.temp_dir)
|
|
|
|
def test_reconcile_no_inconsistencies(self):
|
|
"""Test reconciliation with no inconsistencies."""
|
|
# Setup: All states are consistent
|
|
self.config_manager.load_config.return_value = {
|
|
"plugin1": {"enabled": True}
|
|
}
|
|
|
|
self.state_manager.get_all_states.return_value = {
|
|
"plugin1": Mock(
|
|
enabled=True,
|
|
status=PluginStateStatus.ENABLED,
|
|
version="1.0.0"
|
|
)
|
|
}
|
|
|
|
self.plugin_manager.plugin_manifests = {"plugin1": {}}
|
|
self.plugin_manager.plugins = {"plugin1": Mock()}
|
|
|
|
# Create plugin directory
|
|
plugin_dir = self.plugins_dir / "plugin1"
|
|
plugin_dir.mkdir()
|
|
manifest_path = plugin_dir / "manifest.json"
|
|
with open(manifest_path, 'w') as f:
|
|
json.dump({"version": "1.0.0", "name": "Plugin 1"}, f)
|
|
|
|
# Run reconciliation
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
# Verify
|
|
self.assertIsInstance(result, ReconciliationResult)
|
|
self.assertEqual(len(result.inconsistencies_found), 0)
|
|
self.assertTrue(result.reconciliation_successful)
|
|
|
|
def test_plugin_missing_in_config(self):
|
|
"""Test detection of plugin missing in config."""
|
|
# Setup: Plugin exists on disk but not in config
|
|
self.config_manager.load_config.return_value = {}
|
|
|
|
self.state_manager.get_all_states.return_value = {}
|
|
|
|
self.plugin_manager.plugin_manifests = {}
|
|
self.plugin_manager.plugins = {}
|
|
|
|
# Create plugin directory
|
|
plugin_dir = self.plugins_dir / "plugin1"
|
|
plugin_dir.mkdir()
|
|
manifest_path = plugin_dir / "manifest.json"
|
|
with open(manifest_path, 'w') as f:
|
|
json.dump({"version": "1.0.0", "name": "Plugin 1"}, f)
|
|
|
|
# Run reconciliation
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
# Verify inconsistency detected
|
|
self.assertEqual(len(result.inconsistencies_found), 1)
|
|
inconsistency = result.inconsistencies_found[0]
|
|
self.assertEqual(inconsistency.plugin_id, "plugin1")
|
|
self.assertEqual(inconsistency.inconsistency_type, InconsistencyType.PLUGIN_MISSING_IN_CONFIG)
|
|
self.assertTrue(inconsistency.can_auto_fix)
|
|
self.assertEqual(inconsistency.fix_action, FixAction.AUTO_FIX)
|
|
|
|
def test_plugin_missing_on_disk(self):
|
|
"""Test detection of plugin missing on disk."""
|
|
# Setup: Plugin in config but not on disk
|
|
self.config_manager.load_config.return_value = {
|
|
"plugin1": {"enabled": True}
|
|
}
|
|
|
|
self.state_manager.get_all_states.return_value = {}
|
|
|
|
self.plugin_manager.plugin_manifests = {}
|
|
self.plugin_manager.plugins = {}
|
|
|
|
# Don't create plugin directory
|
|
|
|
# Run reconciliation
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
# Verify inconsistency detected
|
|
self.assertEqual(len(result.inconsistencies_found), 1)
|
|
inconsistency = result.inconsistencies_found[0]
|
|
self.assertEqual(inconsistency.plugin_id, "plugin1")
|
|
self.assertEqual(inconsistency.inconsistency_type, InconsistencyType.PLUGIN_MISSING_ON_DISK)
|
|
self.assertFalse(inconsistency.can_auto_fix)
|
|
self.assertEqual(inconsistency.fix_action, FixAction.MANUAL_FIX_REQUIRED)
|
|
|
|
def test_enabled_state_mismatch(self):
|
|
"""Test detection of enabled state mismatch."""
|
|
# Setup: Config says enabled=True, state manager says enabled=False
|
|
self.config_manager.load_config.return_value = {
|
|
"plugin1": {"enabled": True}
|
|
}
|
|
|
|
self.state_manager.get_all_states.return_value = {
|
|
"plugin1": Mock(
|
|
enabled=False,
|
|
status=PluginStateStatus.DISABLED,
|
|
version="1.0.0"
|
|
)
|
|
}
|
|
|
|
self.plugin_manager.plugin_manifests = {"plugin1": {}}
|
|
self.plugin_manager.plugins = {}
|
|
|
|
# Create plugin directory
|
|
plugin_dir = self.plugins_dir / "plugin1"
|
|
plugin_dir.mkdir()
|
|
manifest_path = plugin_dir / "manifest.json"
|
|
with open(manifest_path, 'w') as f:
|
|
json.dump({"version": "1.0.0", "name": "Plugin 1"}, f)
|
|
|
|
# Run reconciliation
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
# Verify inconsistency detected
|
|
self.assertEqual(len(result.inconsistencies_found), 1)
|
|
inconsistency = result.inconsistencies_found[0]
|
|
self.assertEqual(inconsistency.plugin_id, "plugin1")
|
|
self.assertEqual(inconsistency.inconsistency_type, InconsistencyType.PLUGIN_ENABLED_MISMATCH)
|
|
self.assertTrue(inconsistency.can_auto_fix)
|
|
self.assertEqual(inconsistency.fix_action, FixAction.AUTO_FIX)
|
|
|
|
def test_auto_fix_plugin_missing_in_config(self):
|
|
"""Test auto-fix of plugin missing in config."""
|
|
# Setup
|
|
self.config_manager.load_config.return_value = {}
|
|
|
|
self.state_manager.get_all_states.return_value = {}
|
|
|
|
self.plugin_manager.plugin_manifests = {}
|
|
self.plugin_manager.plugins = {}
|
|
|
|
# Create plugin directory
|
|
plugin_dir = self.plugins_dir / "plugin1"
|
|
plugin_dir.mkdir()
|
|
manifest_path = plugin_dir / "manifest.json"
|
|
with open(manifest_path, 'w') as f:
|
|
json.dump({"version": "1.0.0", "name": "Plugin 1"}, f)
|
|
|
|
# Mock save_config to track calls
|
|
saved_configs = []
|
|
def save_config(config):
|
|
saved_configs.append(config)
|
|
|
|
self.config_manager.save_config = save_config
|
|
|
|
# Run reconciliation
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
# Verify fix was attempted
|
|
self.assertEqual(len(result.inconsistencies_fixed), 1)
|
|
self.assertEqual(len(saved_configs), 1)
|
|
self.assertIn("plugin1", saved_configs[0])
|
|
self.assertEqual(saved_configs[0]["plugin1"]["enabled"], False)
|
|
|
|
def test_auto_fix_enabled_state_mismatch(self):
|
|
"""Test auto-fix of enabled state mismatch."""
|
|
# Setup: Config says enabled=True, state manager says enabled=False
|
|
self.config_manager.load_config.return_value = {
|
|
"plugin1": {"enabled": True}
|
|
}
|
|
|
|
self.state_manager.get_all_states.return_value = {
|
|
"plugin1": Mock(
|
|
enabled=False,
|
|
status=PluginStateStatus.DISABLED,
|
|
version="1.0.0"
|
|
)
|
|
}
|
|
|
|
self.plugin_manager.plugin_manifests = {"plugin1": {}}
|
|
self.plugin_manager.plugins = {}
|
|
|
|
# Create plugin directory
|
|
plugin_dir = self.plugins_dir / "plugin1"
|
|
plugin_dir.mkdir()
|
|
manifest_path = plugin_dir / "manifest.json"
|
|
with open(manifest_path, 'w') as f:
|
|
json.dump({"version": "1.0.0", "name": "Plugin 1"}, f)
|
|
|
|
# Run reconciliation
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
# config.json is the source of truth for enabled state. The fix syncs
|
|
# the state manager to match config (config says True → state set True),
|
|
# rather than overwriting the config with the stale state value.
|
|
self.assertEqual(len(result.inconsistencies_fixed), 1)
|
|
self.state_manager.set_plugin_enabled.assert_called_once_with("plugin1", True)
|
|
|
|
def test_multiple_inconsistencies(self):
|
|
"""Test reconciliation with multiple inconsistencies."""
|
|
# Setup: Multiple plugins with different issues
|
|
self.config_manager.load_config.return_value = {
|
|
"plugin1": {"enabled": True}, # Exists in config but not on disk
|
|
# plugin2 exists on disk but not in config
|
|
}
|
|
|
|
self.state_manager.get_all_states.return_value = {
|
|
"plugin1": Mock(
|
|
enabled=True,
|
|
status=PluginStateStatus.ENABLED,
|
|
version="1.0.0"
|
|
)
|
|
}
|
|
|
|
self.plugin_manager.plugin_manifests = {}
|
|
self.plugin_manager.plugins = {}
|
|
|
|
# Create plugin2 directory (exists on disk but not in config)
|
|
plugin2_dir = self.plugins_dir / "plugin2"
|
|
plugin2_dir.mkdir()
|
|
manifest_path = plugin2_dir / "manifest.json"
|
|
with open(manifest_path, 'w') as f:
|
|
json.dump({"version": "1.0.0", "name": "Plugin 2"}, f)
|
|
|
|
# Run reconciliation
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
# Verify multiple inconsistencies found
|
|
self.assertGreaterEqual(len(result.inconsistencies_found), 2)
|
|
|
|
# Check types
|
|
inconsistency_types = [inc.inconsistency_type for inc in result.inconsistencies_found]
|
|
self.assertIn(InconsistencyType.PLUGIN_MISSING_ON_DISK, inconsistency_types)
|
|
self.assertIn(InconsistencyType.PLUGIN_MISSING_IN_CONFIG, inconsistency_types)
|
|
|
|
def test_reconciliation_with_exception(self):
|
|
"""Test reconciliation handles exceptions gracefully."""
|
|
# Setup: State manager raises exception when getting states
|
|
self.config_manager.load_config.return_value = {}
|
|
self.state_manager.get_all_states.side_effect = Exception("State manager error")
|
|
|
|
# Run reconciliation
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
# Verify error is handled - reconciliation may still succeed if other sources work
|
|
self.assertIsInstance(result, ReconciliationResult)
|
|
# Note: Reconciliation may still succeed if other sources provide valid state
|
|
|
|
def test_fix_failure_handling(self):
|
|
"""Test that fix failures are handled correctly."""
|
|
# Setup: Plugin missing in config, but save fails
|
|
self.config_manager.load_config.return_value = {}
|
|
|
|
self.state_manager.get_all_states.return_value = {}
|
|
|
|
self.plugin_manager.plugin_manifests = {}
|
|
self.plugin_manager.plugins = {}
|
|
|
|
# Create plugin directory
|
|
plugin_dir = self.plugins_dir / "plugin1"
|
|
plugin_dir.mkdir()
|
|
manifest_path = plugin_dir / "manifest.json"
|
|
with open(manifest_path, 'w') as f:
|
|
json.dump({"version": "1.0.0", "name": "Plugin 1"}, f)
|
|
|
|
# Mock save_config to raise exception
|
|
self.config_manager.save_config.side_effect = Exception("Save failed")
|
|
|
|
# Run reconciliation
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
# Verify inconsistency detected but not fixed
|
|
self.assertEqual(len(result.inconsistencies_found), 1)
|
|
self.assertEqual(len(result.inconsistencies_fixed), 0)
|
|
self.assertEqual(len(result.inconsistencies_manual), 1)
|
|
|
|
def test_get_config_state_handles_exception(self):
|
|
"""Test that _get_config_state handles exceptions."""
|
|
# Setup: Config manager raises exception
|
|
self.config_manager.load_config.side_effect = Exception("Config error")
|
|
|
|
# Call method directly
|
|
state = self.reconciler._get_config_state()
|
|
|
|
# Verify empty state returned
|
|
self.assertEqual(state, {})
|
|
|
|
def test_get_disk_state_handles_exception(self):
|
|
"""Test that _get_disk_state handles exceptions."""
|
|
# Setup: Make plugins_dir inaccessible
|
|
with patch.object(self.reconciler, 'plugins_dir', create=True) as mock_dir:
|
|
mock_dir.exists.side_effect = Exception("Disk error")
|
|
mock_dir.iterdir.side_effect = Exception("Disk error")
|
|
|
|
# Call method directly
|
|
state = self.reconciler._get_disk_state()
|
|
|
|
# Verify empty state returned
|
|
self.assertEqual(state, {})
|
|
|
|
|
|
class TestStateReconciliationUnrecoverable(unittest.TestCase):
|
|
"""Tests for the unrecoverable-plugin cache and force reconcile.
|
|
|
|
Regression coverage for the infinite reinstall loop where a config
|
|
entry referenced a plugin not present in the registry (e.g. legacy
|
|
'github' / 'youtube' entries). The reconciler used to retry the
|
|
install on every HTTP request; it now caches the failure for the
|
|
process lifetime and only retries on an explicit ``force=True``
|
|
reconcile call.
|
|
"""
|
|
|
|
def setUp(self):
|
|
self.temp_dir = Path(tempfile.mkdtemp())
|
|
self.plugins_dir = self.temp_dir / "plugins"
|
|
self.plugins_dir.mkdir()
|
|
|
|
self.state_manager = Mock(spec=PluginStateManager)
|
|
self.state_manager.get_all_states.return_value = {}
|
|
self.config_manager = Mock()
|
|
self.config_manager.load_config.return_value = {
|
|
"ghost": {"enabled": True}
|
|
}
|
|
self.plugin_manager = Mock()
|
|
self.plugin_manager.plugin_manifests = {}
|
|
self.plugin_manager.plugins = {}
|
|
|
|
# Store manager with an empty registry — install_plugin always fails
|
|
self.store_manager = Mock()
|
|
self.store_manager.fetch_registry.return_value = {"plugins": []}
|
|
self.store_manager.install_plugin.return_value = False
|
|
self.store_manager.was_recently_uninstalled.return_value = False
|
|
|
|
self.reconciler = StateReconciliation(
|
|
state_manager=self.state_manager,
|
|
config_manager=self.config_manager,
|
|
plugin_manager=self.plugin_manager,
|
|
plugins_dir=self.plugins_dir,
|
|
store_manager=self.store_manager,
|
|
)
|
|
|
|
def tearDown(self):
|
|
shutil.rmtree(self.temp_dir)
|
|
|
|
def test_not_in_registry_marks_unrecoverable_without_install(self):
|
|
"""If the plugin isn't in the registry at all, skip install_plugin."""
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
# One inconsistency, unfixable, no install attempt made.
|
|
self.assertEqual(len(result.inconsistencies_found), 1)
|
|
self.assertEqual(len(result.inconsistencies_fixed), 0)
|
|
self.store_manager.install_plugin.assert_not_called()
|
|
self.assertIn("ghost", self.reconciler._unrecoverable_missing_on_disk)
|
|
|
|
def test_subsequent_reconcile_does_not_retry(self):
|
|
"""Second reconcile pass must not touch install_plugin or fetch_registry again."""
|
|
self.reconciler.reconcile_state()
|
|
self.store_manager.fetch_registry.reset_mock()
|
|
self.store_manager.install_plugin.reset_mock()
|
|
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
# Still one inconsistency, still no install attempt, no new registry fetch
|
|
self.assertEqual(len(result.inconsistencies_found), 1)
|
|
inc = result.inconsistencies_found[0]
|
|
self.assertFalse(inc.can_auto_fix)
|
|
self.assertEqual(inc.fix_action, FixAction.MANUAL_FIX_REQUIRED)
|
|
self.store_manager.install_plugin.assert_not_called()
|
|
self.store_manager.fetch_registry.assert_not_called()
|
|
|
|
def test_force_reconcile_clears_unrecoverable_cache(self):
|
|
"""force=True must re-attempt previously-failed plugins."""
|
|
self.reconciler.reconcile_state()
|
|
self.assertIn("ghost", self.reconciler._unrecoverable_missing_on_disk)
|
|
|
|
# Now pretend the registry gained the plugin so the pre-check passes
|
|
# and install_plugin is actually invoked.
|
|
self.store_manager.fetch_registry.return_value = {
|
|
"plugins": [{"id": "ghost"}]
|
|
}
|
|
self.store_manager.install_plugin.return_value = True
|
|
self.store_manager.install_plugin.reset_mock()
|
|
|
|
# Config still references ghost; disk still missing it — the
|
|
# reconciler should re-attempt install now that force=True cleared
|
|
# the cache. Use assert_called_once_with so a future regression
|
|
# that accidentally triggers a second install attempt on force=True
|
|
# is caught.
|
|
result = self.reconciler.reconcile_state(force=True)
|
|
|
|
self.store_manager.install_plugin.assert_called_once_with("ghost")
|
|
|
|
def test_registry_unreachable_does_not_mark_unrecoverable(self):
|
|
"""Transient registry failures should not poison the cache."""
|
|
self.store_manager.fetch_registry.side_effect = Exception("network down")
|
|
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
self.assertEqual(len(result.inconsistencies_found), 1)
|
|
self.assertNotIn("ghost", self.reconciler._unrecoverable_missing_on_disk)
|
|
self.store_manager.install_plugin.assert_not_called()
|
|
|
|
def test_recently_uninstalled_skips_auto_repair(self):
|
|
"""A freshly-uninstalled plugin must not be resurrected by the reconciler."""
|
|
self.store_manager.was_recently_uninstalled.return_value = True
|
|
self.store_manager.fetch_registry.return_value = {
|
|
"plugins": [{"id": "ghost"}]
|
|
}
|
|
|
|
result = self.reconciler.reconcile_state()
|
|
|
|
self.assertEqual(len(result.inconsistencies_found), 1)
|
|
inc = result.inconsistencies_found[0]
|
|
self.assertFalse(inc.can_auto_fix)
|
|
self.assertEqual(inc.fix_action, FixAction.MANUAL_FIX_REQUIRED)
|
|
self.store_manager.install_plugin.assert_not_called()
|
|
|
|
def test_real_store_manager_empty_registry_on_network_failure(self):
|
|
"""Regression: using the REAL PluginStoreManager (not a Mock), verify
|
|
the reconciler does NOT poison the unrecoverable cache when
|
|
``fetch_registry`` fails with no stale cache available.
|
|
|
|
Previously, the default stale-cache fallback in ``fetch_registry``
|
|
silently returned ``{"plugins": []}`` on network failure with no
|
|
cache. The reconciler's ``_auto_repair_missing_plugin`` saw "no
|
|
candidates in registry" and marked everything unrecoverable — a
|
|
regression that would bite every user doing a fresh boot on flaky
|
|
WiFi. The fix is ``fetch_registry(raise_on_failure=True)`` in
|
|
``_auto_repair_missing_plugin`` so the reconciler can tell a real
|
|
registry miss from a network error.
|
|
"""
|
|
from src.plugin_system.store_manager import PluginStoreManager
|
|
import requests as real_requests
|
|
|
|
real_store = PluginStoreManager(plugins_dir=str(self.plugins_dir))
|
|
real_store.registry_cache = None # fresh boot, no cache
|
|
real_store.registry_cache_time = None
|
|
|
|
# Stub the underlying HTTP so no real network call is made but the
|
|
# real fetch_registry code path runs.
|
|
real_store._http_get_with_retries = Mock(
|
|
side_effect=real_requests.ConnectionError("wifi down")
|
|
)
|
|
|
|
reconciler = StateReconciliation(
|
|
state_manager=self.state_manager,
|
|
config_manager=self.config_manager,
|
|
plugin_manager=self.plugin_manager,
|
|
plugins_dir=self.plugins_dir,
|
|
store_manager=real_store,
|
|
)
|
|
|
|
result = reconciler.reconcile_state()
|
|
|
|
# One inconsistency (ghost is in config, not on disk), but
|
|
# because the registry lookup failed transiently, we must NOT
|
|
# have marked it unrecoverable — a later reconcile (after the
|
|
# network comes back) can still auto-repair.
|
|
self.assertEqual(len(result.inconsistencies_found), 1)
|
|
self.assertNotIn("ghost", reconciler._unrecoverable_missing_on_disk)
|
|
|
|
|
|
if __name__ == '__main__':
|
|
unittest.main()
|
|
|