"""Test network manager.""" import asyncio from ipaddress import IPv4Address, IPv6Address from unittest.mock import patch from dbus_fast import Variant import pytest from supervisor.const import CoreState from supervisor.coresys import CoreSys from supervisor.dbus.const import InterfaceMethod from supervisor.exceptions import HostNotSupportedError from supervisor.homeassistant.const import WSEvent, WSType from supervisor.host.const import WifiMode from tests.dbus_service_mocks.base import DBusServiceMock from tests.dbus_service_mocks.network_active_connection import ( ActiveConnection as ActiveConnectionService, ) from tests.dbus_service_mocks.network_connection_settings import ( SETTINGS_FIXTURE, ConnectionSettings as ConnectionSettingsService, ) from tests.dbus_service_mocks.network_device_wireless import ( DeviceWireless as DeviceWirelessService, ) from tests.dbus_service_mocks.network_manager import ( NetworkManager as NetworkManagerService, ) @pytest.fixture(name="active_connection_service") async def fixture_active_connection_service( network_manager_services: dict[str, DBusServiceMock | dict[str, DBusServiceMock]] ) -> ActiveConnectionService: """Return mock active connection service.""" yield network_manager_services["network_active_connection"] @pytest.fixture(name="wireless_service") async def fixture_wireless_service( network_manager_services: dict[str, DBusServiceMock | dict[str, DBusServiceMock]] ) -> DeviceWirelessService: """Return mock device wireless service.""" yield network_manager_services["network_device_wireless"] async def test_load(coresys: CoreSys, network_manager_service: NetworkManagerService): """Test network manager load.""" network_manager_service.ActivateConnection.calls.clear() network_manager_service.CheckConnectivity.calls.clear() await coresys.host.network.load() assert coresys.host.network.connectivity is True assert len(coresys.host.network.dns_servers) == 1 assert str(coresys.host.network.dns_servers[0]) == "192.168.30.1" assert len(coresys.host.network.interfaces) == 2 assert coresys.host.network.interfaces[0].name == "eth0" assert coresys.host.network.interfaces[0].enabled is True assert coresys.host.network.interfaces[0].ipv4.method == InterfaceMethod.AUTO assert coresys.host.network.interfaces[0].ipv4.gateway == IPv4Address("192.168.2.1") assert coresys.host.network.interfaces[0].ipv4.ready is True assert coresys.host.network.interfaces[0].ipv6.method == InterfaceMethod.AUTO assert coresys.host.network.interfaces[0].ipv6.gateway == IPv6Address( "fe80::da58:d7ff:fe00:9c69" ) assert coresys.host.network.interfaces[0].ipv6.ready is True assert coresys.host.network.interfaces[1].name == "wlan0" assert coresys.host.network.interfaces[1].enabled is False assert network_manager_service.ActivateConnection.calls == [ ( "/org/freedesktop/NetworkManager/Settings/1", "/org/freedesktop/NetworkManager/Devices/1", "/", ) ] assert network_manager_service.CheckConnectivity.calls == [] async def test_load_with_disabled_methods( coresys: CoreSys, network_manager_service: NetworkManagerService, connection_settings_service: ConnectionSettingsService, ): """Test load does not disable methods of interfaces.""" network_manager_service.ActivateConnection.calls.clear() disabled = {"method": Variant("s", "disabled")} connection_settings_service.settings = SETTINGS_FIXTURE | { "ipv4": disabled, "ipv6": disabled, } await coresys.dbus.network.interfaces["eth0"].settings.reload() await coresys.host.network.load() assert network_manager_service.ActivateConnection.calls == [] async def test_load_with_network_connection_issues( coresys: CoreSys, network_manager_service: NetworkManagerService, active_connection_service: ActiveConnectionService, ): """Test load does not update interfaces with network connection issues.""" network_manager_service.ActivateConnection.calls.clear() active_connection_service.emit_properties_changed( {"StateFlags": 0x10, "Ip4Config": "/"} ) await active_connection_service.ping() await coresys.host.network.load() assert network_manager_service.ActivateConnection.calls == [] assert len(coresys.host.network.interfaces) == 2 assert coresys.host.network.interfaces[0].name == "eth0" assert coresys.host.network.interfaces[0].enabled is True assert coresys.host.network.interfaces[0].ipv4.method == InterfaceMethod.AUTO assert coresys.host.network.interfaces[0].ipv4.gateway is None assert coresys.host.network.interfaces[0].ipv6.method == InterfaceMethod.AUTO assert coresys.host.network.interfaces[0].ipv6.gateway == IPv6Address( "fe80::da58:d7ff:fe00:9c69" ) async def test_scan_wifi(coresys: CoreSys): """Test scanning wifi.""" with pytest.raises(HostNotSupportedError): await coresys.host.network.scan_wifi(coresys.host.network.get("eth0")) with patch("supervisor.host.network.asyncio.sleep"): aps = await coresys.host.network.scan_wifi(coresys.host.network.get("wlan0")) assert len(aps) == 2 assert aps[0].mac == "E4:57:40:A9:D7:DE" assert aps[0].mode == WifiMode.INFRASTRUCTURE assert aps[1].mac == "18:4B:0D:23:A1:9C" assert aps[1].mode == WifiMode.INFRASTRUCTURE async def test_scan_wifi_with_failures( coresys: CoreSys, wireless_service: DeviceWirelessService, caplog ): """Test scanning wifi with accesspoint processing failures.""" wireless_service.all_access_points = [ "/org/freedesktop/NetworkManager/AccessPoint/43099", "/org/freedesktop/NetworkManager/AccessPoint/43100", "/org/freedesktop/NetworkManager/AccessPoint/99999", ] with patch("supervisor.host.network.asyncio.sleep"): aps = await coresys.host.network.scan_wifi(coresys.host.network.get("wlan0")) assert len(aps) == 2 assert "Can't process an AP" in caplog.text async def test_host_connectivity_changed( coresys: CoreSys, network_manager_service: NetworkManagerService ): """Test host connectivity changed.""" # pylint: disable=protected-access client = coresys.homeassistant.websocket._client # pylint: enable=protected-access await coresys.host.load() assert coresys.host.network.connectivity is True network_manager_service.emit_properties_changed({"Connectivity": 1}) await network_manager_service.ping() assert coresys.host.network.connectivity is False await asyncio.sleep(0) client.async_send_command.assert_called_once_with( { "type": WSType.SUPERVISOR_EVENT, "data": { "event": WSEvent.SUPERVISOR_UPDATE, "update_key": "network", "data": {"host_internet": False}, }, } ) client.async_send_command.reset_mock() network_manager_service.emit_properties_changed({}, ["Connectivity"]) await network_manager_service.ping() await network_manager_service.ping() assert coresys.host.network.connectivity is True await asyncio.sleep(0) client.async_send_command.assert_called_once_with( { "type": WSType.SUPERVISOR_EVENT, "data": { "event": WSEvent.SUPERVISOR_UPDATE, "update_key": "network", "data": {"host_internet": True}, }, } ) async def test_host_connectivity_disabled( coresys: CoreSys, network_manager_service: NetworkManagerService ): """Test host connectivity check disabled.""" # pylint: disable=protected-access client = coresys.homeassistant.websocket._client # pylint: enable=protected-access await coresys.host.network.load() coresys.core.state = CoreState.RUNNING await asyncio.sleep(0) client.async_send_command.reset_mock() assert "connectivity_check" not in coresys.resolution.unsupported assert coresys.host.network.connectivity is True network_manager_service.emit_properties_changed({"ConnectivityCheckEnabled": False}) await network_manager_service.ping() assert coresys.host.network.connectivity is None await asyncio.sleep(0) client.async_send_command.assert_any_call( { "type": WSType.SUPERVISOR_EVENT, "data": { "event": WSEvent.SUPERVISOR_UPDATE, "update_key": "network", "data": {"host_internet": None}, }, } ) assert "connectivity_check" in coresys.resolution.unsupported client.async_send_command.reset_mock() network_manager_service.emit_properties_changed({"ConnectivityCheckEnabled": True}) await network_manager_service.ping() await network_manager_service.ping() assert coresys.host.network.connectivity is True await asyncio.sleep(0) client.async_send_command.assert_any_call( { "type": WSType.SUPERVISOR_EVENT, "data": { "event": WSEvent.SUPERVISOR_UPDATE, "update_key": "network", "data": {"host_internet": True}, }, } ) assert "connectivity_check" not in coresys.resolution.unsupported