Fix reauthentication for powerwall integration (#72174)
This commit is contained in:
parent
1e7b187fc6
commit
ad5dbae425
4 changed files with 71 additions and 27 deletions
|
@ -29,7 +29,6 @@ from .const import (
|
||||||
POWERWALL_API_CHANGED,
|
POWERWALL_API_CHANGED,
|
||||||
POWERWALL_COORDINATOR,
|
POWERWALL_COORDINATOR,
|
||||||
POWERWALL_HTTP_SESSION,
|
POWERWALL_HTTP_SESSION,
|
||||||
POWERWALL_LOGIN_FAILED_COUNT,
|
|
||||||
UPDATE_INTERVAL,
|
UPDATE_INTERVAL,
|
||||||
)
|
)
|
||||||
from .models import PowerwallBaseInfo, PowerwallData, PowerwallRuntimeData
|
from .models import PowerwallBaseInfo, PowerwallData, PowerwallRuntimeData
|
||||||
|
@ -40,8 +39,6 @@ PLATFORMS = [Platform.BINARY_SENSOR, Platform.SENSOR]
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
MAX_LOGIN_FAILURES = 5
|
|
||||||
|
|
||||||
API_CHANGED_ERROR_BODY = (
|
API_CHANGED_ERROR_BODY = (
|
||||||
"It seems like your powerwall uses an unsupported version. "
|
"It seems like your powerwall uses an unsupported version. "
|
||||||
"Please update the software of your powerwall or if it is "
|
"Please update the software of your powerwall or if it is "
|
||||||
|
@ -68,29 +65,15 @@ class PowerwallDataManager:
|
||||||
self.runtime_data = runtime_data
|
self.runtime_data = runtime_data
|
||||||
self.power_wall = power_wall
|
self.power_wall = power_wall
|
||||||
|
|
||||||
@property
|
|
||||||
def login_failed_count(self) -> int:
|
|
||||||
"""Return the current number of failed logins."""
|
|
||||||
return self.runtime_data[POWERWALL_LOGIN_FAILED_COUNT]
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def api_changed(self) -> int:
|
def api_changed(self) -> int:
|
||||||
"""Return true if the api has changed out from under us."""
|
"""Return true if the api has changed out from under us."""
|
||||||
return self.runtime_data[POWERWALL_API_CHANGED]
|
return self.runtime_data[POWERWALL_API_CHANGED]
|
||||||
|
|
||||||
def _increment_failed_logins(self) -> None:
|
|
||||||
self.runtime_data[POWERWALL_LOGIN_FAILED_COUNT] += 1
|
|
||||||
|
|
||||||
def _clear_failed_logins(self) -> None:
|
|
||||||
self.runtime_data[POWERWALL_LOGIN_FAILED_COUNT] = 0
|
|
||||||
|
|
||||||
def _recreate_powerwall_login(self) -> None:
|
def _recreate_powerwall_login(self) -> None:
|
||||||
"""Recreate the login on auth failure."""
|
"""Recreate the login on auth failure."""
|
||||||
http_session = self.runtime_data[POWERWALL_HTTP_SESSION]
|
if self.power_wall.is_authenticated():
|
||||||
http_session.close()
|
self.power_wall.logout()
|
||||||
http_session = requests.Session()
|
|
||||||
self.runtime_data[POWERWALL_HTTP_SESSION] = http_session
|
|
||||||
self.power_wall = Powerwall(self.ip_address, http_session=http_session)
|
|
||||||
self.power_wall.login(self.password or "")
|
self.power_wall.login(self.password or "")
|
||||||
|
|
||||||
async def async_update_data(self) -> PowerwallData:
|
async def async_update_data(self) -> PowerwallData:
|
||||||
|
@ -121,17 +104,15 @@ class PowerwallDataManager:
|
||||||
raise UpdateFailed("The powerwall api has changed") from err
|
raise UpdateFailed("The powerwall api has changed") from err
|
||||||
except AccessDeniedError as err:
|
except AccessDeniedError as err:
|
||||||
if attempt == 1:
|
if attempt == 1:
|
||||||
self._increment_failed_logins()
|
# failed to authenticate => the credentials must be wrong
|
||||||
raise ConfigEntryAuthFailed from err
|
raise ConfigEntryAuthFailed from err
|
||||||
if self.password is None:
|
if self.password is None:
|
||||||
raise ConfigEntryAuthFailed from err
|
raise ConfigEntryAuthFailed from err
|
||||||
raise UpdateFailed(
|
_LOGGER.debug("Access denied, trying to reauthenticate")
|
||||||
f"Login attempt {self.login_failed_count}/{MAX_LOGIN_FAILURES} failed, will retry: {err}"
|
# there is still an attempt left to authenticate, so we continue in the loop
|
||||||
) from err
|
|
||||||
except APIError as err:
|
except APIError as err:
|
||||||
raise UpdateFailed(f"Updated failed due to {err}, will retry") from err
|
raise UpdateFailed(f"Updated failed due to {err}, will retry") from err
|
||||||
else:
|
else:
|
||||||
self._clear_failed_logins()
|
|
||||||
return data
|
return data
|
||||||
raise RuntimeError("unreachable")
|
raise RuntimeError("unreachable")
|
||||||
|
|
||||||
|
@ -174,7 +155,6 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||||
api_changed=False,
|
api_changed=False,
|
||||||
base_info=base_info,
|
base_info=base_info,
|
||||||
http_session=http_session,
|
http_session=http_session,
|
||||||
login_failed_count=0,
|
|
||||||
coordinator=None,
|
coordinator=None,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -7,7 +7,6 @@ POWERWALL_BASE_INFO: Final = "base_info"
|
||||||
POWERWALL_COORDINATOR: Final = "coordinator"
|
POWERWALL_COORDINATOR: Final = "coordinator"
|
||||||
POWERWALL_API_CHANGED: Final = "api_changed"
|
POWERWALL_API_CHANGED: Final = "api_changed"
|
||||||
POWERWALL_HTTP_SESSION: Final = "http_session"
|
POWERWALL_HTTP_SESSION: Final = "http_session"
|
||||||
POWERWALL_LOGIN_FAILED_COUNT: Final = "login_failed_count"
|
|
||||||
|
|
||||||
UPDATE_INTERVAL = 30
|
UPDATE_INTERVAL = 30
|
||||||
|
|
||||||
|
|
|
@ -45,7 +45,6 @@ class PowerwallRuntimeData(TypedDict):
|
||||||
"""Run time data for the powerwall."""
|
"""Run time data for the powerwall."""
|
||||||
|
|
||||||
coordinator: DataUpdateCoordinator | None
|
coordinator: DataUpdateCoordinator | None
|
||||||
login_failed_count: int
|
|
||||||
base_info: PowerwallBaseInfo
|
base_info: PowerwallBaseInfo
|
||||||
api_changed: bool
|
api_changed: bool
|
||||||
http_session: Session
|
http_session: Session
|
||||||
|
|
66
tests/components/powerwall/test_init.py
Normal file
66
tests/components/powerwall/test_init.py
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
"""Tests for the PowerwallDataManager."""
|
||||||
|
|
||||||
|
import datetime
|
||||||
|
from unittest.mock import MagicMock, patch
|
||||||
|
|
||||||
|
from tesla_powerwall import AccessDeniedError, LoginResponse
|
||||||
|
|
||||||
|
from homeassistant.components.powerwall.const import DOMAIN
|
||||||
|
from homeassistant.config_entries import ConfigEntryState
|
||||||
|
from homeassistant.const import CONF_IP_ADDRESS, CONF_PASSWORD
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
from homeassistant.util.dt import utcnow
|
||||||
|
|
||||||
|
from .mocks import _mock_powerwall_with_fixtures
|
||||||
|
|
||||||
|
from tests.common import MockConfigEntry, async_fire_time_changed
|
||||||
|
|
||||||
|
|
||||||
|
async def test_update_data_reauthenticate_on_access_denied(hass: HomeAssistant):
|
||||||
|
"""Test if _update_data of PowerwallDataManager reauthenticates on AccessDeniedError."""
|
||||||
|
|
||||||
|
mock_powerwall = await _mock_powerwall_with_fixtures(hass)
|
||||||
|
# login responses for the different tests:
|
||||||
|
# 1. login success on entry setup
|
||||||
|
# 2. login success after reauthentication
|
||||||
|
# 3. login failure after reauthentication
|
||||||
|
mock_powerwall.login = MagicMock(name="login", return_value=LoginResponse({}))
|
||||||
|
mock_powerwall.get_charge = MagicMock(name="get_charge", return_value=90.0)
|
||||||
|
mock_powerwall.is_authenticated = MagicMock(
|
||||||
|
name="is_authenticated", return_value=True
|
||||||
|
)
|
||||||
|
mock_powerwall.logout = MagicMock(name="logout")
|
||||||
|
|
||||||
|
config_entry = MockConfigEntry(
|
||||||
|
domain=DOMAIN, data={CONF_IP_ADDRESS: "1.2.3.4", CONF_PASSWORD: "password"}
|
||||||
|
)
|
||||||
|
config_entry.add_to_hass(hass)
|
||||||
|
with patch(
|
||||||
|
"homeassistant.components.powerwall.config_flow.Powerwall",
|
||||||
|
return_value=mock_powerwall,
|
||||||
|
), patch(
|
||||||
|
"homeassistant.components.powerwall.Powerwall", return_value=mock_powerwall
|
||||||
|
):
|
||||||
|
assert await hass.config_entries.async_setup(config_entry.entry_id)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
mock_powerwall.login.reset_mock(return_value=True)
|
||||||
|
mock_powerwall.get_charge.side_effect = [AccessDeniedError("test"), 90.0]
|
||||||
|
|
||||||
|
async_fire_time_changed(hass, utcnow() + datetime.timedelta(minutes=1))
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
flows = hass.config_entries.flow.async_progress(DOMAIN)
|
||||||
|
assert len(flows) == 0
|
||||||
|
|
||||||
|
mock_powerwall.login.reset_mock()
|
||||||
|
mock_powerwall.login.side_effect = AccessDeniedError("test")
|
||||||
|
mock_powerwall.get_charge.side_effect = [AccessDeniedError("test"), 90.0]
|
||||||
|
|
||||||
|
async_fire_time_changed(hass, utcnow() + datetime.timedelta(minutes=1))
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
assert config_entry.state == ConfigEntryState.LOADED
|
||||||
|
|
||||||
|
flows = hass.config_entries.flow.async_progress(DOMAIN)
|
||||||
|
assert len(flows) == 1
|
||||||
|
reauth_flow = flows[0]
|
||||||
|
assert reauth_flow["context"]["source"] == "reauth"
|
Loading…
Add table
Reference in a new issue