2019-02-26 18:18:09 +00:00
|
|
|
"""Config flow to configure the Toon component."""
|
|
|
|
from collections import OrderedDict
|
|
|
|
import logging
|
2019-03-04 23:52:00 +00:00
|
|
|
from functools import partial
|
2019-02-26 18:18:09 +00:00
|
|
|
|
|
|
|
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:
|
2019-03-04 23:52:00 +00:00
|
|
|
toon = await self.hass.async_add_executor_job(partial(
|
|
|
|
Toon, user_input[CONF_USERNAME], user_input[CONF_PASSWORD],
|
|
|
|
app[CONF_CLIENT_ID], app[CONF_CLIENT_SECRET],
|
|
|
|
tenant_id=user_input[CONF_TENANT]))
|
2019-02-26 18:18:09 +00:00
|
|
|
|
|
|
|
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:
|
2019-03-04 23:52:00 +00:00
|
|
|
await self.hass.async_add_executor_job(partial(
|
|
|
|
Toon, self.username, self.password, app[CONF_CLIENT_ID],
|
|
|
|
app[CONF_CLIENT_SECRET], tenant_id=self.tenant,
|
|
|
|
display_common_name=user_input[CONF_DISPLAY]))
|
2019-02-26 18:18:09 +00:00
|
|
|
|
|
|
|
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]
|
|
|
|
}
|
|
|
|
)
|