* 🚜 Rewrite of Toon component * 🔥 Removed manual state from list * 👕 Addresses code review comments * 🔥 Removes a log line that should not have been left behind * 👕 Addresses linting warnings * 👕 Addresses Hound CI warning * 👕 Fixes small code styling issues * ✨ Sets an appropriate SCAN_INTERVAL * ✨ Sets min/max temperature for climate platform * 👕 Makes imports more consistent with codebase * 🚑 Fixes incorrect SCAN_INTERVAL value in climate platform * 🚑 Uses OrderedDict for config_flow schema * 👕 Adds return types for min/max temp * 🚜 Refactors entities into their actual devices * ⬆️ Updates toonapilib to 3.0.7 * 🚜 Refactors binary sensor state inversion * 🚑 Fixes states of OpenTherm connection and Hot Tap Water * ✨ Adds Boiler Preheat binary sensor * ✨ Adds Toon Thermostat Program binary sensor * ✨ Adds Boiler Modulation Level sensor * ✨ Adds Daily Power Cost sensor * 🔥 Cleanup of Toon Thermostat climate attributes * 🚜 Adjusts config_flow with Tenant selection * 🙋 Adds myself to codeowners file as maintainer * ⬆️ Gen requirements * ⬆️ Updates toonapilib to 3.0.9 * ☔ Adds config_flow tests
158 lines
5.2 KiB
Python
158 lines
5.2 KiB
Python
"""Config flow to configure the Toon component."""
|
|
from collections import OrderedDict
|
|
import logging
|
|
|
|
import voluptuous as vol
|
|
|
|
from homeassistant import config_entries
|
|
from homeassistant.const import CONF_PASSWORD, CONF_USERNAME
|
|
from homeassistant.core import callback
|
|
|
|
from .const import (
|
|
CONF_CLIENT_ID, CONF_CLIENT_SECRET, CONF_DISPLAY, CONF_TENANT,
|
|
DATA_TOON_CONFIG, DOMAIN)
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
@callback
|
|
def configured_displays(hass):
|
|
"""Return a set of configured Toon displays."""
|
|
return set(
|
|
entry.data[CONF_DISPLAY]
|
|
for entry in hass.config_entries.async_entries(DOMAIN)
|
|
)
|
|
|
|
|
|
@config_entries.HANDLERS.register(DOMAIN)
|
|
class ToonFlowHandler(config_entries.ConfigFlow):
|
|
"""Handle a Toon config flow."""
|
|
|
|
VERSION = 1
|
|
CONNECTION_CLASS = config_entries.CONN_CLASS_CLOUD_POLL
|
|
|
|
def __init__(self):
|
|
"""Initialize the Toon flow."""
|
|
self.displays = None
|
|
self.username = None
|
|
self.password = None
|
|
self.tenant = None
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Handle a flow initiated by the user."""
|
|
app = self.hass.data.get(DATA_TOON_CONFIG, {})
|
|
|
|
if not app:
|
|
return self.async_abort(reason='no_app')
|
|
|
|
return await self.async_step_authenticate(user_input)
|
|
|
|
async def _show_authenticaticate_form(self, errors=None):
|
|
"""Show the authentication form to the user."""
|
|
fields = OrderedDict()
|
|
fields[vol.Required(CONF_USERNAME)] = str
|
|
fields[vol.Required(CONF_PASSWORD)] = str
|
|
fields[vol.Optional(CONF_TENANT)] = vol.In([
|
|
'eneco', 'electrabel', 'viesgo'
|
|
])
|
|
|
|
return self.async_show_form(
|
|
step_id='authenticate',
|
|
data_schema=vol.Schema(fields),
|
|
errors=errors if errors else {},
|
|
)
|
|
|
|
async def async_step_authenticate(self, user_input=None):
|
|
"""Attempt to authenticate with the Toon account."""
|
|
from toonapilib import Toon
|
|
from toonapilib.toonapilibexceptions import (InvalidConsumerSecret,
|
|
InvalidConsumerKey,
|
|
InvalidCredentials,
|
|
AgreementsRetrievalError)
|
|
|
|
if user_input is None:
|
|
return await self._show_authenticaticate_form()
|
|
|
|
app = self.hass.data.get(DATA_TOON_CONFIG, {})
|
|
try:
|
|
toon = Toon(user_input[CONF_USERNAME],
|
|
user_input[CONF_PASSWORD],
|
|
app[CONF_CLIENT_ID],
|
|
app[CONF_CLIENT_SECRET],
|
|
tenant_id=user_input[CONF_TENANT])
|
|
|
|
displays = toon.display_names
|
|
|
|
except InvalidConsumerKey:
|
|
return self.async_abort(reason='client_id')
|
|
|
|
except InvalidConsumerSecret:
|
|
return self.async_abort(reason='client_secret')
|
|
|
|
except InvalidCredentials:
|
|
return await self._show_authenticaticate_form({
|
|
'base': 'credentials'
|
|
})
|
|
|
|
except AgreementsRetrievalError:
|
|
return self.async_abort(reason='no_agreements')
|
|
|
|
except Exception: # pylint: disable=broad-except
|
|
_LOGGER.exception("Unexpected error while authenticating")
|
|
return self.async_abort(reason='unknown_auth_fail')
|
|
|
|
self.displays = displays
|
|
self.username = user_input[CONF_USERNAME]
|
|
self.password = user_input[CONF_PASSWORD]
|
|
self.tenant = user_input[CONF_TENANT]
|
|
|
|
return await self.async_step_display()
|
|
|
|
async def _show_display_form(self, errors=None):
|
|
"""Show the select display form to the user."""
|
|
fields = OrderedDict()
|
|
fields[vol.Required(CONF_DISPLAY)] = vol.In(self.displays)
|
|
|
|
return self.async_show_form(
|
|
step_id='display',
|
|
data_schema=vol.Schema(fields),
|
|
errors=errors if errors else {},
|
|
)
|
|
|
|
async def async_step_display(self, user_input=None):
|
|
"""Select Toon display to add."""
|
|
from toonapilib import Toon
|
|
|
|
if not self.displays:
|
|
return self.async_abort(reason='no_displays')
|
|
|
|
if user_input is None:
|
|
return await self._show_display_form()
|
|
|
|
if user_input[CONF_DISPLAY] in configured_displays(self.hass):
|
|
return await self._show_display_form({
|
|
'base': 'display_exists'
|
|
})
|
|
|
|
app = self.hass.data.get(DATA_TOON_CONFIG, {})
|
|
try:
|
|
Toon(self.username,
|
|
self.password,
|
|
app[CONF_CLIENT_ID],
|
|
app[CONF_CLIENT_SECRET],
|
|
tenant_id=self.tenant,
|
|
display_common_name=user_input[CONF_DISPLAY])
|
|
|
|
except Exception: # pylint: disable=broad-except
|
|
_LOGGER.exception("Unexpected error while authenticating")
|
|
return self.async_abort(reason='unknown_auth_fail')
|
|
|
|
return self.async_create_entry(
|
|
title=user_input[CONF_DISPLAY],
|
|
data={
|
|
CONF_USERNAME: self.username,
|
|
CONF_PASSWORD: self.password,
|
|
CONF_TENANT: self.tenant,
|
|
CONF_DISPLAY: user_input[CONF_DISPLAY]
|
|
}
|
|
)
|