Add switch error handling for Peblar Rocksolid EV Chargers (#133805)

This commit is contained in:
Franck Nijhof 2024-12-22 19:47:31 +01:00 committed by GitHub
parent c6789d70a4
commit de5a49363e
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 154 additions and 3 deletions

View File

@ -20,6 +20,7 @@ from .coordinator import (
PeblarRuntimeData,
)
from .entity import PeblarEntity
from .helpers import peblar_exception_handler
PARALLEL_UPDATES = 1
@ -78,11 +79,13 @@ class PeblarSwitchEntity(
"""Return state of the switch."""
return self.entity_description.is_on_fn(self.coordinator.data)
@peblar_exception_handler
async def async_turn_on(self, **kwargs: Any) -> None:
"""Turn the entity on."""
await self.entity_description.set_fn(self.coordinator.api, True)
await self.coordinator.async_request_refresh()
@peblar_exception_handler
async def async_turn_off(self, **kwargs: Any) -> None:
"""Turn the entity off."""
await self.entity_description.set_fn(self.coordinator.api, False)

View File

@ -1,18 +1,31 @@
"""Tests for the Peblar switch platform."""
from unittest.mock import MagicMock
from peblar import PeblarAuthenticationError, PeblarConnectionError, PeblarError
import pytest
from syrupy.assertion import SnapshotAssertion
from homeassistant.components.peblar.const import DOMAIN
from homeassistant.const import Platform
from homeassistant.components.switch import (
DOMAIN as SWITCH_DOMAIN,
SERVICE_TURN_OFF,
SERVICE_TURN_ON,
)
from homeassistant.config_entries import SOURCE_REAUTH, ConfigEntryState
from homeassistant.const import ATTR_ENTITY_ID, Platform
from homeassistant.core import HomeAssistant
from homeassistant.exceptions import HomeAssistantError
from homeassistant.helpers import device_registry as dr, entity_registry as er
from tests.common import MockConfigEntry, snapshot_platform
pytestmark = [
pytest.mark.parametrize("init_integration", [Platform.SWITCH], indirect=True),
pytest.mark.usefixtures("init_integration"),
]
@pytest.mark.parametrize("init_integration", [Platform.SWITCH], indirect=True)
@pytest.mark.usefixtures("init_integration")
async def test_entities(
hass: HomeAssistant,
snapshot: SnapshotAssertion,
@ -33,3 +46,138 @@ async def test_entities(
)
for entity_entry in entity_entries:
assert entity_entry.device_id == device_entry.id
@pytest.mark.parametrize(
("service", "force_single_phase"),
[
(SERVICE_TURN_ON, True),
(SERVICE_TURN_OFF, False),
],
)
@pytest.mark.usefixtures("entity_registry_enabled_by_default")
async def test_switch(
hass: HomeAssistant,
mock_peblar: MagicMock,
service: str,
force_single_phase: bool,
) -> None:
"""Test the Peblar EV charger switches."""
entity_id = "switch.peblar_ev_charger_force_single_phase"
mocked_method = mock_peblar.rest_api.return_value.ev_interface
mocked_method.reset_mock()
# Test normal happy path for changing the switch state
await hass.services.async_call(
SWITCH_DOMAIN,
service,
{ATTR_ENTITY_ID: entity_id},
blocking=True,
)
assert len(mocked_method.mock_calls) == 2
mocked_method.mock_calls[0].assert_called_with(
{"force_single_phase": force_single_phase}
)
@pytest.mark.parametrize(
("error", "error_match", "translation_key", "translation_placeholders"),
[
(
PeblarConnectionError("Could not connect"),
(
r"An error occurred while communicating "
r"with the Peblar device: Could not connect"
),
"communication_error",
{"error": "Could not connect"},
),
(
PeblarError("Unknown error"),
(
r"An unknown error occurred while communicating "
r"with the Peblar device: Unknown error"
),
"unknown_error",
{"error": "Unknown error"},
),
],
)
@pytest.mark.parametrize("service", [SERVICE_TURN_ON, SERVICE_TURN_OFF])
@pytest.mark.usefixtures("entity_registry_enabled_by_default")
async def test_switch_communication_error(
hass: HomeAssistant,
mock_peblar: MagicMock,
error: Exception,
error_match: str,
translation_key: str,
translation_placeholders: dict,
service: str,
) -> None:
"""Test the Peblar EV charger when a communication error occurs."""
entity_id = "switch.peblar_ev_charger_force_single_phase"
mock_peblar.rest_api.return_value.ev_interface.side_effect = error
with pytest.raises(
HomeAssistantError,
match=error_match,
) as excinfo:
await hass.services.async_call(
SWITCH_DOMAIN,
service,
{ATTR_ENTITY_ID: entity_id},
blocking=True,
)
assert excinfo.value.translation_domain == DOMAIN
assert excinfo.value.translation_key == translation_key
assert excinfo.value.translation_placeholders == translation_placeholders
@pytest.mark.parametrize("service", [SERVICE_TURN_ON, SERVICE_TURN_OFF])
async def test_switch_authentication_error(
hass: HomeAssistant,
mock_peblar: MagicMock,
mock_config_entry: MockConfigEntry,
service: str,
) -> None:
"""Test the Peblar EV charger when an authentication error occurs."""
entity_id = "switch.peblar_ev_charger_force_single_phase"
mock_peblar.rest_api.return_value.ev_interface.side_effect = (
PeblarAuthenticationError("Authentication error")
)
mock_peblar.login.side_effect = PeblarAuthenticationError("Authentication error")
with pytest.raises(
HomeAssistantError,
match=(
r"An authentication failure occurred while communicating "
r"with the Peblar device"
),
) as excinfo:
await hass.services.async_call(
SWITCH_DOMAIN,
service,
{ATTR_ENTITY_ID: entity_id},
blocking=True,
)
assert excinfo.value.translation_domain == DOMAIN
assert excinfo.value.translation_key == "authentication_error"
assert not excinfo.value.translation_placeholders
# Ensure the device is reloaded on authentication error and triggers
# a reauthentication flow.
await hass.async_block_till_done()
assert mock_config_entry.state is ConfigEntryState.SETUP_ERROR
flows = hass.config_entries.flow.async_progress()
assert len(flows) == 1
flow = flows[0]
assert flow["step_id"] == "reauth_confirm"
assert flow["handler"] == DOMAIN
assert "context" in flow
assert flow["context"].get("source") == SOURCE_REAUTH
assert flow["context"].get("entry_id") == mock_config_entry.entry_id