Files
meshcore-stats/tests/scripts/test_collect_repeater.py
Jorijn Schrijvershof ca13e31aae test: stabilize suite and broaden integration coverage (#32)
* tests: cache integration/report fixtures to speed up tests

* fix: speed up yearly aggregation and refresh timings report

* chore: remove the report

* fix: unrecognized named-value: 'runner'. Located at position 1 within expression: runner.temp

* fix: ruff linting error

* test: strengthen assertions and stabilize tests

* test(integration): expand rendered chart metrics
2026-01-08 21:20:34 +01:00

881 lines
32 KiB
Python

"""Tests for collect_repeater.py script entry point.
These tests verify the actual script behavior, including:
- Finding repeater contact by name or key prefix
- Circuit breaker integration
- Exit codes for monitoring
- Database writes
"""
import inspect
from unittest.mock import AsyncMock, MagicMock, patch
import pytest
from tests.scripts.conftest import load_script_module
def load_collect_repeater():
"""Load collect_repeater.py as a module."""
return load_script_module("collect_repeater.py")
class TestCollectRepeaterImport:
"""Verify script can be imported without errors."""
def test_imports_successfully(self, configured_env):
"""Script should import without errors."""
module = load_collect_repeater()
assert hasattr(module, "main")
assert hasattr(module, "collect_repeater")
assert hasattr(module, "find_repeater_contact")
assert hasattr(module, "query_repeater_with_retry")
assert callable(module.main)
def test_collect_repeater_is_async(self, configured_env):
"""collect_repeater() should be an async function."""
module = load_collect_repeater()
assert inspect.iscoroutinefunction(module.collect_repeater)
def test_find_repeater_contact_is_async(self, configured_env):
"""find_repeater_contact() should be an async function."""
module = load_collect_repeater()
assert inspect.iscoroutinefunction(module.find_repeater_contact)
class TestFindRepeaterContact:
"""Test the find_repeater_contact function."""
@pytest.mark.asyncio
async def test_finds_contact_by_name(self, configured_env, monkeypatch):
"""Should find repeater by advertised name."""
monkeypatch.setenv("REPEATER_NAME", "MyRepeater")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mc = MagicMock()
mc.commands = MagicMock()
mc.contacts = {"abc123": {"adv_name": "MyRepeater", "public_key": "abc123"}}
with (
patch.object(module, "run_command") as mock_run,
patch.object(module, "get_contact_by_name") as mock_get,
):
mock_run.return_value = (True, "CONTACTS", mc.contacts, None)
mock_get.return_value = mc.contacts["abc123"]
contact = await module.find_repeater_contact(mc)
assert contact is not None
assert contact["adv_name"] == "MyRepeater"
mock_get.assert_called_once_with(mc, "MyRepeater")
@pytest.mark.asyncio
async def test_finds_contact_by_key_prefix(self, configured_env, monkeypatch):
"""Should find repeater by public key prefix when name not set."""
monkeypatch.setenv("REPEATER_KEY_PREFIX", "abc123")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mc = MagicMock()
mc.commands = MagicMock()
mc.contacts = {"abc123def456": {"adv_name": "SomeNode", "public_key": "abc123def456"}}
with (
patch.object(module, "run_command") as mock_run,
patch.object(module, "get_contact_by_name", return_value=None),
patch.object(module, "get_contact_by_key_prefix") as mock_get,
):
mock_run.return_value = (True, "CONTACTS", mc.contacts, None)
mock_get.return_value = mc.contacts["abc123def456"]
contact = await module.find_repeater_contact(mc)
assert contact is not None
assert contact["public_key"] == "abc123def456"
mock_get.assert_called_once()
@pytest.mark.asyncio
async def test_fallback_to_manual_name_search(self, configured_env, monkeypatch):
"""Should fallback to manual name search in payload dict."""
monkeypatch.setenv("REPEATER_NAME", "ManualFind")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mc = MagicMock()
mc.commands = MagicMock()
contacts_dict = {"xyz789": {"adv_name": "ManualFind", "public_key": "xyz789"}}
with (
patch.object(module, "run_command") as mock_run,
patch.object(module, "get_contact_by_name", return_value=None),
):
mock_run.return_value = (True, "CONTACTS", contacts_dict, None)
# get_contact_by_name returns None, forcing manual search
contact = await module.find_repeater_contact(mc)
assert contact is not None
assert contact["adv_name"] == "ManualFind"
@pytest.mark.asyncio
async def test_case_insensitive_name_match(self, configured_env, monkeypatch):
"""Name search should be case-insensitive."""
monkeypatch.setenv("REPEATER_NAME", "myrepeater") # lowercase
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mc = MagicMock()
mc.commands = MagicMock()
contacts_dict = {"key1": {"adv_name": "MyRepeater", "public_key": "key1"}} # Mixed case
with (
patch.object(module, "run_command") as mock_run,
patch.object(module, "get_contact_by_name", return_value=None),
):
mock_run.return_value = (True, "CONTACTS", contacts_dict, None)
contact = await module.find_repeater_contact(mc)
assert contact is not None
assert contact["adv_name"] == "MyRepeater"
@pytest.mark.asyncio
async def test_returns_none_when_not_found(self, configured_env, monkeypatch):
"""Should return None when repeater not in contacts."""
monkeypatch.setenv("REPEATER_NAME", "NonExistent")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mc = MagicMock()
mc.commands = MagicMock()
with (
patch.object(module, "run_command") as mock_run,
patch.object(module, "get_contact_by_name", return_value=None),
):
mock_run.return_value = (True, "CONTACTS", {}, None)
contact = await module.find_repeater_contact(mc)
assert contact is None
@pytest.mark.asyncio
async def test_returns_none_when_get_contacts_fails(self, configured_env, monkeypatch):
"""Should return None when get_contacts command fails."""
monkeypatch.setenv("REPEATER_NAME", "AnyName")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mc = MagicMock()
mc.commands = MagicMock()
with patch.object(module, "run_command") as mock_run:
mock_run.return_value = (False, None, None, "Connection failed")
contact = await module.find_repeater_contact(mc)
assert contact is None
class TestCircuitBreakerIntegration:
"""Test circuit breaker integration in collect_repeater."""
@pytest.mark.asyncio
async def test_skips_collection_when_circuit_open(self, configured_env):
"""Should return 0 and skip collection when circuit breaker is open."""
module = load_collect_repeater()
# Create mock circuit breaker that is open
mock_cb = MagicMock()
mock_cb.is_open.return_value = True
mock_cb.cooldown_remaining.return_value = 1800
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock") as mock_connect,
):
exit_code = await module.collect_repeater()
# Should return 0 (not an error, just skipped)
assert exit_code == 0
# Should not have tried to connect
mock_cb.is_open.assert_called_once()
mock_connect.assert_not_called()
@pytest.mark.asyncio
async def test_records_success_on_successful_status(
self, configured_env, monkeypatch, async_context_manager_factory
):
"""Should record success when status query succeeds."""
monkeypatch.setenv("REPEATER_NAME", "TestRepeater")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
mc = MagicMock()
mc.commands = MagicMock()
ctx_mock = async_context_manager_factory(mc)
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
patch.object(module, "run_command") as mock_run,
patch.object(module, "find_repeater_contact") as mock_find,
patch.object(module, "query_repeater_with_retry") as mock_query,
patch.object(module, "insert_metrics", return_value=2),
):
mock_run.return_value = (True, "OK", {}, None)
mock_find.return_value = {"adv_name": "TestRepeater"}
mock_query.return_value = (True, {"bat": 3850, "uptime": 86400}, None)
await module.collect_repeater()
mock_cb.record_success.assert_called_once()
mock_cb.record_failure.assert_not_called()
@pytest.mark.asyncio
async def test_records_failure_on_status_timeout(
self, configured_env, monkeypatch, async_context_manager_factory
):
"""Should record failure when status query times out."""
monkeypatch.setenv("REPEATER_NAME", "TestRepeater")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
mc = MagicMock()
mc.commands = MagicMock()
ctx_mock = async_context_manager_factory(mc)
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
patch.object(module, "run_command") as mock_run,
patch.object(module, "find_repeater_contact") as mock_find,
patch.object(module, "query_repeater_with_retry") as mock_query,
):
mock_run.return_value = (True, "OK", {}, None)
mock_find.return_value = {"adv_name": "TestRepeater"}
mock_query.return_value = (False, None, "Timeout")
exit_code = await module.collect_repeater()
mock_cb.record_failure.assert_called_once()
mock_cb.record_success.assert_not_called()
assert exit_code == 1
class TestCollectRepeaterExitCodes:
"""Test exit code behavior - critical for monitoring."""
@pytest.mark.asyncio
async def test_returns_zero_on_successful_collection(
self, configured_env, monkeypatch, async_context_manager_factory
):
"""Successful collection should return exit code 0."""
monkeypatch.setenv("REPEATER_NAME", "TestRepeater")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
mc = MagicMock()
mc.commands = MagicMock()
ctx_mock = async_context_manager_factory(mc)
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
patch.object(module, "run_command") as mock_run,
patch.object(module, "find_repeater_contact") as mock_find,
patch.object(module, "query_repeater_with_retry") as mock_query,
patch.object(module, "insert_metrics") as mock_insert,
):
mock_run.return_value = (True, "OK", {}, None)
mock_find.return_value = {"adv_name": "TestRepeater"}
mock_query.return_value = (
True,
{"bat": 3850, "uptime": 86400, "nb_recv": 100},
None,
)
exit_code = await module.collect_repeater()
assert exit_code == 0
mock_insert.assert_called_once()
insert_kwargs = mock_insert.call_args.kwargs
assert insert_kwargs["role"] == "repeater"
assert insert_kwargs["metrics"]["bat"] == 3850
assert insert_kwargs["metrics"]["nb_recv"] == 100
@pytest.mark.asyncio
async def test_returns_one_on_connection_failure(
self, configured_env, async_context_manager_factory
):
"""Failed connection should return exit code 1."""
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
ctx_mock = async_context_manager_factory(None) # Connection returns None
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
):
exit_code = await module.collect_repeater()
assert exit_code == 1
@pytest.mark.asyncio
async def test_returns_one_when_repeater_not_found(
self, configured_env, monkeypatch, async_context_manager_factory
):
"""Should return 1 when repeater contact not found."""
monkeypatch.setenv("REPEATER_NAME", "NonExistent")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
mc = MagicMock()
mc.commands = MagicMock()
ctx_mock = async_context_manager_factory(mc)
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
patch.object(module, "run_command") as mock_run,
patch.object(module, "find_repeater_contact") as mock_find,
):
mock_run.return_value = (True, "OK", {}, None)
mock_find.return_value = None # Not found
exit_code = await module.collect_repeater()
assert exit_code == 1
class TestQueryRepeaterWithRetry:
"""Test the retry wrapper for repeater queries."""
@pytest.mark.asyncio
async def test_returns_success_on_first_try(self, configured_env):
"""Should return success when command succeeds immediately."""
module = load_collect_repeater()
mc = MagicMock()
contact = {"adv_name": "Test"}
async def successful_command():
return {"bat": 3850}
with patch.object(module, "with_retries") as mock_retries:
mock_retries.return_value = (True, {"bat": 3850}, None)
success, payload, err = await module.query_repeater_with_retry(
mc, contact, "test_cmd", successful_command
)
assert success is True
assert payload == {"bat": 3850}
assert err is None
@pytest.mark.asyncio
async def test_returns_failure_after_retries_exhausted(self, configured_env):
"""Should return failure when all retries fail."""
module = load_collect_repeater()
mc = MagicMock()
contact = {"adv_name": "Test"}
async def failing_command():
raise Exception("Timeout")
with patch.object(module, "with_retries") as mock_retries:
mock_retries.return_value = (False, None, Exception("Timeout"))
success, payload, err = await module.query_repeater_with_retry(
mc, contact, "test_cmd", failing_command
)
assert success is False
assert payload is None
assert "Timeout" in err
class TestMainEntryPoint:
"""Test the main() entry point behavior."""
def test_main_calls_init_db(self, configured_env):
"""main() should initialize database before collection."""
module = load_collect_repeater()
with (
patch.object(module, "init_db") as mock_init,
patch.object(module, "collect_repeater", new=MagicMock(return_value=0)),
patch.object(module, "asyncio") as mock_asyncio,
patch.object(module, "sys"),
):
# Patch collect_repeater to return a non-coroutine to avoid unawaited coroutine warning
mock_asyncio.run.return_value = 0
module.main()
mock_init.assert_called_once()
def test_main_exits_with_collection_result(self, configured_env):
"""main() should exit with the collection exit code."""
module = load_collect_repeater()
with (
patch.object(module, "init_db"),
patch.object(module, "collect_repeater", new=MagicMock(return_value=1)),
patch.object(module, "asyncio") as mock_asyncio,
patch.object(module, "sys") as mock_sys,
):
# Patch collect_repeater to return a non-coroutine to avoid unawaited coroutine warning
mock_asyncio.run.return_value = 1 # Collection failed
module.main()
mock_sys.exit.assert_called_once_with(1)
class TestDatabaseIntegration:
"""Test that collection actually writes to database."""
@pytest.mark.asyncio
async def test_writes_metrics_to_database(
self, configured_env, initialized_db, monkeypatch, async_context_manager_factory
):
"""Collection should write metrics to database."""
from meshmon.db import get_latest_metrics
monkeypatch.setenv("REPEATER_NAME", "TestRepeater")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
mc = MagicMock()
mc.commands = MagicMock()
ctx_mock = async_context_manager_factory(mc)
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
patch.object(module, "run_command") as mock_run,
patch.object(module, "find_repeater_contact") as mock_find,
patch.object(module, "query_repeater_with_retry") as mock_query,
):
mock_run.return_value = (True, "OK", {}, None)
mock_find.return_value = {"adv_name": "TestRepeater"}
mock_query.return_value = (
True,
{"bat": 3777, "uptime": 99999, "nb_recv": 1234, "nb_sent": 567},
None,
)
exit_code = await module.collect_repeater()
assert exit_code == 0
# Verify data was written to database
latest = get_latest_metrics("repeater")
assert latest is not None
assert latest["bat"] == 3777
assert latest["nb_recv"] == 1234
class TestFindRepeaterContactEdgeCases:
"""Test edge cases in find_repeater_contact."""
@pytest.mark.asyncio
async def test_finds_contact_in_payload_dict(self, configured_env, monkeypatch):
"""Should find contact in payload dict when mc.contacts is empty."""
monkeypatch.setenv("REPEATER_NAME", "PayloadRepeater")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mc = MagicMock()
mc.commands = MagicMock()
mc.contacts = {} # Empty contacts attribute
payload_dict = {"pk123": {"adv_name": "PayloadRepeater", "public_key": "pk123"}}
with (
patch.object(module, "run_command") as mock_run,
patch.object(module, "get_contact_by_name", return_value=None),
):
# Return contacts in payload
mock_run.return_value = (True, "CONTACTS", payload_dict, None)
# get_contact_by_name returns None, forcing manual search in payload
contact = await module.find_repeater_contact(mc)
assert contact is not None
assert contact["adv_name"] == "PayloadRepeater"
@pytest.mark.asyncio
async def test_finds_contact_by_key_prefix_manual_search(self, configured_env, monkeypatch):
"""Should find contact by key prefix via manual search in payload."""
monkeypatch.setenv("REPEATER_KEY_PREFIX", "abc")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mc = MagicMock()
mc.commands = MagicMock()
contacts_dict = {"abc123xyz": {"adv_name": "KeyPrefixNode"}}
with (
patch.object(module, "run_command") as mock_run,
patch.object(module, "get_contact_by_name", return_value=None),
patch.object(module, "get_contact_by_key_prefix", return_value=None),
):
mock_run.return_value = (True, "CONTACTS", contacts_dict, None)
# Both helper functions return None, forcing manual search
contact = await module.find_repeater_contact(mc)
assert contact is not None
assert contact["adv_name"] == "KeyPrefixNode"
@pytest.mark.asyncio
async def test_prints_available_contacts_when_not_found(self, configured_env, monkeypatch):
"""Should print available contacts when repeater not found."""
monkeypatch.setenv("REPEATER_NAME", "NonExistent")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mc = MagicMock()
mc.commands = MagicMock()
contacts_dict = {
"key1": {"adv_name": "Node1", "name": "alt1"},
"key2": {"adv_name": "Node2"},
"key3": {}, # No name fields
}
with (
patch.object(module, "run_command") as mock_run,
patch.object(module, "get_contact_by_name", return_value=None),
patch.object(module, "log") as mock_log,
):
mock_run.return_value = (True, "CONTACTS", contacts_dict, None)
contact = await module.find_repeater_contact(mc)
assert contact is None
# Should have logged available contacts
mock_log.info.assert_called()
class TestLoginFunctionality:
"""Test optional login functionality."""
@pytest.mark.asyncio
async def test_attempts_login_when_password_set(
self, configured_env, monkeypatch, async_context_manager_factory
):
"""Should attempt login when REPEATER_PASSWORD is set."""
monkeypatch.setenv("REPEATER_NAME", "TestRepeater")
monkeypatch.setenv("REPEATER_PASSWORD", "secret123")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
mc = MagicMock()
mc.commands = MagicMock()
mc.commands.send_login = MagicMock()
ctx_mock = async_context_manager_factory(mc)
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
patch.object(module, "run_command") as mock_run,
patch.object(module, "find_repeater_contact") as mock_find,
patch.object(module, "extract_contact_info") as mock_extract,
patch.object(module, "query_repeater_with_retry") as mock_query,
patch.object(module, "insert_metrics", return_value=1),
):
# Return success for all commands
mock_run.return_value = (True, "OK", {}, None)
mock_find.return_value = {"adv_name": "TestRepeater"}
mock_extract.return_value = {"adv_name": "TestRepeater"}
mock_query.return_value = (True, {"bat": 3850}, None)
await module.collect_repeater()
# Verify login was attempted (run_command called with send_login)
login_calls = [c for c in mock_run.call_args_list if c[0][2] == "send_login"]
assert len(login_calls) == 1
@pytest.mark.asyncio
async def test_handles_login_exception(
self, configured_env, monkeypatch, async_context_manager_factory
):
"""Should handle exception during login gracefully."""
monkeypatch.setenv("REPEATER_NAME", "TestRepeater")
monkeypatch.setenv("REPEATER_PASSWORD", "secret123")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
mc = MagicMock()
mc.commands = MagicMock()
mc.commands.send_login = MagicMock(side_effect=Exception("Login not supported"))
ctx_mock = async_context_manager_factory(mc)
call_count = 0
async def mock_run_command(mc, coro, name):
nonlocal call_count
call_count += 1
if name == "send_login":
raise Exception("Login not supported")
return (True, "OK", {}, None)
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
patch.object(module, "run_command", side_effect=mock_run_command),
patch.object(module, "find_repeater_contact") as mock_find,
patch.object(module, "extract_contact_info") as mock_extract,
patch.object(module, "query_repeater_with_retry") as mock_query,
patch.object(module, "insert_metrics", return_value=1),
):
mock_find.return_value = {"adv_name": "TestRepeater"}
mock_extract.return_value = {"adv_name": "TestRepeater"}
mock_query.return_value = (True, {"bat": 3850}, None)
# Should not raise - login failure should be handled
exit_code = await module.collect_repeater()
assert exit_code == 0
class TestTelemetryCollection:
"""Test telemetry collection when enabled."""
@pytest.mark.asyncio
async def test_collects_telemetry_when_enabled(
self, configured_env, monkeypatch, initialized_db, async_context_manager_factory
):
"""Should collect telemetry when TELEMETRY_ENABLED=1."""
monkeypatch.setenv("REPEATER_NAME", "TestRepeater")
monkeypatch.setenv("TELEMETRY_ENABLED", "1")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
mc = MagicMock()
mc.commands = MagicMock()
ctx_mock = async_context_manager_factory(mc)
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
patch.object(module, "run_command") as mock_run,
patch.object(module, "find_repeater_contact") as mock_find,
patch.object(module, "extract_contact_info") as mock_extract,
patch.object(module, "query_repeater_with_retry") as mock_query,
patch.object(
module,
"with_retries",
new=AsyncMock(return_value=(True, {"lpp": b"\x00\x67\x01\x00"}, None)),
),
patch.object(module, "extract_lpp_from_payload") as mock_lpp,
patch.object(module, "extract_telemetry_metrics") as mock_telem,
):
mock_run.return_value = (True, "OK", {}, None)
mock_find.return_value = {"adv_name": "TestRepeater"}
mock_extract.return_value = {"adv_name": "TestRepeater"}
mock_query.return_value = (True, {"bat": 3850}, None)
mock_lpp.return_value = {"temperature": [(0, 25.5)]}
mock_telem.return_value = {"telemetry.temperature.0": 25.5}
exit_code = await module.collect_repeater()
assert exit_code == 0
# Verify telemetry was processed
mock_lpp.assert_called_once()
mock_telem.assert_called_once()
@pytest.mark.asyncio
async def test_handles_telemetry_failure_gracefully(
self, configured_env, monkeypatch, async_context_manager_factory
):
"""Should continue when telemetry collection fails."""
monkeypatch.setenv("REPEATER_NAME", "TestRepeater")
monkeypatch.setenv("TELEMETRY_ENABLED", "1")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
mc = MagicMock()
mc.commands = MagicMock()
ctx_mock = async_context_manager_factory(mc)
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
patch.object(module, "run_command") as mock_run,
patch.object(module, "find_repeater_contact") as mock_find,
patch.object(module, "extract_contact_info") as mock_extract,
patch.object(module, "query_repeater_with_retry") as mock_query,
patch.object(module, "insert_metrics", return_value=1),
patch.object(
module,
"with_retries",
new=AsyncMock(return_value=(False, None, Exception("Timeout"))),
),
):
mock_run.return_value = (True, "OK", {}, None)
mock_find.return_value = {"adv_name": "TestRepeater"}
mock_extract.return_value = {"adv_name": "TestRepeater"}
mock_query.return_value = (True, {"bat": 3850}, None)
# Should still succeed (status metrics were saved)
exit_code = await module.collect_repeater()
assert exit_code == 0
class TestDatabaseErrorHandling:
"""Test database error handling."""
@pytest.mark.asyncio
async def test_returns_one_on_status_db_error(
self, configured_env, monkeypatch, async_context_manager_factory
):
"""Should return 1 when status metrics DB write fails."""
monkeypatch.setenv("REPEATER_NAME", "TestRepeater")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
mc = MagicMock()
mc.commands = MagicMock()
ctx_mock = async_context_manager_factory(mc)
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
patch.object(module, "run_command") as mock_run,
patch.object(module, "find_repeater_contact") as mock_find,
patch.object(module, "extract_contact_info") as mock_extract,
patch.object(module, "query_repeater_with_retry") as mock_query,
patch.object(module, "insert_metrics", side_effect=Exception("DB error")),
):
mock_run.return_value = (True, "OK", {}, None)
mock_find.return_value = {"adv_name": "TestRepeater"}
mock_extract.return_value = {"adv_name": "TestRepeater"}
mock_query.return_value = (True, {"bat": 3850}, None)
exit_code = await module.collect_repeater()
assert exit_code == 1
class TestExceptionHandling:
"""Test general exception handling."""
@pytest.mark.asyncio
async def test_records_failure_on_exception(
self, configured_env, monkeypatch, async_context_manager_factory
):
"""Should record circuit breaker failure on unexpected exception."""
monkeypatch.setenv("REPEATER_NAME", "TestRepeater")
import meshmon.env
meshmon.env._config = None
module = load_collect_repeater()
mock_cb = MagicMock()
mock_cb.is_open.return_value = False
mc = MagicMock()
mc.commands = MagicMock()
ctx_mock = async_context_manager_factory(mc)
with (
patch.object(module, "get_repeater_circuit_breaker", return_value=mock_cb),
patch.object(module, "connect_with_lock", return_value=ctx_mock),
patch.object(module, "run_command") as mock_run,
patch.object(module, "find_repeater_contact") as mock_find,
patch.object(module, "extract_contact_info") as mock_extract,
):
mock_run.return_value = (True, "OK", {}, None)
mock_find.return_value = {"adv_name": "TestRepeater"}
mock_extract.side_effect = Exception("Unexpected error")
await module.collect_repeater()
# Circuit breaker should record failure
mock_cb.record_failure.assert_called_once()