"""Config flow for Hyundai / Kia Connect integration.""" from __future__ import annotations import hashlib import logging from typing import Any from hyundai_kia_connect_api import Token, VehicleManager import voluptuous as vol from homeassistant import config_entries from homeassistant.config_entries import ConfigEntry from homeassistant.const import ( CONF_PASSWORD, CONF_PIN, CONF_REGION, CONF_SCAN_INTERVAL, CONF_USERNAME, ) from homeassistant.core import HomeAssistant, callback from homeassistant.data_entry_flow import FlowResult from homeassistant.exceptions import HomeAssistantError from .const import ( BRANDS, CONF_BRAND, CONF_FORCE_REFRESH_INTERVAL, CONF_NO_FORCE_REFRESH_HOUR_FINISH, CONF_NO_FORCE_REFRESH_HOUR_START, DEFAULT_FORCE_REFRESH_INTERVAL, DEFAULT_NO_FORCE_REFRESH_HOUR_FINISH, DEFAULT_NO_FORCE_REFRESH_HOUR_START, DEFAULT_PIN, DEFAULT_SCAN_INTERVAL, DOMAIN, REGIONS, CONF_ENABLE_GEOLOCATION_ENTITY, CONF_USE_EMAIL_WITH_GEOCODE_API, DEFAULT_ENABLE_GEOLOCATION_ENTITY, DEFAULT_USE_EMAIL_WITH_GEOCODE_API, ) _LOGGER = logging.getLogger(__name__) STEP_USER_DATA_SCHEMA = vol.Schema( { vol.Required(CONF_USERNAME): str, vol.Required(CONF_PASSWORD): str, vol.Optional(CONF_PIN, default=DEFAULT_PIN): str, vol.Required(CONF_REGION): vol.In(REGIONS), vol.Required(CONF_BRAND): vol.In(BRANDS), } ) async def validate_input(hass: HomeAssistant, user_input: dict[str, Any]) -> Token: """Validate the user input allows us to connect.""" api = VehicleManager.get_implementation_by_region_brand( user_input[CONF_REGION], user_input[CONF_BRAND], ) token: Token = await hass.async_add_executor_job( api.login, user_input[CONF_USERNAME], user_input[CONF_PASSWORD] ) if token is None: raise InvalidAuth return token class HyundaiKiaConnectOptionFlowHandler(config_entries.OptionsFlow): """Handle an option flow for Hyundai / Kia Connect.""" def __init__(self, config_entry: ConfigEntry) -> None: """Initialize option flow instance.""" self.config_entry = config_entry self.schema = vol.Schema( { vol.Required( CONF_SCAN_INTERVAL, default=self.config_entry.options.get( CONF_SCAN_INTERVAL, DEFAULT_SCAN_INTERVAL ), ): vol.All(vol.Coerce(int), vol.Range(min=5, max=999)), vol.Required( CONF_FORCE_REFRESH_INTERVAL, default=self.config_entry.options.get( CONF_FORCE_REFRESH_INTERVAL, DEFAULT_FORCE_REFRESH_INTERVAL ), ): vol.All(vol.Coerce(int), vol.Range(min=30, max=999)), vol.Required( CONF_NO_FORCE_REFRESH_HOUR_START, default=self.config_entry.options.get( CONF_NO_FORCE_REFRESH_HOUR_START, DEFAULT_NO_FORCE_REFRESH_HOUR_START, ), ): vol.All(vol.Coerce(int), vol.Range(min=0, max=23)), vol.Required( CONF_NO_FORCE_REFRESH_HOUR_FINISH, default=self.config_entry.options.get( CONF_NO_FORCE_REFRESH_HOUR_FINISH, DEFAULT_NO_FORCE_REFRESH_HOUR_FINISH, ), ): vol.All(vol.Coerce(int), vol.Range(min=0, max=23)), vol.Optional( CONF_ENABLE_GEOLOCATION_ENTITY, default=self.config_entry.options.get( CONF_ENABLE_GEOLOCATION_ENTITY, DEFAULT_ENABLE_GEOLOCATION_ENTITY, ), ): bool, vol.Optional( CONF_USE_EMAIL_WITH_GEOCODE_API, default=self.config_entry.options.get( CONF_USE_EMAIL_WITH_GEOCODE_API, DEFAULT_USE_EMAIL_WITH_GEOCODE_API, ), ): bool, } ) async def async_step_init(self, user_input=None) -> FlowResult: """Handle options init setup.""" if user_input is not None: return self.async_create_entry( title=self.config_entry.title, data=user_input ) return self.async_show_form(step_id="init", data_schema=self.schema) class ConfigFlow(config_entries.ConfigFlow, domain=DOMAIN): """Handle a config flow for Hyundai / Kia Connect.""" VERSION = 2 @staticmethod @callback def async_get_options_flow(config_entry: ConfigEntry): """Initiate options flow instance.""" return HyundaiKiaConnectOptionFlowHandler(config_entry) async def async_step_user( self, user_input: dict[str, Any] | None = None ) -> FlowResult: """Handle the initial step.""" if user_input is None: return self.async_show_form( step_id="user", data_schema=STEP_USER_DATA_SCHEMA ) errors = {} try: await validate_input(self.hass, user_input) except InvalidAuth: errors["base"] = "invalid_auth" except Exception: # pylint: disable=broad-except _LOGGER.exception("Unexpected exception") errors["base"] = "unknown" else: title = f"{BRANDS[user_input[CONF_BRAND]]} {REGIONS[user_input[CONF_REGION]]} {user_input[CONF_USERNAME]}" await self.async_set_unique_id( hashlib.sha256(title.encode("utf-8")).hexdigest() ) self._abort_if_unique_id_configured() return self.async_create_entry(title=title, data=user_input) return self.async_show_form( step_id="user", data_schema=STEP_USER_DATA_SCHEMA, errors=errors ) class InvalidAuth(HomeAssistantError): """Error to indicate there is invalid auth."""