core/homeassistant/components/purpleair/config_flow.py

474 lines
16 KiB
Python
Raw Normal View History

2022-12-13 03:32:11 +00:00
"""Config flow for PurpleAir integration."""
from __future__ import annotations
import asyncio
from collections.abc import Mapping
from copy import deepcopy
2022-12-13 03:32:11 +00:00
from dataclasses import dataclass, field
from typing import Any, cast
2022-12-13 03:32:11 +00:00
from aiopurpleair import API
from aiopurpleair.endpoints.sensors import NearbySensorResult
2022-12-13 03:32:11 +00:00
from aiopurpleair.errors import InvalidApiKeyError, PurpleAirError
import voluptuous as vol
from homeassistant import config_entries
from homeassistant.config_entries import ConfigEntry
2022-12-13 03:32:11 +00:00
from homeassistant.const import CONF_API_KEY, CONF_LATITUDE, CONF_LONGITUDE
2023-07-24 11:18:38 +00:00
from homeassistant.core import HomeAssistant, callback
2022-12-13 03:32:11 +00:00
from homeassistant.data_entry_flow import FlowResult
from homeassistant.helpers import (
aiohttp_client,
config_validation as cv,
device_registry as dr,
entity_registry as er,
)
2023-07-24 11:18:38 +00:00
from homeassistant.helpers.event import (
EventStateChangedData,
async_track_state_change_event,
)
2022-12-13 03:32:11 +00:00
from homeassistant.helpers.selector import (
SelectOptionDict,
SelectSelector,
SelectSelectorConfig,
SelectSelectorMode,
)
2023-07-24 11:18:38 +00:00
from homeassistant.helpers.typing import EventType
2022-12-13 03:32:11 +00:00
from .const import CONF_SENSOR_INDICES, CONF_SHOW_ON_MAP, DOMAIN, LOGGER
2022-12-13 03:32:11 +00:00
CONF_DISTANCE = "distance"
CONF_NEARBY_SENSOR_OPTIONS = "nearby_sensor_options"
CONF_SENSOR_DEVICE_ID = "sensor_device_id"
2022-12-13 03:32:11 +00:00
CONF_SENSOR_INDEX = "sensor_index"
DEFAULT_DISTANCE = 5
API_KEY_SCHEMA = vol.Schema(
{
vol.Required(CONF_API_KEY): cv.string,
}
)
@callback
def async_get_api(hass: HomeAssistant, api_key: str) -> API:
"""Get an aiopurpleair API object."""
session = aiohttp_client.async_get_clientsession(hass)
return API(api_key, session=session)
@callback
def async_get_coordinates_schema(hass: HomeAssistant) -> vol.Schema:
"""Define a schema for searching for sensors near a coordinate pair."""
return vol.Schema(
{
vol.Inclusive(
CONF_LATITUDE, "coords", default=hass.config.latitude
): cv.latitude,
vol.Inclusive(
CONF_LONGITUDE, "coords", default=hass.config.longitude
): cv.longitude,
vol.Optional(CONF_DISTANCE, default=DEFAULT_DISTANCE): cv.positive_int,
}
)
@callback
def async_get_nearby_sensors_options(
nearby_sensor_results: list[NearbySensorResult],
) -> list[SelectOptionDict]:
"""Return a set of nearby sensors as SelectOptionDict objects."""
return [
SelectOptionDict(
value=str(result.sensor.sensor_index), label=cast(str, result.sensor.name)
)
for result in nearby_sensor_results
]
2022-12-13 03:32:11 +00:00
@callback
def async_get_nearby_sensors_schema(options: list[SelectOptionDict]) -> vol.Schema:
"""Define a schema for selecting a sensor from a list."""
return vol.Schema(
{
vol.Required(CONF_SENSOR_INDEX): SelectSelector(
SelectSelectorConfig(options=options, mode=SelectSelectorMode.DROPDOWN)
)
}
)
@callback
def async_get_remove_sensor_options(
hass: HomeAssistant, config_entry: ConfigEntry
) -> list[SelectOptionDict]:
"""Return a set of already-configured sensors as SelectOptionDict objects."""
device_registry = dr.async_get(hass)
return [
SelectOptionDict(value=device_entry.id, label=cast(str, device_entry.name))
for device_entry in device_registry.devices.values()
if config_entry.entry_id in device_entry.config_entries
]
@callback
def async_get_remove_sensor_schema(sensors: list[SelectOptionDict]) -> vol.Schema:
"""Define a schema removing a sensor."""
return vol.Schema(
{
vol.Required(CONF_SENSOR_DEVICE_ID): SelectSelector(
SelectSelectorConfig(options=sensors, mode=SelectSelectorMode.DROPDOWN)
)
}
)
2022-12-13 03:32:11 +00:00
@dataclass
class ValidationResult:
"""Define a validation result."""
data: Any = None
errors: dict[str, Any] = field(default_factory=dict)
async def async_validate_api_key(hass: HomeAssistant, api_key: str) -> ValidationResult:
"""Validate an API key.
This method returns a dictionary of errors (if appropriate).
"""
api = async_get_api(hass, api_key)
errors = {}
try:
await api.async_check_api_key()
except InvalidApiKeyError:
errors["base"] = "invalid_api_key"
except PurpleAirError as err:
LOGGER.error("PurpleAir error while checking API key: %s", err)
errors["base"] = "unknown"
except Exception as err: # pylint: disable=broad-except
LOGGER.exception("Unexpected exception while checking API key: %s", err)
errors["base"] = "unknown"
if errors:
return ValidationResult(errors=errors)
return ValidationResult(data=None)
async def async_validate_coordinates(
hass: HomeAssistant,
api_key: str,
latitude: float,
longitude: float,
distance: float,
) -> ValidationResult:
"""Validate coordinates."""
api = async_get_api(hass, api_key)
errors = {}
try:
nearby_sensor_results = await api.sensors.async_get_nearby_sensors(
["name"], latitude, longitude, distance, limit_results=5
)
except PurpleAirError as err:
LOGGER.error("PurpleAir error while getting nearby sensors: %s", err)
errors["base"] = "unknown"
except Exception as err: # pylint: disable=broad-except
LOGGER.exception("Unexpected exception while getting nearby sensors: %s", err)
errors["base"] = "unknown"
else:
if not nearby_sensor_results:
errors["base"] = "no_sensors_near_coordinates"
if errors:
return ValidationResult(errors=errors)
return ValidationResult(data=nearby_sensor_results)
class ConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
"""Handle a config flow for PurpleAir."""
VERSION = 1
def __init__(self) -> None:
"""Initialize."""
self._flow_data: dict[str, Any] = {}
self._reauth_entry: ConfigEntry | None = None
2022-12-13 03:32:11 +00:00
@staticmethod
@callback
def async_get_options_flow(
config_entry: ConfigEntry,
) -> PurpleAirOptionsFlowHandler:
"""Define the config flow to handle options."""
return PurpleAirOptionsFlowHandler(config_entry)
2022-12-13 03:32:11 +00:00
async def async_step_by_coordinates(
self, user_input: dict[str, Any] | None = None
) -> FlowResult:
"""Handle the discovery of sensors near a latitude/longitude."""
if user_input is None:
return self.async_show_form(
step_id="by_coordinates",
data_schema=async_get_coordinates_schema(self.hass),
)
validation = await async_validate_coordinates(
self.hass,
self._flow_data[CONF_API_KEY],
user_input[CONF_LATITUDE],
user_input[CONF_LONGITUDE],
user_input[CONF_DISTANCE],
)
if validation.errors:
return self.async_show_form(
step_id="by_coordinates",
data_schema=async_get_coordinates_schema(self.hass),
errors=validation.errors,
)
self._flow_data[CONF_NEARBY_SENSOR_OPTIONS] = async_get_nearby_sensors_options(
validation.data
)
2022-12-13 03:32:11 +00:00
return await self.async_step_choose_sensor()
async def async_step_choose_sensor(
self, user_input: dict[str, Any] | None = None
) -> FlowResult:
"""Handle the selection of a sensor."""
if user_input is None:
options = self._flow_data.pop(CONF_NEARBY_SENSOR_OPTIONS)
return self.async_show_form(
step_id="choose_sensor",
data_schema=async_get_nearby_sensors_schema(options),
)
return self.async_create_entry(
title=self._flow_data[CONF_API_KEY][:5],
data=self._flow_data,
# Note that we store the sensor indices in options so that later on, we can
# add/remove additional sensors via an options flow:
options={CONF_SENSOR_INDICES: [int(user_input[CONF_SENSOR_INDEX])]},
)
async def async_step_reauth(self, entry_data: Mapping[str, Any]) -> FlowResult:
"""Handle configuration by re-auth."""
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
) -> FlowResult:
"""Handle the re-auth step."""
if user_input is None:
return self.async_show_form(
step_id="reauth_confirm", data_schema=API_KEY_SCHEMA
)
api_key = user_input[CONF_API_KEY]
validation = await async_validate_api_key(self.hass, api_key)
if validation.errors:
return self.async_show_form(
step_id="reauth_confirm",
data_schema=API_KEY_SCHEMA,
errors=validation.errors,
)
assert self._reauth_entry
self.hass.config_entries.async_update_entry(
self._reauth_entry, data={CONF_API_KEY: api_key}
)
self.hass.async_create_task(
self.hass.config_entries.async_reload(self._reauth_entry.entry_id)
)
return self.async_abort(reason="reauth_successful")
2022-12-13 03:32:11 +00:00
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=API_KEY_SCHEMA)
api_key = user_input[CONF_API_KEY]
self._async_abort_entries_match({CONF_API_KEY: api_key})
2022-12-13 03:32:11 +00:00
validation = await async_validate_api_key(self.hass, api_key)
if validation.errors:
return self.async_show_form(
step_id="user",
data_schema=API_KEY_SCHEMA,
errors=validation.errors,
2022-12-13 03:32:11 +00:00
)
self._flow_data = {CONF_API_KEY: api_key}
return await self.async_step_by_coordinates()
class PurpleAirOptionsFlowHandler(config_entries.OptionsFlow):
"""Handle a PurpleAir options flow."""
def __init__(self, config_entry: ConfigEntry) -> None:
"""Initialize."""
self._flow_data: dict[str, Any] = {}
self.config_entry = config_entry
@property
def settings_schema(self) -> vol.Schema:
"""Return the settings schema."""
return vol.Schema(
{
vol.Optional(
CONF_SHOW_ON_MAP,
description={
"suggested_value": self.config_entry.options.get(
CONF_SHOW_ON_MAP
)
},
): bool
}
)
async def async_step_add_sensor(
self, user_input: dict[str, Any] | None = None
) -> FlowResult:
"""Add a sensor."""
if user_input is None:
return self.async_show_form(
step_id="add_sensor",
data_schema=async_get_coordinates_schema(self.hass),
)
validation = await async_validate_coordinates(
self.hass,
self.config_entry.data[CONF_API_KEY],
user_input[CONF_LATITUDE],
user_input[CONF_LONGITUDE],
user_input[CONF_DISTANCE],
)
if validation.errors:
return self.async_show_form(
step_id="add_sensor",
data_schema=async_get_coordinates_schema(self.hass),
errors=validation.errors,
)
self._flow_data[CONF_NEARBY_SENSOR_OPTIONS] = async_get_nearby_sensors_options(
validation.data
)
return await self.async_step_choose_sensor()
async def async_step_choose_sensor(
self, user_input: dict[str, Any] | None = None
) -> FlowResult:
"""Choose a sensor."""
if user_input is None:
options = self._flow_data.pop(CONF_NEARBY_SENSOR_OPTIONS)
return self.async_show_form(
step_id="choose_sensor",
data_schema=async_get_nearby_sensors_schema(options),
)
sensor_index = int(user_input[CONF_SENSOR_INDEX])
if sensor_index in self.config_entry.options[CONF_SENSOR_INDICES]:
return self.async_abort(reason="already_configured")
options = deepcopy({**self.config_entry.options})
options[CONF_SENSOR_INDICES].append(sensor_index)
return self.async_create_entry(data=options)
async def async_step_init(
self, user_input: dict[str, Any] | None = None
) -> FlowResult:
"""Manage the options."""
return self.async_show_menu(
step_id="init",
menu_options=["add_sensor", "remove_sensor", "settings"],
)
async def async_step_remove_sensor(
self, user_input: dict[str, Any] | None = None
) -> FlowResult:
"""Remove a sensor."""
if user_input is None:
return self.async_show_form(
step_id="remove_sensor",
data_schema=async_get_remove_sensor_schema(
async_get_remove_sensor_options(self.hass, self.config_entry)
),
)
device_registry = dr.async_get(self.hass)
entity_registry = er.async_get(self.hass)
device_id = user_input[CONF_SENSOR_DEVICE_ID]
device_entry = cast(dr.DeviceEntry, device_registry.async_get(device_id))
# Determine the entity entries that belong to this device.
entity_entries = er.async_entries_for_device(
entity_registry, device_id, include_disabled_entities=True
)
device_entities_removed_event = asyncio.Event()
@callback
2023-07-24 11:18:38 +00:00
def async_device_entity_state_changed(
_: EventType[EventStateChangedData],
) -> None:
"""Listen and respond when all device entities are removed."""
if all(
self.hass.states.get(entity_entry.entity_id) is None
for entity_entry in entity_entries
):
device_entities_removed_event.set()
# Track state changes for this device's entities and when they're removed,
# finish the flow:
cancel_state_track = async_track_state_change_event(
self.hass,
[entity_entry.entity_id for entity_entry in entity_entries],
async_device_entity_state_changed,
)
device_registry.async_update_device(
device_id, remove_config_entry_id=self.config_entry.entry_id
)
await device_entities_removed_event.wait()
# Once we're done, we can cancel the state change tracker callback:
cancel_state_track()
# Build new config entry options:
removed_sensor_index = next(
sensor_index
for sensor_index in self.config_entry.options[CONF_SENSOR_INDICES]
if (DOMAIN, str(sensor_index)) in device_entry.identifiers
)
options = deepcopy({**self.config_entry.options})
options[CONF_SENSOR_INDICES].remove(removed_sensor_index)
return self.async_create_entry(data=options)
async def async_step_settings(
self, user_input: dict[str, Any] | None = None
) -> FlowResult:
"""Manage settings."""
if user_input is None:
return self.async_show_form(
step_id="settings", data_schema=self.settings_schema
)
options = deepcopy({**self.config_entry.options})
return self.async_create_entry(data=options | user_input)