175 lines
5.7 KiB
Python
175 lines
5.7 KiB
Python
"""Config flow for the Immich integration."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from collections.abc import Mapping
|
|
import logging
|
|
from typing import Any
|
|
|
|
from aioimmich import Immich
|
|
from aioimmich.const import CONNECT_ERRORS
|
|
from aioimmich.exceptions import ImmichUnauthorizedError
|
|
from aioimmich.users.models import ImmichUser
|
|
import voluptuous as vol
|
|
from yarl import URL
|
|
|
|
from homeassistant.config_entries import ConfigFlow, ConfigFlowResult
|
|
from homeassistant.const import (
|
|
CONF_API_KEY,
|
|
CONF_HOST,
|
|
CONF_PORT,
|
|
CONF_SSL,
|
|
CONF_URL,
|
|
CONF_VERIFY_SSL,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.exceptions import HomeAssistantError
|
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
|
from homeassistant.helpers.selector import (
|
|
TextSelector,
|
|
TextSelectorConfig,
|
|
TextSelectorType,
|
|
)
|
|
|
|
from .const import DEFAULT_VERIFY_SSL, DOMAIN
|
|
|
|
|
|
class InvalidUrl(HomeAssistantError):
|
|
"""Error to indicate invalid URL."""
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
STEP_USER_DATA_SCHEMA = vol.Schema(
|
|
{
|
|
vol.Required(CONF_URL): TextSelector(
|
|
config=TextSelectorConfig(type=TextSelectorType.URL)
|
|
),
|
|
vol.Required(CONF_API_KEY): TextSelector(
|
|
config=TextSelectorConfig(type=TextSelectorType.PASSWORD)
|
|
),
|
|
vol.Required(CONF_VERIFY_SSL, default=DEFAULT_VERIFY_SSL): bool,
|
|
}
|
|
)
|
|
|
|
|
|
def _parse_url(url: str) -> tuple[str, int, bool]:
|
|
"""Parse the URL and return host, port, and ssl."""
|
|
parsed_url = URL(url)
|
|
if (
|
|
(host := parsed_url.host) is None
|
|
or (port := parsed_url.port) is None
|
|
or (scheme := parsed_url.scheme) is None
|
|
):
|
|
raise InvalidUrl
|
|
return host, port, scheme == "https"
|
|
|
|
|
|
async def check_user_info(
|
|
hass: HomeAssistant, host: str, port: int, ssl: bool, verify_ssl: bool, api_key: str
|
|
) -> ImmichUser:
|
|
"""Test connection and fetch own user info."""
|
|
session = async_get_clientsession(hass, verify_ssl)
|
|
immich = Immich(session, api_key, host, port, ssl)
|
|
return await immich.users.async_get_my_user()
|
|
|
|
|
|
class ImmichConfigFlow(ConfigFlow, domain=DOMAIN):
|
|
"""Handle a config flow for Immich."""
|
|
|
|
VERSION = 1
|
|
|
|
_name: str
|
|
_current_data: Mapping[str, Any]
|
|
|
|
async def async_step_user(
|
|
self, user_input: dict[str, Any] | None = None
|
|
) -> ConfigFlowResult:
|
|
"""Handle the initial step."""
|
|
errors: dict[str, str] = {}
|
|
if user_input is not None:
|
|
try:
|
|
(host, port, ssl) = _parse_url(user_input[CONF_URL])
|
|
except InvalidUrl:
|
|
errors[CONF_URL] = "invalid_url"
|
|
else:
|
|
try:
|
|
my_user_info = await check_user_info(
|
|
self.hass,
|
|
host,
|
|
port,
|
|
ssl,
|
|
user_input[CONF_VERIFY_SSL],
|
|
user_input[CONF_API_KEY],
|
|
)
|
|
except ImmichUnauthorizedError:
|
|
errors["base"] = "invalid_auth"
|
|
except CONNECT_ERRORS:
|
|
errors["base"] = "cannot_connect"
|
|
except Exception:
|
|
_LOGGER.exception("Unexpected exception")
|
|
errors["base"] = "unknown"
|
|
else:
|
|
await self.async_set_unique_id(my_user_info.user_id)
|
|
self._abort_if_unique_id_configured()
|
|
return self.async_create_entry(
|
|
title=my_user_info.name,
|
|
data={
|
|
CONF_HOST: host,
|
|
CONF_PORT: port,
|
|
CONF_SSL: ssl,
|
|
CONF_VERIFY_SSL: user_input[CONF_VERIFY_SSL],
|
|
CONF_API_KEY: user_input[CONF_API_KEY],
|
|
},
|
|
)
|
|
|
|
return self.async_show_form(
|
|
step_id="user", data_schema=STEP_USER_DATA_SCHEMA, errors=errors
|
|
)
|
|
|
|
async def async_step_reauth(
|
|
self, entry_data: Mapping[str, Any]
|
|
) -> ConfigFlowResult:
|
|
"""Trigger a reauthentication flow."""
|
|
self._current_data = entry_data
|
|
self._name = entry_data[CONF_HOST]
|
|
|
|
return await self.async_step_reauth_confirm()
|
|
|
|
async def async_step_reauth_confirm(
|
|
self, user_input: dict[str, Any] | None = None
|
|
) -> ConfigFlowResult:
|
|
"""Handle reauthorization flow."""
|
|
errors = {}
|
|
|
|
if user_input is not None:
|
|
try:
|
|
my_user_info = await check_user_info(
|
|
self.hass,
|
|
self._current_data[CONF_HOST],
|
|
self._current_data[CONF_PORT],
|
|
self._current_data[CONF_SSL],
|
|
self._current_data[CONF_VERIFY_SSL],
|
|
user_input[CONF_API_KEY],
|
|
)
|
|
except ImmichUnauthorizedError:
|
|
errors["base"] = "invalid_auth"
|
|
except CONNECT_ERRORS:
|
|
errors["base"] = "cannot_connect"
|
|
except Exception:
|
|
_LOGGER.exception("Unexpected exception")
|
|
errors["base"] = "unknown"
|
|
else:
|
|
await self.async_set_unique_id(my_user_info.user_id)
|
|
self._abort_if_unique_id_mismatch()
|
|
return self.async_update_reload_and_abort(
|
|
self._get_reauth_entry(), data_updates=user_input
|
|
)
|
|
|
|
return self.async_show_form(
|
|
step_id="reauth_confirm",
|
|
data_schema=vol.Schema({vol.Required(CONF_API_KEY): str}),
|
|
description_placeholders={"name": self._name},
|
|
errors=errors,
|
|
)
|