2019-10-29 06:47:31 +00:00
|
|
|
"""Config flow to connect with Home Assistant."""
|
2021-08-23 07:35:03 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2019-10-29 06:47:31 +00:00
|
|
|
import asyncio
|
|
|
|
import logging
|
2021-08-23 07:35:03 +00:00
|
|
|
from typing import Any
|
2019-10-29 06:47:31 +00:00
|
|
|
|
|
|
|
from aiohttp import ClientError
|
2019-12-09 10:27:16 +00:00
|
|
|
import async_timeout
|
2019-10-29 06:47:31 +00:00
|
|
|
from pyalmond import AlmondLocalAuth, WebAlmondAPI
|
2019-12-09 10:27:16 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
from yarl import URL
|
2019-10-29 06:47:31 +00:00
|
|
|
|
2019-12-09 10:27:16 +00:00
|
|
|
from homeassistant import config_entries, core, data_entry_flow
|
2021-12-03 13:05:56 +00:00
|
|
|
from homeassistant.components.hassio import HassioServiceInfo
|
2021-08-23 07:35:03 +00:00
|
|
|
from homeassistant.data_entry_flow import FlowResult
|
2019-12-09 10:27:16 +00:00
|
|
|
from homeassistant.helpers import aiohttp_client, config_entry_oauth2_flow
|
2019-10-29 06:47:31 +00:00
|
|
|
|
2020-08-29 05:59:24 +00:00
|
|
|
from .const import DOMAIN as ALMOND_DOMAIN, TYPE_LOCAL, TYPE_OAUTH2
|
2019-10-29 06:47:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def async_verify_local_connection(hass: core.HomeAssistant, host: str):
|
|
|
|
"""Verify that a local connection works."""
|
|
|
|
websession = aiohttp_client.async_get_clientsession(hass)
|
|
|
|
api = WebAlmondAPI(AlmondLocalAuth(host, websession))
|
|
|
|
|
|
|
|
try:
|
2021-11-04 15:07:50 +00:00
|
|
|
async with async_timeout.timeout(10):
|
2019-10-29 06:47:31 +00:00
|
|
|
await api.async_list_apps()
|
|
|
|
|
|
|
|
return True
|
|
|
|
except (asyncio.TimeoutError, ClientError):
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2020-08-29 05:59:24 +00:00
|
|
|
@config_entries.HANDLERS.register(ALMOND_DOMAIN)
|
2019-10-29 06:47:31 +00:00
|
|
|
class AlmondFlowHandler(config_entry_oauth2_flow.AbstractOAuth2FlowHandler):
|
|
|
|
"""Implementation of the Almond OAuth2 config flow."""
|
|
|
|
|
2020-08-29 05:59:24 +00:00
|
|
|
DOMAIN = ALMOND_DOMAIN
|
2019-10-29 06:47:31 +00:00
|
|
|
|
|
|
|
host = None
|
|
|
|
hassio_discovery = None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def logger(self) -> logging.Logger:
|
|
|
|
"""Return logger."""
|
|
|
|
return logging.getLogger(__name__)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def extra_authorize_data(self) -> dict:
|
|
|
|
"""Extra data that needs to be appended to the authorize url."""
|
|
|
|
return {"scope": "profile user-read user-read-results user-exec-command"}
|
|
|
|
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
|
|
"""Handle a flow start."""
|
|
|
|
# Only allow 1 instance.
|
|
|
|
if self._async_current_entries():
|
2020-10-06 12:12:12 +00:00
|
|
|
return self.async_abort(reason="single_instance_allowed")
|
2019-10-29 06:47:31 +00:00
|
|
|
|
|
|
|
return await super().async_step_user(user_input)
|
|
|
|
|
|
|
|
async def async_step_auth(self, user_input=None):
|
|
|
|
"""Handle authorize step."""
|
|
|
|
result = await super().async_step_auth(user_input)
|
|
|
|
|
|
|
|
if result["type"] == data_entry_flow.RESULT_TYPE_EXTERNAL_STEP:
|
|
|
|
self.host = str(URL(result["url"]).with_path("me"))
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
2021-08-23 07:35:03 +00:00
|
|
|
async def async_oauth_create_entry(self, data: dict) -> FlowResult:
|
2019-10-29 06:47:31 +00:00
|
|
|
"""Create an entry for the flow.
|
|
|
|
|
|
|
|
Ok to override if you want to fetch extra info or even add another step.
|
|
|
|
"""
|
|
|
|
data["type"] = TYPE_OAUTH2
|
|
|
|
data["host"] = self.host
|
|
|
|
return self.async_create_entry(title=self.flow_impl.name, data=data)
|
|
|
|
|
2021-08-23 07:35:03 +00:00
|
|
|
async def async_step_import(self, user_input: dict[str, Any]) -> FlowResult:
|
2019-10-29 06:47:31 +00:00
|
|
|
"""Import data."""
|
|
|
|
# Only allow 1 instance.
|
|
|
|
if self._async_current_entries():
|
2020-10-06 12:12:12 +00:00
|
|
|
return self.async_abort(reason="single_instance_allowed")
|
2019-10-29 06:47:31 +00:00
|
|
|
|
|
|
|
if not await async_verify_local_connection(self.hass, user_input["host"]):
|
|
|
|
self.logger.warning(
|
|
|
|
"Aborting import of Almond because we're unable to connect"
|
|
|
|
)
|
|
|
|
return self.async_abort(reason="cannot_connect")
|
|
|
|
|
|
|
|
return self.async_create_entry(
|
|
|
|
title="Configuration.yaml",
|
|
|
|
data={"type": TYPE_LOCAL, "host": user_input["host"]},
|
|
|
|
)
|
|
|
|
|
2021-12-03 13:05:56 +00:00
|
|
|
async def async_step_hassio(self, discovery_info: HassioServiceInfo) -> FlowResult:
|
2019-10-29 06:47:31 +00:00
|
|
|
"""Receive a Hass.io discovery."""
|
|
|
|
if self._async_current_entries():
|
2020-10-06 12:12:12 +00:00
|
|
|
return self.async_abort(reason="single_instance_allowed")
|
2019-10-29 06:47:31 +00:00
|
|
|
|
2021-12-03 13:05:56 +00:00
|
|
|
self.hassio_discovery = discovery_info.config
|
2019-10-29 06:47:31 +00:00
|
|
|
|
|
|
|
return await self.async_step_hassio_confirm()
|
|
|
|
|
|
|
|
async def async_step_hassio_confirm(self, user_input=None):
|
|
|
|
"""Confirm a Hass.io discovery."""
|
|
|
|
data = self.hassio_discovery
|
|
|
|
|
|
|
|
if user_input is not None:
|
|
|
|
return self.async_create_entry(
|
|
|
|
title=data["addon"],
|
|
|
|
data={
|
|
|
|
"is_hassio": True,
|
|
|
|
"type": TYPE_LOCAL,
|
|
|
|
"host": f"http://{data['host']}:{data['port']}",
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
return self.async_show_form(
|
|
|
|
step_id="hassio_confirm",
|
|
|
|
description_placeholders={"addon": data["addon"]},
|
|
|
|
data_schema=vol.Schema({}),
|
|
|
|
)
|