core/homeassistant/components/co2signal/config_flow.py

188 lines
6.1 KiB
Python

"""Config flow for Co2signal integration."""
from __future__ import annotations
from typing import Any
import voluptuous as vol
from homeassistant import config_entries
from homeassistant.const import CONF_API_KEY, CONF_LATITUDE, CONF_LONGITUDE, CONF_TOKEN
from homeassistant.data_entry_flow import FlowResult
import homeassistant.helpers.config_validation as cv
from . import APIRatelimitExceeded, CO2Error, InvalidAuth, get_data
from .const import CONF_COUNTRY_CODE, DOMAIN
from .util import get_extra_name
TYPE_USE_HOME = "Use home location"
TYPE_SPECIFY_COORDINATES = "Specify coordinates"
TYPE_SPECIFY_COUNTRY = "Specify country code"
def _get_entry_type(config: dict) -> str:
"""Get entry type from the configuration."""
if CONF_LATITUDE in config:
return TYPE_SPECIFY_COORDINATES
if CONF_COUNTRY_CODE in config:
return TYPE_SPECIFY_COUNTRY
return TYPE_USE_HOME
class ConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
"""Handle a config flow for Co2signal."""
VERSION = 1
_data: dict | None
async def async_step_import(self, import_info):
"""Set the config entry up from yaml."""
data = {CONF_API_KEY: import_info[CONF_TOKEN]}
if CONF_COUNTRY_CODE in import_info:
data[CONF_COUNTRY_CODE] = import_info[CONF_COUNTRY_CODE]
new_entry_type = TYPE_SPECIFY_COUNTRY
elif (
CONF_LATITUDE in import_info
and import_info[CONF_LATITUDE] != self.hass.config.latitude
and import_info[CONF_LONGITUDE] != self.hass.config.longitude
):
data[CONF_LATITUDE] = import_info[CONF_LATITUDE]
data[CONF_LONGITUDE] = import_info[CONF_LONGITUDE]
new_entry_type = TYPE_SPECIFY_COORDINATES
else:
new_entry_type = TYPE_USE_HOME
for entry in self._async_current_entries(include_ignore=False):
if (cur_entry_type := _get_entry_type(entry.data)) != new_entry_type:
continue
if cur_entry_type == TYPE_USE_HOME and new_entry_type == TYPE_USE_HOME:
return self.async_abort(reason="already_configured")
if (
cur_entry_type == TYPE_SPECIFY_COUNTRY
and data[CONF_COUNTRY_CODE] == entry.data[CONF_COUNTRY_CODE]
):
return self.async_abort(reason="already_configured")
if (
cur_entry_type == TYPE_SPECIFY_COORDINATES
and data[CONF_LATITUDE] == entry.data[CONF_LATITUDE]
and data[CONF_LONGITUDE] == entry.data[CONF_LONGITUDE]
):
return self.async_abort(reason="already_configured")
try:
await self.hass.async_add_executor_job(get_data, self.hass, data)
except CO2Error:
return self.async_abort(reason="unknown")
return self.async_create_entry(
title=get_extra_name(data) or "CO2 Signal", data=data
)
async def async_step_user(
self, user_input: dict[str, Any] | None = None
) -> FlowResult:
"""Handle the initial step."""
data_schema = vol.Schema(
{
vol.Required("location", default=TYPE_USE_HOME): vol.In(
(
TYPE_USE_HOME,
TYPE_SPECIFY_COORDINATES,
TYPE_SPECIFY_COUNTRY,
)
),
vol.Required(CONF_API_KEY): cv.string,
}
)
if user_input is None:
return self.async_show_form(
step_id="user",
data_schema=data_schema,
)
data = {CONF_API_KEY: user_input[CONF_API_KEY]}
if user_input["location"] == TYPE_SPECIFY_COORDINATES:
self._data = data
return await self.async_step_coordinates()
if user_input["location"] == TYPE_SPECIFY_COUNTRY:
self._data = data
return await self.async_step_country()
return await self._validate_and_create("user", data_schema, data)
async def async_step_coordinates(
self, user_input: dict[str, Any] | None = None
) -> FlowResult:
"""Validate coordinates."""
data_schema = vol.Schema(
{
vol.Required(
CONF_LATITUDE,
): cv.latitude,
vol.Required(
CONF_LONGITUDE,
): cv.longitude,
}
)
if user_input is None:
return self.async_show_form(step_id="coordinates", data_schema=data_schema)
assert self._data is not None
return await self._validate_and_create(
"coordinates", data_schema, {**self._data, **user_input}
)
async def async_step_country(
self, user_input: dict[str, Any] | None = None
) -> FlowResult:
"""Validate country."""
data_schema = vol.Schema(
{
vol.Required(CONF_COUNTRY_CODE): cv.string,
}
)
if user_input is None:
return self.async_show_form(step_id="country", data_schema=data_schema)
assert self._data is not None
return await self._validate_and_create(
"country", data_schema, {**self._data, **user_input}
)
async def _validate_and_create(
self, step_id: str, data_schema: vol.Schema, data: dict
) -> FlowResult:
"""Validate data and show form if it is invalid."""
errors: dict[str, str] = {}
try:
await self.hass.async_add_executor_job(get_data, self.hass, data)
except InvalidAuth:
errors["base"] = "invalid_auth"
except APIRatelimitExceeded:
errors["base"] = "api_ratelimit"
except Exception: # pylint: disable=broad-except
errors["base"] = "unknown"
else:
return self.async_create_entry(
title=get_extra_name(data) or "CO2 Signal",
data=data,
)
return self.async_show_form(
step_id=step_id,
data_schema=data_schema,
errors=errors,
)