2019-10-13 18:01:04 +00:00
|
|
|
"""Config flow for the Abode Security System component."""
|
|
|
|
from abodepy import Abode
|
2020-11-27 12:39:26 +00:00
|
|
|
from abodepy.exceptions import AbodeAuthenticationException, AbodeException
|
|
|
|
from abodepy.helpers.errors import MFA_CODE_REQUIRED
|
2019-10-13 18:01:04 +00:00
|
|
|
from requests.exceptions import ConnectTimeout, HTTPError
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant import config_entries
|
2020-04-09 19:43:42 +00:00
|
|
|
from homeassistant.const import CONF_PASSWORD, CONF_USERNAME, HTTP_BAD_REQUEST
|
2019-10-13 18:01:04 +00:00
|
|
|
|
2021-03-30 04:02:56 +00:00
|
|
|
from .const import DEFAULT_CACHEDB, DOMAIN, LOGGER
|
2019-10-13 18:01:04 +00:00
|
|
|
|
2020-11-27 12:39:26 +00:00
|
|
|
CONF_MFA = "mfa_code"
|
2019-10-13 18:01:04 +00:00
|
|
|
CONF_POLLING = "polling"
|
|
|
|
|
|
|
|
|
|
|
|
class AbodeFlowHandler(config_entries.ConfigFlow, domain=DOMAIN):
|
|
|
|
"""Config flow for Abode."""
|
|
|
|
|
|
|
|
VERSION = 1
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
"""Initialize."""
|
|
|
|
self.data_schema = {
|
|
|
|
vol.Required(CONF_USERNAME): str,
|
|
|
|
vol.Required(CONF_PASSWORD): str,
|
|
|
|
}
|
2020-11-27 12:39:26 +00:00
|
|
|
self.mfa_data_schema = {
|
|
|
|
vol.Required(CONF_MFA): str,
|
|
|
|
}
|
|
|
|
|
|
|
|
self._cache = None
|
|
|
|
self._mfa_code = None
|
|
|
|
self._password = None
|
|
|
|
self._polling = False
|
|
|
|
self._username = None
|
|
|
|
|
|
|
|
async def _async_abode_login(self, step_id):
|
|
|
|
"""Handle login with Abode."""
|
|
|
|
self._cache = self.hass.config.path(DEFAULT_CACHEDB)
|
|
|
|
errors = {}
|
|
|
|
|
|
|
|
try:
|
|
|
|
await self.hass.async_add_executor_job(
|
|
|
|
Abode, self._username, self._password, True, False, False, self._cache
|
|
|
|
)
|
|
|
|
|
|
|
|
except (AbodeException, ConnectTimeout, HTTPError) as ex:
|
|
|
|
if ex.errcode == MFA_CODE_REQUIRED[0]:
|
|
|
|
return await self.async_step_mfa()
|
|
|
|
|
|
|
|
LOGGER.error("Unable to connect to Abode: %s", ex)
|
|
|
|
|
|
|
|
if ex.errcode == HTTP_BAD_REQUEST:
|
|
|
|
errors = {"base": "invalid_auth"}
|
|
|
|
|
|
|
|
else:
|
|
|
|
errors = {"base": "cannot_connect"}
|
|
|
|
|
|
|
|
if errors:
|
|
|
|
return self.async_show_form(
|
|
|
|
step_id=step_id, data_schema=vol.Schema(self.data_schema), errors=errors
|
|
|
|
)
|
|
|
|
|
|
|
|
return await self._async_create_entry()
|
|
|
|
|
|
|
|
async def _async_abode_mfa_login(self):
|
|
|
|
"""Handle multi-factor authentication (MFA) login with Abode."""
|
|
|
|
try:
|
|
|
|
# Create instance to access login method for passing MFA code
|
|
|
|
abode = Abode(
|
|
|
|
auto_login=False,
|
|
|
|
get_devices=False,
|
|
|
|
get_automations=False,
|
|
|
|
cache_path=self._cache,
|
|
|
|
)
|
|
|
|
await self.hass.async_add_executor_job(
|
|
|
|
abode.login, self._username, self._password, self._mfa_code
|
|
|
|
)
|
|
|
|
|
|
|
|
except AbodeAuthenticationException:
|
|
|
|
return self.async_show_form(
|
|
|
|
step_id="mfa",
|
|
|
|
data_schema=vol.Schema(self.mfa_data_schema),
|
|
|
|
errors={"base": "invalid_mfa_code"},
|
|
|
|
)
|
|
|
|
|
|
|
|
return await self._async_create_entry()
|
|
|
|
|
|
|
|
async def _async_create_entry(self):
|
|
|
|
"""Create the config entry."""
|
|
|
|
config_data = {
|
|
|
|
CONF_USERNAME: self._username,
|
|
|
|
CONF_PASSWORD: self._password,
|
|
|
|
CONF_POLLING: self._polling,
|
|
|
|
}
|
|
|
|
existing_entry = await self.async_set_unique_id(self._username)
|
|
|
|
|
|
|
|
if existing_entry:
|
|
|
|
self.hass.config_entries.async_update_entry(
|
|
|
|
existing_entry, data=config_data
|
|
|
|
)
|
|
|
|
# Reload the Abode config entry otherwise devices will remain unavailable
|
|
|
|
self.hass.async_create_task(
|
|
|
|
self.hass.config_entries.async_reload(existing_entry.entry_id)
|
|
|
|
)
|
|
|
|
|
|
|
|
return self.async_abort(reason="reauth_successful")
|
|
|
|
|
|
|
|
return self.async_create_entry(title=self._username, data=config_data)
|
2019-10-13 18:01:04 +00:00
|
|
|
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
|
|
"""Handle a flow initialized by the user."""
|
|
|
|
if self._async_current_entries():
|
|
|
|
return self.async_abort(reason="single_instance_allowed")
|
|
|
|
|
2020-11-27 12:39:26 +00:00
|
|
|
if user_input is None:
|
|
|
|
return self.async_show_form(
|
|
|
|
step_id="user", data_schema=vol.Schema(self.data_schema)
|
|
|
|
)
|
2019-10-13 18:01:04 +00:00
|
|
|
|
2020-11-27 12:39:26 +00:00
|
|
|
self._username = user_input[CONF_USERNAME]
|
|
|
|
self._password = user_input[CONF_PASSWORD]
|
2019-10-13 18:01:04 +00:00
|
|
|
|
2020-11-27 12:39:26 +00:00
|
|
|
return await self._async_abode_login(step_id="user")
|
|
|
|
|
|
|
|
async def async_step_mfa(self, user_input=None):
|
|
|
|
"""Handle a multi-factor authentication (MFA) flow."""
|
|
|
|
if user_input is None:
|
|
|
|
return self.async_show_form(
|
|
|
|
step_id="mfa", data_schema=vol.Schema(self.mfa_data_schema)
|
2019-11-02 00:28:50 +00:00
|
|
|
)
|
2019-10-13 18:01:04 +00:00
|
|
|
|
2020-11-27 12:39:26 +00:00
|
|
|
self._mfa_code = user_input[CONF_MFA]
|
|
|
|
|
|
|
|
return await self._async_abode_mfa_login()
|
|
|
|
|
|
|
|
async def async_step_reauth(self, config):
|
|
|
|
"""Handle reauthorization request from Abode."""
|
|
|
|
self._username = config[CONF_USERNAME]
|
|
|
|
|
|
|
|
return await self.async_step_reauth_confirm()
|
|
|
|
|
|
|
|
async def async_step_reauth_confirm(self, user_input=None):
|
|
|
|
"""Handle reauthorization flow."""
|
|
|
|
if user_input is None:
|
|
|
|
return self.async_show_form(
|
|
|
|
step_id="reauth_confirm",
|
|
|
|
data_schema=vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_USERNAME, default=self._username): str,
|
|
|
|
vol.Required(CONF_PASSWORD): str,
|
|
|
|
}
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
self._username = user_input[CONF_USERNAME]
|
|
|
|
self._password = user_input[CONF_PASSWORD]
|
|
|
|
|
|
|
|
return await self._async_abode_login(step_id="reauth_confirm")
|