2019-04-09 12:26:58 +00:00
|
|
|
"""Config flow to configure Logi Circle component."""
|
|
|
|
import asyncio
|
|
|
|
from collections import OrderedDict
|
2021-09-22 18:59:52 +00:00
|
|
|
from http import HTTPStatus
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
import async_timeout
|
2019-12-05 05:14:57 +00:00
|
|
|
from logi_circle import LogiCircle
|
|
|
|
from logi_circle.exception import AuthorizationFailed
|
2019-04-09 12:26:58 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant import config_entries
|
|
|
|
from homeassistant.components.http import HomeAssistantView
|
2020-05-30 15:27:20 +00:00
|
|
|
from homeassistant.const import (
|
2021-02-11 02:20:40 +00:00
|
|
|
CONF_API_KEY,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_CLIENT_ID,
|
|
|
|
CONF_CLIENT_SECRET,
|
2020-05-30 15:27:20 +00:00
|
|
|
CONF_SENSORS,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2020-05-30 15:27:20 +00:00
|
|
|
from homeassistant.core import callback
|
|
|
|
|
2021-02-11 02:20:40 +00:00
|
|
|
from .const import CONF_REDIRECT_URI, DEFAULT_CACHEDB, DOMAIN
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
_TIMEOUT = 15 # seconds
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DATA_FLOW_IMPL = "logi_circle_flow_implementation"
|
|
|
|
EXTERNAL_ERRORS = "logi_errors"
|
|
|
|
AUTH_CALLBACK_PATH = "/api/logi_circle"
|
|
|
|
AUTH_CALLBACK_NAME = "api:logi_circle"
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
@callback
|
2019-07-31 19:25:30 +00:00
|
|
|
def register_flow_implementation(
|
|
|
|
hass, domain, client_id, client_secret, api_key, redirect_uri, sensors
|
|
|
|
):
|
2019-04-09 12:26:58 +00:00
|
|
|
"""Register a flow implementation.
|
|
|
|
|
|
|
|
domain: Domain of the component responsible for the implementation.
|
|
|
|
client_id: Client ID.
|
|
|
|
client_secret: Client secret.
|
|
|
|
api_key: API key issued by Logitech.
|
|
|
|
redirect_uri: Auth callback redirect URI.
|
|
|
|
sensors: Sensor config.
|
|
|
|
"""
|
|
|
|
if DATA_FLOW_IMPL not in hass.data:
|
|
|
|
hass.data[DATA_FLOW_IMPL] = OrderedDict()
|
|
|
|
|
|
|
|
hass.data[DATA_FLOW_IMPL][domain] = {
|
|
|
|
CONF_CLIENT_ID: client_id,
|
|
|
|
CONF_CLIENT_SECRET: client_secret,
|
|
|
|
CONF_API_KEY: api_key,
|
|
|
|
CONF_REDIRECT_URI: redirect_uri,
|
|
|
|
CONF_SENSORS: sensors,
|
2019-07-31 19:25:30 +00:00
|
|
|
EXTERNAL_ERRORS: None,
|
2019-04-09 12:26:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-04-30 21:28:25 +00:00
|
|
|
class LogiCircleFlowHandler(config_entries.ConfigFlow, domain=DOMAIN):
|
2019-04-09 12:26:58 +00:00
|
|
|
"""Config flow for Logi Circle component."""
|
|
|
|
|
|
|
|
VERSION = 1
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
"""Initialize flow."""
|
|
|
|
self.flow_impl = None
|
|
|
|
|
|
|
|
async def async_step_import(self, user_input=None):
|
|
|
|
"""Handle external yaml configuration."""
|
2021-05-11 20:00:12 +00:00
|
|
|
self._async_abort_entries_match()
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
self.flow_impl = DOMAIN
|
|
|
|
|
|
|
|
return await self.async_step_auth()
|
|
|
|
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
|
|
"""Handle a flow start."""
|
|
|
|
flows = self.hass.data.get(DATA_FLOW_IMPL, {})
|
|
|
|
|
2021-05-11 20:00:12 +00:00
|
|
|
self._async_abort_entries_match()
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
if not flows:
|
2020-10-11 15:11:45 +00:00
|
|
|
return self.async_abort(reason="missing_configuration")
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
if len(flows) == 1:
|
|
|
|
self.flow_impl = list(flows)[0]
|
|
|
|
return await self.async_step_auth()
|
|
|
|
|
|
|
|
if user_input is not None:
|
2019-07-31 19:25:30 +00:00
|
|
|
self.flow_impl = user_input["flow_impl"]
|
2019-04-09 12:26:58 +00:00
|
|
|
return await self.async_step_auth()
|
|
|
|
|
|
|
|
return self.async_show_form(
|
2019-07-31 19:25:30 +00:00
|
|
|
step_id="user",
|
|
|
|
data_schema=vol.Schema({vol.Required("flow_impl"): vol.In(list(flows))}),
|
|
|
|
)
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
async def async_step_auth(self, user_input=None):
|
|
|
|
"""Create an entry for auth."""
|
2021-05-12 10:47:06 +00:00
|
|
|
if self._async_current_entries():
|
2019-07-31 19:25:30 +00:00
|
|
|
return self.async_abort(reason="external_setup")
|
2019-04-09 12:26:58 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
external_error = self.hass.data[DATA_FLOW_IMPL][DOMAIN][EXTERNAL_ERRORS]
|
2019-04-09 12:26:58 +00:00
|
|
|
errors = {}
|
|
|
|
if external_error:
|
|
|
|
# Handle error from another flow
|
2019-07-31 19:25:30 +00:00
|
|
|
errors["base"] = external_error
|
2019-04-09 12:26:58 +00:00
|
|
|
self.hass.data[DATA_FLOW_IMPL][DOMAIN][EXTERNAL_ERRORS] = None
|
|
|
|
elif user_input is not None:
|
2019-07-31 19:25:30 +00:00
|
|
|
errors["base"] = "follow_link"
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
url = self._get_authorization_url()
|
|
|
|
|
|
|
|
return self.async_show_form(
|
2019-07-31 19:25:30 +00:00
|
|
|
step_id="auth",
|
|
|
|
description_placeholders={"authorization_url": url},
|
|
|
|
errors=errors,
|
|
|
|
)
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
def _get_authorization_url(self):
|
|
|
|
"""Create temporary Circle session and generate authorization url."""
|
|
|
|
flow = self.hass.data[DATA_FLOW_IMPL][self.flow_impl]
|
|
|
|
client_id = flow[CONF_CLIENT_ID]
|
|
|
|
client_secret = flow[CONF_CLIENT_SECRET]
|
|
|
|
api_key = flow[CONF_API_KEY]
|
|
|
|
redirect_uri = flow[CONF_REDIRECT_URI]
|
|
|
|
|
|
|
|
logi_session = LogiCircle(
|
|
|
|
client_id=client_id,
|
|
|
|
client_secret=client_secret,
|
|
|
|
api_key=api_key,
|
2019-07-31 19:25:30 +00:00
|
|
|
redirect_uri=redirect_uri,
|
|
|
|
)
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
self.hass.http.register_view(LogiCircleAuthCallbackView())
|
|
|
|
|
|
|
|
return logi_session.authorize_url
|
|
|
|
|
|
|
|
async def async_step_code(self, code=None):
|
|
|
|
"""Received code for authentication."""
|
2021-05-11 20:00:12 +00:00
|
|
|
self._async_abort_entries_match()
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
return await self._async_create_session(code)
|
|
|
|
|
|
|
|
async def _async_create_session(self, code):
|
|
|
|
"""Create Logi Circle session and entries."""
|
|
|
|
flow = self.hass.data[DATA_FLOW_IMPL][DOMAIN]
|
|
|
|
client_id = flow[CONF_CLIENT_ID]
|
|
|
|
client_secret = flow[CONF_CLIENT_SECRET]
|
|
|
|
api_key = flow[CONF_API_KEY]
|
|
|
|
redirect_uri = flow[CONF_REDIRECT_URI]
|
|
|
|
sensors = flow[CONF_SENSORS]
|
|
|
|
|
|
|
|
logi_session = LogiCircle(
|
|
|
|
client_id=client_id,
|
|
|
|
client_secret=client_secret,
|
|
|
|
api_key=api_key,
|
|
|
|
redirect_uri=redirect_uri,
|
2019-07-31 19:25:30 +00:00
|
|
|
cache_file=self.hass.config.path(DEFAULT_CACHEDB),
|
|
|
|
)
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
try:
|
2021-11-04 15:07:50 +00:00
|
|
|
async with async_timeout.timeout(_TIMEOUT):
|
2019-04-09 12:26:58 +00:00
|
|
|
await logi_session.authorize(code)
|
|
|
|
except AuthorizationFailed:
|
2020-10-11 15:11:45 +00:00
|
|
|
(self.hass.data[DATA_FLOW_IMPL][DOMAIN][EXTERNAL_ERRORS]) = "invalid_auth"
|
2019-07-31 19:25:30 +00:00
|
|
|
return self.async_abort(reason="external_error")
|
2019-04-09 12:26:58 +00:00
|
|
|
except asyncio.TimeoutError:
|
2020-10-11 15:11:45 +00:00
|
|
|
(
|
|
|
|
self.hass.data[DATA_FLOW_IMPL][DOMAIN][EXTERNAL_ERRORS]
|
|
|
|
) = "authorize_url_timeout"
|
2019-07-31 19:25:30 +00:00
|
|
|
return self.async_abort(reason="external_error")
|
2019-04-09 12:26:58 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
account_id = (await logi_session.account)["accountId"]
|
2019-04-09 12:26:58 +00:00
|
|
|
await logi_session.close()
|
|
|
|
return self.async_create_entry(
|
2019-09-03 19:14:00 +00:00
|
|
|
title=f"Logi Circle ({account_id})",
|
2019-04-09 12:26:58 +00:00
|
|
|
data={
|
|
|
|
CONF_CLIENT_ID: client_id,
|
|
|
|
CONF_CLIENT_SECRET: client_secret,
|
|
|
|
CONF_API_KEY: api_key,
|
|
|
|
CONF_REDIRECT_URI: redirect_uri,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_SENSORS: sensors,
|
|
|
|
},
|
|
|
|
)
|
2019-04-09 12:26:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
class LogiCircleAuthCallbackView(HomeAssistantView):
|
|
|
|
"""Logi Circle Authorization Callback View."""
|
|
|
|
|
|
|
|
requires_auth = False
|
|
|
|
url = AUTH_CALLBACK_PATH
|
|
|
|
name = AUTH_CALLBACK_NAME
|
|
|
|
|
|
|
|
async def get(self, request):
|
|
|
|
"""Receive authorization code."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass = request.app["hass"]
|
|
|
|
if "code" in request.query:
|
2019-04-09 12:26:58 +00:00
|
|
|
hass.async_create_task(
|
|
|
|
hass.config_entries.flow.async_init(
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN, context={"source": "code"}, data=request.query["code"]
|
|
|
|
)
|
|
|
|
)
|
2019-04-09 12:26:58 +00:00
|
|
|
return self.json_message("Authorisation code saved")
|
2019-07-31 19:25:30 +00:00
|
|
|
return self.json_message(
|
2021-09-22 18:59:52 +00:00
|
|
|
"Authorisation code missing from query string",
|
|
|
|
status_code=HTTPStatus.BAD_REQUEST,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|