"""Config flow for Fibaro integration.""" from __future__ import annotations from collections.abc import Mapping import logging from typing import Any from slugify import slugify import voluptuous as vol from homeassistant.config_entries import ConfigEntry, ConfigFlow, ConfigFlowResult from homeassistant.const import CONF_PASSWORD, CONF_URL, CONF_USERNAME from homeassistant.core import HomeAssistant from . import FibaroAuthFailed, FibaroConnectFailed, FibaroController from .const import CONF_IMPORT_PLUGINS, DOMAIN _LOGGER = logging.getLogger(__name__) STEP_USER_DATA_SCHEMA = vol.Schema( { vol.Required(CONF_URL): str, vol.Required(CONF_USERNAME): str, vol.Required(CONF_PASSWORD): str, vol.Optional(CONF_IMPORT_PLUGINS, default=False): bool, } ) def _connect_to_fibaro(data: dict[str, Any]) -> FibaroController: """Validate the user input allows us to connect to fibaro.""" controller = FibaroController(data) controller.connect_with_error_handling() return controller async def _validate_input(hass: HomeAssistant, data: dict[str, Any]) -> dict[str, Any]: """Validate the user input allows us to connect. Data has the keys from STEP_USER_DATA_SCHEMA with values provided by the user. """ controller = await hass.async_add_executor_job(_connect_to_fibaro, data) _LOGGER.debug( "Successfully connected to fibaro home center %s with name %s", controller.hub_serial, controller.hub_name, ) return { "serial_number": slugify(controller.hub_serial), "name": controller.hub_name, } def _normalize_url(url: str) -> str: """Try to fix errors in the entered url. We know that the url should be in the format http:///api/ """ if url.endswith("/api"): return f"{url}/" if not url.endswith("/api/"): return f"{url}api/" if url.endswith("/") else f"{url}/api/" return url class FibaroConfigFlow(ConfigFlow, domain=DOMAIN): """Handle a config flow for Fibaro.""" VERSION = 1 def __init__(self) -> None: """Initialize.""" self._reauth_entry: ConfigEntry | None = None async def async_step_user( self, user_input: dict[str, Any] | None = None ) -> ConfigFlowResult: """Handle the initial step.""" errors = {} if user_input is not None: try: user_input[CONF_URL] = _normalize_url(user_input[CONF_URL]) info = await _validate_input(self.hass, user_input) except FibaroConnectFailed: errors["base"] = "cannot_connect" except FibaroAuthFailed: errors["base"] = "invalid_auth" else: await self.async_set_unique_id(info["serial_number"]) self._abort_if_unique_id_configured(updates=user_input) return self.async_create_entry(title=info["name"], data=user_input) return self.async_show_form( step_id="user", data_schema=STEP_USER_DATA_SCHEMA, errors=errors ) async def async_step_reauth( self, entry_data: Mapping[str, Any] ) -> ConfigFlowResult: """Handle reauthentication.""" self._reauth_entry = self.hass.config_entries.async_get_entry( self.context["entry_id"] ) return await self.async_step_reauth_confirm() async def async_step_reauth_confirm( self, user_input: dict[str, Any] | None = None ) -> ConfigFlowResult: """Handle a flow initiated by reauthentication.""" errors = {} assert self._reauth_entry if user_input is not None: new_data = self._reauth_entry.data | user_input try: await _validate_input(self.hass, new_data) except FibaroConnectFailed: errors["base"] = "cannot_connect" except FibaroAuthFailed: errors["base"] = "invalid_auth" else: self.hass.config_entries.async_update_entry( self._reauth_entry, data=new_data ) self.hass.async_create_task( self.hass.config_entries.async_reload(self._reauth_entry.entry_id) ) return self.async_abort(reason="reauth_successful") return self.async_show_form( step_id="reauth_confirm", data_schema=vol.Schema({vol.Required(CONF_PASSWORD): str}), errors=errors, description_placeholders={ CONF_USERNAME: self._reauth_entry.data[CONF_USERNAME] }, )