|
| 1 | +"""Unit tests for the opt-in periodic interface-change monitor.""" |
| 2 | + |
| 3 | +from __future__ import annotations |
| 4 | + |
| 5 | +import asyncio |
| 6 | +from collections.abc import Callable |
| 7 | +from unittest.mock import Mock, patch |
| 8 | + |
| 9 | +import pytest |
| 10 | + |
| 11 | +from zeroconf._utils import interface_monitor as im |
| 12 | +from zeroconf._utils.interface_monitor import InterfaceMonitor |
| 13 | +from zeroconf.asyncio import AsyncZeroconf |
| 14 | + |
| 15 | + |
| 16 | +def _snapshot_cycler(snapshots: list[frozenset]) -> Callable[[], frozenset]: |
| 17 | + """Return the next snapshot each call, sticking on the last one.""" |
| 18 | + it = iter(snapshots) |
| 19 | + |
| 20 | + def _next() -> frozenset: |
| 21 | + try: |
| 22 | + return next(it) |
| 23 | + except StopIteration: |
| 24 | + return snapshots[-1] |
| 25 | + |
| 26 | + return _next |
| 27 | + |
| 28 | + |
| 29 | +def test_adapter_snapshot() -> None: |
| 30 | + adapter = Mock() |
| 31 | + adapter.index = 1 |
| 32 | + ip = Mock() |
| 33 | + ip.ip = "192.168.1.5" |
| 34 | + adapter.ips = [ip] |
| 35 | + with patch.object(im.ifaddr, "get_adapters", return_value=[adapter]): |
| 36 | + assert im._adapter_snapshot() == frozenset({(1, "192.168.1.5")}) |
| 37 | + |
| 38 | + |
| 39 | +@pytest.mark.asyncio |
| 40 | +async def test_monitor_rescans_on_change(aiozc_loopback: AsyncZeroconf) -> None: |
| 41 | + """A changed adapter snapshot triggers a rescan.""" |
| 42 | + zc = aiozc_loopback.zeroconf |
| 43 | + await zc.async_wait_for_start() |
| 44 | + updated = asyncio.Event() |
| 45 | + |
| 46 | + async def _fake_update() -> None: |
| 47 | + updated.set() |
| 48 | + |
| 49 | + with ( |
| 50 | + patch.object( |
| 51 | + im, "_adapter_snapshot", side_effect=_snapshot_cycler([frozenset({"a"}), frozenset({"b"})]) |
| 52 | + ), |
| 53 | + patch.object(zc, "async_update_interfaces", side_effect=_fake_update), |
| 54 | + ): |
| 55 | + await aiozc_loopback.async_start_interface_monitor(interval=0.001) |
| 56 | + await asyncio.wait_for(updated.wait(), timeout=1.0) |
| 57 | + await aiozc_loopback.async_stop_interface_monitor() |
| 58 | + |
| 59 | + |
| 60 | +@pytest.mark.asyncio |
| 61 | +async def test_monitor_no_rescan_when_unchanged(aiozc_loopback: AsyncZeroconf) -> None: |
| 62 | + """An unchanged snapshot does not trigger a rescan.""" |
| 63 | + zc = aiozc_loopback.zeroconf |
| 64 | + await zc.async_wait_for_start() |
| 65 | + with ( |
| 66 | + patch.object(im, "_adapter_snapshot", return_value=frozenset({"same"})), |
| 67 | + patch.object(zc, "async_update_interfaces") as mock_update, |
| 68 | + ): |
| 69 | + await aiozc_loopback.async_start_interface_monitor(interval=0.001) |
| 70 | + await asyncio.sleep(0.02) |
| 71 | + await aiozc_loopback.async_stop_interface_monitor() |
| 72 | + mock_update.assert_not_called() |
| 73 | + |
| 74 | + |
| 75 | +@pytest.mark.asyncio |
| 76 | +async def test_monitor_survives_rescan_error(aiozc_loopback: AsyncZeroconf) -> None: |
| 77 | + """A failed rescan is logged and the monitor keeps running.""" |
| 78 | + zc = aiozc_loopback.zeroconf |
| 79 | + await zc.async_wait_for_start() |
| 80 | + calls = [] |
| 81 | + |
| 82 | + async def _boom() -> None: |
| 83 | + calls.append(1) |
| 84 | + raise RuntimeError("boom") |
| 85 | + |
| 86 | + snapshots = [frozenset({"a"}), frozenset({"b"}), frozenset({"c"})] |
| 87 | + with ( |
| 88 | + patch.object(im, "_adapter_snapshot", side_effect=_snapshot_cycler(snapshots)), |
| 89 | + patch.object(zc, "async_update_interfaces", side_effect=_boom), |
| 90 | + ): |
| 91 | + await aiozc_loopback.async_start_interface_monitor(interval=0.001) |
| 92 | + await asyncio.sleep(0.05) |
| 93 | + await aiozc_loopback.async_stop_interface_monitor() |
| 94 | + assert len(calls) >= 2 |
| 95 | + |
| 96 | + |
| 97 | +@pytest.mark.asyncio |
| 98 | +async def test_start_interface_monitor_idempotent(aiozc_loopback: AsyncZeroconf) -> None: |
| 99 | + """Starting an already-running monitor is a no-op.""" |
| 100 | + zc = aiozc_loopback.zeroconf |
| 101 | + await zc.async_wait_for_start() |
| 102 | + with patch.object(im, "_adapter_snapshot", return_value=frozenset()): |
| 103 | + await aiozc_loopback.async_start_interface_monitor(interval=10) |
| 104 | + monitor = zc._interface_monitor |
| 105 | + assert monitor is not None |
| 106 | + task = monitor._task |
| 107 | + await aiozc_loopback.async_start_interface_monitor(interval=10) |
| 108 | + assert zc._interface_monitor is monitor |
| 109 | + assert monitor._task is task |
| 110 | + await aiozc_loopback.async_stop_interface_monitor() |
| 111 | + |
| 112 | + |
| 113 | +@pytest.mark.asyncio |
| 114 | +async def test_monitor_start_idempotent(aiozc_loopback: AsyncZeroconf) -> None: |
| 115 | + """InterfaceMonitor.start is a no-op when a task is already scheduled.""" |
| 116 | + zc = aiozc_loopback.zeroconf |
| 117 | + await zc.async_wait_for_start() |
| 118 | + with patch.object(im, "_adapter_snapshot", return_value=frozenset()): |
| 119 | + monitor = InterfaceMonitor(zc, interval=10) |
| 120 | + monitor.start() |
| 121 | + task = monitor._task |
| 122 | + monitor.start() |
| 123 | + assert monitor._task is task |
| 124 | + await monitor.async_stop() |
| 125 | + |
| 126 | + |
| 127 | +@pytest.mark.asyncio |
| 128 | +async def test_monitor_stop_without_start(aiozc_loopback: AsyncZeroconf) -> None: |
| 129 | + """Stopping a monitor that never started is a no-op.""" |
| 130 | + zc = aiozc_loopback.zeroconf |
| 131 | + await zc.async_wait_for_start() |
| 132 | + with patch.object(im, "_adapter_snapshot", return_value=frozenset()): |
| 133 | + monitor = InterfaceMonitor(zc) |
| 134 | + await monitor.async_stop() |
| 135 | + |
| 136 | + |
| 137 | +@pytest.mark.asyncio |
| 138 | +async def test_core_stop_interface_monitor_when_none(aiozc_loopback: AsyncZeroconf) -> None: |
| 139 | + """Stopping the monitor when none is running is a no-op.""" |
| 140 | + await aiozc_loopback.zeroconf.async_wait_for_start() |
| 141 | + await aiozc_loopback.async_stop_interface_monitor() |
| 142 | + |
| 143 | + |
| 144 | +@pytest.mark.asyncio |
| 145 | +async def test_monitor_stopped_on_close() -> None: |
| 146 | + """async_close stops a running interface monitor.""" |
| 147 | + aiozc = AsyncZeroconf(interfaces=["127.0.0.1"]) |
| 148 | + zc = aiozc.zeroconf |
| 149 | + await zc.async_wait_for_start() |
| 150 | + with patch.object(im, "_adapter_snapshot", return_value=frozenset()): |
| 151 | + await aiozc.async_start_interface_monitor(interval=10) |
| 152 | + assert zc._interface_monitor is not None |
| 153 | + await aiozc.async_close() |
| 154 | + assert zc._interface_monitor is None |
0 commit comments