mirror of
https://github.com/home-assistant/supervisor.git
synced 2025-12-26 21:47:15 +00:00
Improve DNS plug-in restart (#5999)
* Improve DNS plug-in restart Instead of simply go by PrimaryConnectioon change, use the DnsManager Configuration property. This property is ultimately used to write the DNS plug-in configuration, so it is really the relevant information we pass on to the plug-in. * Check for changes and restart DNS plugin * Check for changes in plug-in DNS Cache last local (NetworkManager) provided DNS servers. Check against this DNS server list when deciding when to restart the DNS plug-in. * Check connectivity unthrottled in certain situations * Fix pytest * Fix pytest * Improve test coverage for DNS plugins restart functionality * Apply suggestions from code review Co-authored-by: Mike Degatano <michael.degatano@gmail.com> * Debounce local DNS changes and event based connectivity checks * Remove connection check logic * Remove unthrottled connectivity check * Fix delayed call * Store restart task and cancel in case a restart is running * Improve DNS configuration change tests * Remove stale code * Improve DNS plug-in tests, less mocking * Cover multiple private functions at once Improve tests around notify_locals_changed() to cover multiple functions at once. --------- Co-authored-by: Mike Degatano <michael.degatano@gmail.com>
This commit is contained in:
@@ -66,6 +66,7 @@ from .dbus_service_mocks.base import DBusServiceMock
|
||||
from .dbus_service_mocks.network_connection_settings import (
|
||||
ConnectionSettings as ConnectionSettingsService,
|
||||
)
|
||||
from .dbus_service_mocks.network_dns_manager import DnsManager as DnsManagerService
|
||||
from .dbus_service_mocks.network_manager import NetworkManager as NetworkManagerService
|
||||
|
||||
# pylint: disable=redefined-outer-name, protected-access
|
||||
@@ -220,6 +221,14 @@ async def network_manager_service(
|
||||
yield network_manager_services["network_manager"]
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
async def dns_manager_service(
|
||||
network_manager_services: dict[str, DBusServiceMock | dict[str, DBusServiceMock]],
|
||||
) -> AsyncGenerator[DnsManagerService]:
|
||||
"""Return DNS Manager service mock."""
|
||||
yield network_manager_services["network_dns_manager"]
|
||||
|
||||
|
||||
@pytest.fixture(name="connection_settings_service")
|
||||
async def fixture_connection_settings_service(
|
||||
network_manager_services: dict[str, DBusServiceMock | dict[str, DBusServiceMock]],
|
||||
|
||||
@@ -2,8 +2,9 @@
|
||||
|
||||
# pylint: disable=protected-access
|
||||
import asyncio
|
||||
from unittest.mock import AsyncMock, PropertyMock, patch
|
||||
from unittest.mock import PropertyMock, patch
|
||||
|
||||
from dbus_fast import Variant
|
||||
import pytest
|
||||
|
||||
from supervisor.coresys import CoreSys
|
||||
@@ -87,23 +88,47 @@ async def test_connectivity_events(coresys: CoreSys, force: bool):
|
||||
)
|
||||
|
||||
|
||||
async def test_dns_restart_on_connection_change(
|
||||
coresys: CoreSys, network_manager_service: NetworkManagerService
|
||||
async def test_dns_configuration_change_triggers_notify_locals_changed(
|
||||
coresys: CoreSys, dns_manager_service
|
||||
):
|
||||
"""Test dns plugin is restarted when primary connection changes."""
|
||||
"""Test that DNS configuration changes trigger notify_locals_changed."""
|
||||
await coresys.host.network.load()
|
||||
with (
|
||||
patch.object(PluginDns, "restart") as restart,
|
||||
patch.object(
|
||||
PluginDns, "is_running", new_callable=AsyncMock, return_value=True
|
||||
),
|
||||
):
|
||||
network_manager_service.emit_properties_changed({"PrimaryConnection": "/"})
|
||||
await network_manager_service.ping()
|
||||
restart.assert_not_called()
|
||||
|
||||
network_manager_service.emit_properties_changed(
|
||||
{"PrimaryConnection": "/org/freedesktop/NetworkManager/ActiveConnection/2"}
|
||||
with patch.object(PluginDns, "notify_locals_changed") as notify_locals_changed:
|
||||
# Test that non-Configuration changes don't trigger notify_locals_changed
|
||||
dns_manager_service.emit_properties_changed({"Mode": "default"})
|
||||
await dns_manager_service.ping()
|
||||
notify_locals_changed.assert_not_called()
|
||||
|
||||
# Test that Configuration changes trigger notify_locals_changed
|
||||
configuration = [
|
||||
{
|
||||
"nameservers": Variant("as", ["192.168.2.2"]),
|
||||
"domains": Variant("as", ["lan"]),
|
||||
"interface": Variant("s", "eth0"),
|
||||
"priority": Variant("i", 100),
|
||||
"vpn": Variant("b", False),
|
||||
}
|
||||
]
|
||||
|
||||
dns_manager_service.emit_properties_changed({"Configuration": configuration})
|
||||
await dns_manager_service.ping()
|
||||
notify_locals_changed.assert_called_once()
|
||||
|
||||
notify_locals_changed.reset_mock()
|
||||
# Test that subsequent Configuration changes also trigger notify_locals_changed
|
||||
different_configuration = [
|
||||
{
|
||||
"nameservers": Variant("as", ["8.8.8.8"]),
|
||||
"domains": Variant("as", ["example.com"]),
|
||||
"interface": Variant("s", "wlan0"),
|
||||
"priority": Variant("i", 200),
|
||||
"vpn": Variant("b", True),
|
||||
}
|
||||
]
|
||||
|
||||
dns_manager_service.emit_properties_changed(
|
||||
{"Configuration": different_configuration}
|
||||
)
|
||||
await network_manager_service.ping()
|
||||
restart.assert_called_once()
|
||||
await dns_manager_service.ping()
|
||||
notify_locals_changed.assert_called_once()
|
||||
|
||||
@@ -35,6 +35,17 @@ async def fixture_write_json() -> Mock:
|
||||
yield write_json_file
|
||||
|
||||
|
||||
@pytest.fixture(name="mock_call_later")
|
||||
def fixture_mock_call_later(coresys: CoreSys):
|
||||
"""Mock sys_call_later with zero delay for testing."""
|
||||
|
||||
def mock_call_later(_delay, *args, **kwargs) -> asyncio.TimerHandle:
|
||||
"""Mock to remove delay."""
|
||||
return coresys.call_later(0, *args, **kwargs)
|
||||
|
||||
return mock_call_later
|
||||
|
||||
|
||||
async def test_config_write(
|
||||
coresys: CoreSys,
|
||||
docker_interface: tuple[AsyncMock, AsyncMock],
|
||||
@@ -98,6 +109,7 @@ async def test_reset(coresys: CoreSys):
|
||||
unlink.assert_called_once()
|
||||
write_hosts.assert_called_once()
|
||||
|
||||
# Verify the hosts data structure is properly initialized
|
||||
# pylint: disable=protected-access
|
||||
assert coresys.plugins.dns._hosts == [
|
||||
HostEntry(
|
||||
@@ -239,3 +251,158 @@ async def test_load_error_writing_resolv(
|
||||
|
||||
assert "Can't write/update /etc/resolv.conf" in caplog.text
|
||||
assert coresys.core.healthy is False
|
||||
|
||||
|
||||
async def test_notify_locals_changed_end_to_end_with_changes_and_running(
|
||||
coresys: CoreSys, mock_call_later
|
||||
):
|
||||
"""Test notify_locals_changed end-to-end: local DNS changes detected and plugin restarted."""
|
||||
dns_plugin = coresys.plugins.dns
|
||||
|
||||
# Set cached locals to something different from current network state
|
||||
current_locals = dns_plugin._compute_locals()
|
||||
dns_plugin._cached_locals = (
|
||||
["dns://192.168.1.1"]
|
||||
if current_locals != ["dns://192.168.1.1"]
|
||||
else ["dns://192.168.1.2"]
|
||||
)
|
||||
|
||||
with (
|
||||
patch.object(dns_plugin, "restart") as mock_restart,
|
||||
patch.object(dns_plugin.instance, "is_running", return_value=True),
|
||||
patch.object(dns_plugin, "sys_call_later", new=mock_call_later),
|
||||
):
|
||||
# Call notify_locals_changed
|
||||
dns_plugin.notify_locals_changed()
|
||||
|
||||
# Wait for the async task to complete
|
||||
await asyncio.sleep(0.1)
|
||||
|
||||
# Verify restart was called and cached locals were updated
|
||||
mock_restart.assert_called_once()
|
||||
assert dns_plugin._cached_locals == current_locals
|
||||
|
||||
|
||||
async def test_notify_locals_changed_end_to_end_with_changes_but_not_running(
|
||||
coresys: CoreSys, mock_call_later
|
||||
):
|
||||
"""Test notify_locals_changed end-to-end: local DNS changes detected but plugin not running."""
|
||||
dns_plugin = coresys.plugins.dns
|
||||
|
||||
# Set cached locals to something different from current network state
|
||||
current_locals = dns_plugin._compute_locals()
|
||||
dns_plugin._cached_locals = (
|
||||
["dns://192.168.1.1"]
|
||||
if current_locals != ["dns://192.168.1.1"]
|
||||
else ["dns://192.168.1.2"]
|
||||
)
|
||||
|
||||
with (
|
||||
patch.object(dns_plugin, "restart") as mock_restart,
|
||||
patch.object(dns_plugin.instance, "is_running", return_value=False),
|
||||
patch.object(dns_plugin, "sys_call_later", new=mock_call_later),
|
||||
):
|
||||
# Call notify_locals_changed
|
||||
dns_plugin.notify_locals_changed()
|
||||
|
||||
# Wait for the async task to complete
|
||||
await asyncio.sleep(0.1)
|
||||
|
||||
# Verify restart was NOT called but cached locals were still updated
|
||||
mock_restart.assert_not_called()
|
||||
assert dns_plugin._cached_locals == current_locals
|
||||
|
||||
|
||||
async def test_notify_locals_changed_end_to_end_no_changes(
|
||||
coresys: CoreSys, mock_call_later
|
||||
):
|
||||
"""Test notify_locals_changed end-to-end: no local DNS changes detected."""
|
||||
dns_plugin = coresys.plugins.dns
|
||||
|
||||
# Set cached locals to match current network state
|
||||
current_locals = dns_plugin._compute_locals()
|
||||
dns_plugin._cached_locals = current_locals
|
||||
|
||||
with (
|
||||
patch.object(dns_plugin, "restart") as mock_restart,
|
||||
patch.object(dns_plugin, "sys_call_later", new=mock_call_later),
|
||||
):
|
||||
# Call notify_locals_changed
|
||||
dns_plugin.notify_locals_changed()
|
||||
|
||||
# Wait for the async task to complete
|
||||
await asyncio.sleep(0.1)
|
||||
|
||||
# Verify restart was NOT called since no changes
|
||||
mock_restart.assert_not_called()
|
||||
assert dns_plugin._cached_locals == current_locals
|
||||
|
||||
|
||||
async def test_notify_locals_changed_debouncing_cancels_previous_timer(
|
||||
coresys: CoreSys,
|
||||
):
|
||||
"""Test notify_locals_changed debouncing cancels previous timer before creating new one."""
|
||||
dns_plugin = coresys.plugins.dns
|
||||
|
||||
# Set cached locals to trigger change detection
|
||||
current_locals = dns_plugin._compute_locals()
|
||||
dns_plugin._cached_locals = (
|
||||
["dns://192.168.1.1"]
|
||||
if current_locals != ["dns://192.168.1.1"]
|
||||
else ["dns://192.168.1.2"]
|
||||
)
|
||||
|
||||
call_count = 0
|
||||
handles = []
|
||||
|
||||
def mock_call_later_with_tracking(_delay, *args, **kwargs) -> asyncio.TimerHandle:
|
||||
"""Mock to remove delay and track calls."""
|
||||
nonlocal call_count
|
||||
call_count += 1
|
||||
handle = coresys.call_later(0, *args, **kwargs)
|
||||
handles.append(handle)
|
||||
return handle
|
||||
|
||||
with (
|
||||
patch.object(dns_plugin, "restart") as mock_restart,
|
||||
patch.object(dns_plugin.instance, "is_running", return_value=True),
|
||||
patch.object(dns_plugin, "sys_call_later", new=mock_call_later_with_tracking),
|
||||
):
|
||||
# First call sets up timer
|
||||
dns_plugin.notify_locals_changed()
|
||||
assert call_count == 1
|
||||
first_handle = dns_plugin._locals_changed_handle
|
||||
assert first_handle is not None
|
||||
|
||||
# Second call should cancel first timer and create new one
|
||||
dns_plugin.notify_locals_changed()
|
||||
assert call_count == 2
|
||||
second_handle = dns_plugin._locals_changed_handle
|
||||
assert second_handle is not None
|
||||
assert first_handle != second_handle
|
||||
|
||||
# Wait for the async task to complete
|
||||
await asyncio.sleep(0.1)
|
||||
|
||||
# Verify restart was called once for the final timer
|
||||
mock_restart.assert_called_once()
|
||||
assert dns_plugin._cached_locals == current_locals
|
||||
|
||||
|
||||
async def test_stop_cancels_pending_timers_and_tasks(coresys: CoreSys):
|
||||
"""Test stop cancels pending locals change timers and restart tasks to prevent resource leaks."""
|
||||
dns_plugin = coresys.plugins.dns
|
||||
|
||||
mock_timer_handle = Mock()
|
||||
mock_task_handle = Mock()
|
||||
dns_plugin._locals_changed_handle = mock_timer_handle
|
||||
dns_plugin._restart_after_locals_change_handle = mock_task_handle
|
||||
|
||||
with patch.object(dns_plugin.instance, "stop"):
|
||||
await dns_plugin.stop()
|
||||
|
||||
# Should cancel pending timer and task, then clean up
|
||||
mock_timer_handle.cancel.assert_called_once()
|
||||
mock_task_handle.cancel.assert_called_once()
|
||||
assert dns_plugin._locals_changed_handle is None
|
||||
assert dns_plugin._restart_after_locals_change_handle is None
|
||||
|
||||
Reference in New Issue
Block a user