"""Config flow for ozw integration.""" import logging import voluptuous as vol from homeassistant import config_entries from homeassistant.core import callback from homeassistant.data_entry_flow import AbortFlow from .const import CONF_INTEGRATION_CREATED_ADDON from .const import DOMAIN # pylint:disable=unused-import _LOGGER = logging.getLogger(__name__) CONF_ADDON_DEVICE = "device" CONF_ADDON_NETWORK_KEY = "network_key" CONF_NETWORK_KEY = "network_key" CONF_USB_PATH = "usb_path" CONF_USE_ADDON = "use_addon" TITLE = "OpenZWave" ON_SUPERVISOR_SCHEMA = vol.Schema({vol.Optional(CONF_USE_ADDON, default=False): bool}) class DomainConfigFlow(config_entries.ConfigFlow, domain=DOMAIN): """Handle a config flow for ozw.""" VERSION = 1 CONNECTION_CLASS = config_entries.CONN_CLASS_LOCAL_PUSH def __init__(self): """Set up flow instance.""" self.addon_config = None self.network_key = None self.usb_path = None self.use_addon = False # If we install the add-on we should uninstall it on entry remove. self.integration_created_addon = False async def async_step_user(self, user_input=None): """Handle the initial step.""" if self._async_current_entries(): return self.async_abort(reason="single_instance_allowed") # Currently all flow results need the MQTT integration. # This will change when we have the direct MQTT client connection. # When that is implemented, move this check to _async_use_mqtt_integration. if "mqtt" not in self.hass.config.components: return self.async_abort(reason="mqtt_required") if not self.hass.components.hassio.is_hassio(): return self._async_use_mqtt_integration() return await self.async_step_on_supervisor() def _async_create_entry_from_vars(self): """Return a config entry for the flow.""" return self.async_create_entry( title=TITLE, data={ CONF_USB_PATH: self.usb_path, CONF_NETWORK_KEY: self.network_key, CONF_USE_ADDON: self.use_addon, CONF_INTEGRATION_CREATED_ADDON: self.integration_created_addon, }, ) @callback def _async_use_mqtt_integration(self): """Handle logic when using the MQTT integration. This is the entry point for the logic that is needed when this integration will depend on the MQTT integration. """ return self._async_create_entry_from_vars() async def async_step_on_supervisor(self, user_input=None): """Handle logic when on Supervisor host.""" if user_input is None: return self.async_show_form( step_id="on_supervisor", data_schema=ON_SUPERVISOR_SCHEMA ) if not user_input[CONF_USE_ADDON]: return self._async_create_entry_from_vars() self.use_addon = True if await self._async_is_addon_running(): return self._async_create_entry_from_vars() if await self._async_is_addon_installed(): return await self.async_step_start_addon() return await self.async_step_install_addon() async def async_step_install_addon(self): """Install OpenZWave add-on.""" try: await self.hass.components.hassio.async_install_addon("core_zwave") except self.hass.components.hassio.HassioAPIError as err: _LOGGER.error("Failed to install OpenZWave add-on: %s", err) return self.async_abort(reason="addon_install_failed") self.integration_created_addon = True return await self.async_step_start_addon() async def async_step_start_addon(self, user_input=None): """Ask for config and start OpenZWave add-on.""" if self.addon_config is None: self.addon_config = await self._async_get_addon_config() errors = {} if user_input is not None: self.network_key = user_input[CONF_NETWORK_KEY] self.usb_path = user_input[CONF_USB_PATH] new_addon_config = {CONF_ADDON_DEVICE: self.usb_path} if self.network_key: new_addon_config[CONF_ADDON_NETWORK_KEY] = self.network_key if new_addon_config != self.addon_config: await self._async_set_addon_config(new_addon_config) try: await self.hass.components.hassio.async_start_addon("core_zwave") except self.hass.components.hassio.HassioAPIError as err: _LOGGER.error("Failed to start OpenZWave add-on: %s", err) errors["base"] = "addon_start_failed" else: return self._async_create_entry_from_vars() self.usb_path = self.addon_config.get(CONF_ADDON_DEVICE, "") self.network_key = self.addon_config.get(CONF_ADDON_NETWORK_KEY, "") data_schema = vol.Schema( { vol.Required(CONF_USB_PATH, default=self.usb_path): str, vol.Optional(CONF_NETWORK_KEY, default=self.network_key): str, } ) return self.async_show_form( step_id="start_addon", data_schema=data_schema, errors=errors ) async def _async_get_addon_info(self): """Return and cache OpenZWave add-on info.""" try: addon_info = await self.hass.components.hassio.async_get_addon_info( "core_zwave" ) except self.hass.components.hassio.HassioAPIError as err: _LOGGER.error("Failed to get OpenZWave add-on info: %s", err) raise AbortFlow("addon_info_failed") from err return addon_info async def _async_is_addon_running(self): """Return True if OpenZWave add-on is running.""" addon_info = await self._async_get_addon_info() return addon_info["state"] == "started" async def _async_is_addon_installed(self): """Return True if OpenZWave add-on is installed.""" addon_info = await self._async_get_addon_info() return addon_info["version"] is not None async def _async_get_addon_config(self): """Get OpenZWave add-on config.""" addon_info = await self._async_get_addon_info() return addon_info["options"] async def _async_set_addon_config(self, config): """Set OpenZWave add-on config.""" options = {"options": config} try: await self.hass.components.hassio.async_set_addon_options( "core_zwave", options ) except self.hass.components.hassio.HassioAPIError as err: _LOGGER.error("Failed to set OpenZWave add-on config: %s", err) raise AbortFlow("addon_set_config_failed") from err