Fix reauthentication for powerwall integration (#72174)

This commit is contained in:
jrester 2022-05-20 21:53:43 +02:00 committed by GitHub
parent 1e7b187fc6
commit ad5dbae425
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 71 additions and 27 deletions

View file

@ -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,
) )

View file

@ -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

View file

@ -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

View 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"