2022-12-13 03:32:11 +00:00
|
|
|
"""Config flow for PurpleAir integration."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
2022-12-14 19:29:07 +00:00
|
|
|
from collections.abc import Mapping
|
2022-12-18 18:30:05 +00:00
|
|
|
from copy import deepcopy
|
2022-12-13 03:32:11 +00:00
|
|
|
from dataclasses import dataclass, field
|
2022-12-18 18:30:05 +00:00
|
|
|
from typing import Any, cast
|
2022-12-13 03:32:11 +00:00
|
|
|
|
|
|
|
from aiopurpleair import API
|
2022-12-18 18:30:05 +00:00
|
|
|
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
|
2022-12-18 18:15:03 +00:00
|
|
|
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
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
|
|
from homeassistant.data_entry_flow import FlowResult
|
2022-12-18 18:30:05 +00:00
|
|
|
from homeassistant.helpers import (
|
|
|
|
aiohttp_client,
|
|
|
|
config_validation as cv,
|
|
|
|
device_registry as dr,
|
|
|
|
)
|
2022-12-13 03:32:11 +00:00
|
|
|
from homeassistant.helpers.selector import (
|
|
|
|
SelectOptionDict,
|
|
|
|
SelectSelector,
|
|
|
|
SelectSelectorConfig,
|
|
|
|
SelectSelectorMode,
|
|
|
|
)
|
|
|
|
|
2022-12-18 18:30:05 +00:00
|
|
|
from .const import CONF_LAST_UPDATE_SENSOR_ADD, CONF_SENSOR_INDICES, DOMAIN, LOGGER
|
2022-12-13 03:32:11 +00:00
|
|
|
|
|
|
|
CONF_DISTANCE = "distance"
|
|
|
|
CONF_NEARBY_SENSOR_OPTIONS = "nearby_sensor_options"
|
2022-12-18 18:30:05 +00:00
|
|
|
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,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-12-18 18:30:05 +00:00
|
|
|
@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(
|
2022-12-31 10:16:09 +00:00
|
|
|
value=str(result.sensor.sensor_index), label=cast(str, result.sensor.name)
|
2022-12-18 18:30:05 +00:00
|
|
|
)
|
|
|
|
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)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-12-18 18:30:05 +00:00
|
|
|
@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)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_get_sensor_index(
|
|
|
|
hass: HomeAssistant, config_entry: ConfigEntry, device_entry: dr.DeviceEntry
|
|
|
|
) -> int:
|
|
|
|
"""Get the sensor index related to a config and device entry.
|
|
|
|
|
|
|
|
Note that this method expects that there will always be a single sensor index per
|
|
|
|
DeviceEntry.
|
|
|
|
"""
|
2022-12-31 20:07:12 +00:00
|
|
|
sensor_index = next(
|
2022-12-18 18:30:05 +00:00
|
|
|
sensor_index
|
|
|
|
for sensor_index in config_entry.options[CONF_SENSOR_INDICES]
|
|
|
|
if (DOMAIN, str(sensor_index)) in device_entry.identifiers
|
2022-12-31 20:07:12 +00:00
|
|
|
)
|
2022-12-18 18:30:05 +00:00
|
|
|
|
|
|
|
return cast(int, sensor_index)
|
|
|
|
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_remove_sensor_by_device_id(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config_entry: ConfigEntry,
|
|
|
|
device_id: str,
|
|
|
|
*,
|
|
|
|
remove_device: bool = True,
|
|
|
|
) -> dict[str, Any]:
|
|
|
|
"""Remove a sensor and return update config entry options."""
|
|
|
|
device_registry = dr.async_get(hass)
|
|
|
|
device_entry = device_registry.async_get(device_id)
|
|
|
|
assert device_entry
|
|
|
|
|
|
|
|
removed_sensor_index = async_get_sensor_index(hass, config_entry, device_entry)
|
|
|
|
options = deepcopy({**config_entry.options})
|
|
|
|
options[CONF_LAST_UPDATE_SENSOR_ADD] = False
|
|
|
|
options[CONF_SENSOR_INDICES].remove(removed_sensor_index)
|
|
|
|
|
|
|
|
if remove_device:
|
|
|
|
device_registry.async_update_device(
|
|
|
|
device_entry.id, remove_config_entry_id=config_entry.entry_id
|
|
|
|
)
|
|
|
|
|
|
|
|
return options
|
|
|
|
|
|
|
|
|
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] = {}
|
2022-12-18 18:15:03 +00:00
|
|
|
self._reauth_entry: ConfigEntry | None = None
|
2022-12-13 03:32:11 +00:00
|
|
|
|
2022-12-18 18:30:05 +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,
|
|
|
|
)
|
|
|
|
|
2022-12-18 18:30:05 +00:00
|
|
|
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])]},
|
|
|
|
)
|
|
|
|
|
2022-12-14 19:29:07 +00:00
|
|
|
async def async_step_reauth(self, entry_data: Mapping[str, Any]) -> FlowResult:
|
|
|
|
"""Handle configuration by re-auth."""
|
2022-12-18 18:15:03 +00:00
|
|
|
self._reauth_entry = self.hass.config_entries.async_get_entry(
|
|
|
|
self.context["entry_id"]
|
|
|
|
)
|
2022-12-14 19:29:07 +00:00
|
|
|
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,
|
|
|
|
)
|
|
|
|
|
2022-12-18 18:15:03 +00:00
|
|
|
assert self._reauth_entry
|
|
|
|
|
2022-12-14 19:29:07 +00:00
|
|
|
self.hass.config_entries.async_update_entry(
|
2022-12-18 18:15:03 +00:00
|
|
|
self._reauth_entry, data={CONF_API_KEY: api_key}
|
2022-12-14 19:29:07 +00:00
|
|
|
)
|
|
|
|
self.hass.async_create_task(
|
2022-12-18 18:15:03 +00:00
|
|
|
self.hass.config_entries.async_reload(self._reauth_entry.entry_id)
|
2022-12-14 19:29:07 +00:00
|
|
|
)
|
|
|
|
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]
|
|
|
|
|
2022-12-14 19:29:07 +00:00
|
|
|
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(
|
2022-12-14 19:29:07 +00:00
|
|
|
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()
|
2022-12-18 18:30:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
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:
|
|
|
|
"""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),
|
|
|
|
)
|
|
|
|
|
|
|
|
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_LAST_UPDATE_SENSOR_ADD] = True
|
|
|
|
options[CONF_SENSOR_INDICES].append(sensor_index)
|
|
|
|
return self.async_create_entry(title="", 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"],
|
|
|
|
)
|
|
|
|
|
|
|
|
async def async_step_remove_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="remove_sensor",
|
|
|
|
data_schema=async_get_remove_sensor_schema(
|
|
|
|
async_get_remove_sensor_options(self.hass, self.config_entry)
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
new_entry_options = async_remove_sensor_by_device_id(
|
|
|
|
self.hass, self.config_entry, user_input[CONF_SENSOR_DEVICE_ID]
|
|
|
|
)
|
|
|
|
|
|
|
|
return self.async_create_entry(title="", data=new_entry_options)
|