"""Config flow for IntelliFire integration.""" from __future__ import annotations from dataclasses import dataclass from typing import Any from aiohttp import ClientConnectionError from intellifire4py import AsyncUDPFireplaceFinder, IntellifireAsync import voluptuous as vol from homeassistant import config_entries from homeassistant.components.dhcp import DhcpServiceInfo from homeassistant.const import CONF_HOST from homeassistant.data_entry_flow import FlowResult from .const import DOMAIN, LOGGER STEP_USER_DATA_SCHEMA = vol.Schema({vol.Required(CONF_HOST): str}) MANUAL_ENTRY_STRING = "IP Address" # Simplified so it does not have to be translated @dataclass class DiscoveredHostInfo: """Host info for discovery.""" ip: str serial: str | None async def validate_host_input(host: str) -> str: """Validate the user input allows us to connect. Data has the keys from STEP_USER_DATA_SCHEMA with values provided by the user. """ api = IntellifireAsync(host) await api.poll() serial = api.data.serial LOGGER.debug("Found a fireplace: %s", serial) # Return the serial number which will be used to calculate a unique ID for the device/sensors return serial class ConfigFlow(config_entries.ConfigFlow, domain=DOMAIN): """Handle a config flow for IntelliFire.""" VERSION = 1 def __init__(self): """Initialize the Config Flow Handler.""" self._config_context = {} self._not_configured_hosts: list[DiscoveredHostInfo] = [] self._discovered_host: DiscoveredHostInfo async def _find_fireplaces(self): """Perform UDP discovery.""" fireplace_finder = AsyncUDPFireplaceFinder() discovered_hosts = await fireplace_finder.search_fireplace(timeout=1) configured_hosts = { entry.data[CONF_HOST] for entry in self._async_current_entries(include_ignore=False) if CONF_HOST in entry.data # CONF_HOST will be missing for ignored entries } self._not_configured_hosts = [ DiscoveredHostInfo(ip, None) for ip in discovered_hosts if ip not in configured_hosts ] LOGGER.debug("Discovered Hosts: %s", discovered_hosts) LOGGER.debug("Configured Hosts: %s", configured_hosts) LOGGER.debug("Not Configured Hosts: %s", self._not_configured_hosts) async def _async_validate_and_create_entry(self, host: str) -> FlowResult: """Validate and create the entry.""" self._async_abort_entries_match({CONF_HOST: host}) serial = await validate_host_input(host) await self.async_set_unique_id(serial, raise_on_progress=False) self._abort_if_unique_id_configured(updates={CONF_HOST: host}) return self.async_create_entry( title=f"Fireplace {serial}", data={CONF_HOST: host}, ) async def async_step_manual_device_entry(self, user_input=None): """Handle manual input of local IP configuration.""" errors = {} host = user_input.get(CONF_HOST) if user_input else None if user_input is not None: try: return await self._async_validate_and_create_entry(host) except (ConnectionError, ClientConnectionError): errors["base"] = "cannot_connect" return self.async_show_form( step_id="manual_device_entry", errors=errors, data_schema=vol.Schema({vol.Required(CONF_HOST, default=host): str}), ) async def async_step_pick_device( self, user_input: dict[str, Any] | None = None ) -> FlowResult: """Pick which device to configure.""" errors = {} if user_input is not None: if user_input[CONF_HOST] == MANUAL_ENTRY_STRING: return await self.async_step_manual_device_entry() try: return await self._async_validate_and_create_entry( user_input[CONF_HOST] ) except (ConnectionError, ClientConnectionError): errors["base"] = "cannot_connect" return self.async_show_form( step_id="pick_device", errors=errors, data_schema=vol.Schema( { vol.Required(CONF_HOST): vol.In( [host.ip for host in self._not_configured_hosts] + [MANUAL_ENTRY_STRING] ) } ), ) async def async_step_user( self, user_input: dict[str, Any] | None = None ) -> FlowResult: """Start the user flow.""" # Launch fireplaces discovery await self._find_fireplaces() if self._not_configured_hosts: LOGGER.debug("Running Step: pick_device") return await self.async_step_pick_device() LOGGER.debug("Running Step: manual_device_entry") return await self.async_step_manual_device_entry() async def async_step_dhcp(self, discovery_info: DhcpServiceInfo) -> FlowResult: """Handle DHCP Discovery.""" # Run validation logic on ip host = discovery_info.ip self._async_abort_entries_match({CONF_HOST: host}) try: serial = await validate_host_input(host) except (ConnectionError, ClientConnectionError): return self.async_abort(reason="not_intellifire_device") await self.async_set_unique_id(serial) self._abort_if_unique_id_configured(updates={CONF_HOST: host}) self._discovered_host = DiscoveredHostInfo(ip=host, serial=serial) placeholders = {CONF_HOST: host, "serial": serial} self.context["title_placeholders"] = placeholders self._set_confirm_only() return await self.async_step_dhcp_confirm() async def async_step_dhcp_confirm(self, user_input=None): """Attempt to confirm.""" # Add the hosts one by one host = self._discovered_host.ip serial = self._discovered_host.serial if user_input is None: # Show the confirmation dialog return self.async_show_form( step_id="dhcp_confirm", description_placeholders={CONF_HOST: host, "serial": serial}, ) return self.async_create_entry( title=f"Fireplace {serial}", data={CONF_HOST: host}, )